You've already forked FrameTour-RenderWorker
refactor(entity): 优化视频变速和缩放效果处理
-将视频变速实现从 minterpolate 改为使用 setpts,避免 PTS 冲突问题 -简化缩放效果处理逻辑
This commit is contained in:
@@ -271,39 +271,33 @@ class FfmpegTask(object):
|
|||||||
if param == '':
|
if param == '':
|
||||||
param = "1"
|
param = "1"
|
||||||
if param != "1":
|
if param != "1":
|
||||||
# 视频变速:使用minterpolate实现,避免PTS冲突
|
# 视频变速:使用fps实现,避免PTS冲突
|
||||||
effect_index += 1
|
effect_index += 1
|
||||||
speed_factor = 1.0 / float(param) # 慢放倍数转换为速度因子
|
filter_args.append(f"{video_output_str}setpts={param}*PTS[v_eff{effect_index}]")
|
||||||
filter_args.append(f"{video_output_str}minterpolate=fps=25*{speed_factor}:mi_mode=mci[v_eff{effect_index}]")
|
|
||||||
video_output_str = f"[v_eff{effect_index}]"
|
video_output_str = f"[v_eff{effect_index}]"
|
||||||
elif effect.startswith("zoom:"):
|
elif effect.startswith("zoom:"):
|
||||||
param = effect.split(":", 2)[1]
|
param = effect.split(":", 2)[1]
|
||||||
if param == '':
|
if param == '':
|
||||||
continue
|
continue
|
||||||
_split = param.split(",")
|
_split = param.split(",")
|
||||||
if len(_split) < 3:
|
if len(_split) < 2:
|
||||||
continue
|
continue
|
||||||
try:
|
try:
|
||||||
start_time = float(_split[0])
|
start_time = float(_split[0])
|
||||||
zoom_factor = float(_split[1])
|
zoom_factor = float(_split[1])
|
||||||
duration = float(_split[2])
|
|
||||||
if start_time < 0:
|
if start_time < 0:
|
||||||
start_time = 0
|
start_time = 0
|
||||||
if duration < 0:
|
|
||||||
duration = 0
|
|
||||||
if zoom_factor <= 0:
|
if zoom_factor <= 0:
|
||||||
zoom_factor = 1
|
zoom_factor = 1
|
||||||
except (ValueError, IndexError):
|
except (ValueError, IndexError):
|
||||||
start_time = 0
|
start_time = 0
|
||||||
duration = 0
|
|
||||||
zoom_factor = 1
|
zoom_factor = 1
|
||||||
if zoom_factor == 1:
|
if zoom_factor == 1:
|
||||||
continue
|
continue
|
||||||
effect_index += 1
|
effect_index += 1
|
||||||
|
|
||||||
# 获取缩放中心点(从pos_json或使用默认中心)
|
left_x = f"iw/(2*{zoom_factor})"
|
||||||
center_x = "iw/2"
|
top_y = f"ih/(2*{zoom_factor})"
|
||||||
center_y = "ih/2"
|
|
||||||
pos_json_str = self.ext_data.get('posJson', '{}')
|
pos_json_str = self.ext_data.get('posJson', '{}')
|
||||||
try:
|
try:
|
||||||
pos_json = json.loads(pos_json_str) if pos_json_str != '{}' else {}
|
pos_json = json.loads(pos_json_str) if pos_json_str != '{}' else {}
|
||||||
@@ -319,23 +313,14 @@ class FfmpegTask(object):
|
|||||||
center_x_ratio = (_f_x + _f_x2) / (2 * _v_w)
|
center_x_ratio = (_f_x + _f_x2) / (2 * _v_w)
|
||||||
center_y_ratio = (_f_y + _f_y2) / (2 * _v_h)
|
center_y_ratio = (_f_y + _f_y2) / (2 * _v_h)
|
||||||
# 转换为视频坐标系统
|
# 转换为视频坐标系统
|
||||||
center_x = f"iw*{center_x_ratio:.6f}"
|
left_x = f"iw*({center_x_ratio:.6f}-1/(2*{zoom_factor}))"
|
||||||
center_y = f"ih*{center_y_ratio:.6f}"
|
top_y = f"ih*({center_y_ratio:.6f}-1/(2*{zoom_factor}))"
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
# 解析失败使用默认中心
|
# 解析失败使用默认中心
|
||||||
pass
|
pass
|
||||||
|
|
||||||
if duration == 0:
|
# 静态缩放(整个视频时长)
|
||||||
# 静态缩放(整个视频时长)
|
filter_args.append(f"{video_output_str}scale={zoom_factor}*iw:{zoom_factor}*ih,crop=iw/{zoom_factor}:ih/{zoom_factor}:{left_x}:{top_y}[v_eff{effect_index}]")
|
||||||
x_expr = f"({center_x})-(ow*zoom)/2"
|
|
||||||
y_expr = f"({center_y})-(oh*zoom)/2"
|
|
||||||
filter_args.append(f"{video_output_str}trim=start={start_time},zoompan=z={zoom_factor}:x={x_expr}:y={y_expr}:d=1[v_eff{effect_index}]")
|
|
||||||
else:
|
|
||||||
# 动态缩放(指定时间段内)
|
|
||||||
zoom_expr = f"if(between(t\\,{start_time}\\,{start_time + duration})\\,{zoom_factor}\\,1)"
|
|
||||||
x_expr = f"({center_x})-(ow*zoom)/2"
|
|
||||||
y_expr = f"({center_y})-(oh*zoom)/2"
|
|
||||||
filter_args.append(f"{video_output_str}zoompan=z={zoom_expr}:x={x_expr}:y={y_expr}:d=1[v_eff{effect_index}]")
|
|
||||||
video_output_str = f"[v_eff{effect_index}]"
|
video_output_str = f"[v_eff{effect_index}]"
|
||||||
elif effect.startswith("skip:"):
|
elif effect.startswith("skip:"):
|
||||||
param = effect.split(":", 2)[1]
|
param = effect.split(":", 2)[1]
|
||||||
|
Reference in New Issue
Block a user