python-ffmpeg转码视频(添加视频合并)

查看 186|回复 10
作者:qianaonan   
视频转码1.1帖子链接:新版调用ffmpeg视频转码器:https://www.52pojie.cn/thread-1901828-1-1.html
更新:
这版视频转码器添加功能合并视频功能,支持最多三个视频合并(一个视频就别试了),合并过程中会日志输出到log.txt中,而转码视频就没有添加该代码会有cmd弹窗,懂python的可以自行添加,同目录中会生成concat.txt文件是视频合并顺序;
注:

1、三个视频顺序由三个按钮从左到右顺序进行合并(不要重叠在一起,尽量保持一个水平线上),两个视频就是按钮视频1和2,按钮下面的文本框没啥用,只是用来看选中的那些文件名。


[color=]★三个视频的要求分辨率一致,不一致会报错。

2、这次代码有可能会在合并部分有点乱,自行理解。
3、想调用显卡进行加速,奈何python读取本地配置这部分代码是真的差,我的显示器输入线插在RX6600xt上就检查不出来(不知道啥原因),而且也没有N卡进行调试,后期如果添加显卡加速代码,以a卡为主。
4、视频越大速度越慢,不是程序卡死不动,而是视频太大转码太慢,python认为是卡死了,如果添加防卡死代码,怕有人以为是秒转就关闭程序。(自己曾用4K转码程序卡死,用任务管理器终止程序后,文件目录生成了一段转码后的视频文件)

5、点击视频合并自动开始选择文件,点击合并也保存到同目录里,也就说上面的选择文件和保存目录跟视频合并没有关系。

[color=]win1164位打包有可能32位的无法使用
[color=]转载需注明来源与作者!
转载需注明来源与作者!
转载需注明来源与作者!
[color=]重要的事情说三遍!
[color=]放代码:
[Python] 纯文本查看 复制代码# -*- coding:utf-8 -*-
import wx
import subprocess
import os
import re
import sys
def get_path(relative_path):
    try:
        base_path = sys._MEIPASS
    except AttributeError:
        base_path = os.path.abspath(".")
    return os.path.normpath(os.path.join(base_path, relative_path))
ffmpeg_path =get_path("assets/ffmpeg.exe")#资源的路径
#弹窗消息
def show_message(message):
    """
    显示消息框。
    参数:
    message (str): 要显示的消息。
    """
    app = wx.App(False)
    dlg = wx.MessageDialog(None, message, "提示", wx.OK | wx.ICON_INFORMATION)
    dlg.ShowModal()
    dlg.Destroy()
    app.MainLoop()
#转mp4
def convert_to_mp4(input_video_path, output_directory):
    """
    将输入的视频文件转换为 MP4 格式。
    参数:
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    """
    video_name = os.path.splitext(os.path.basename(input_video_path))[0]
    output_mp4_path = os.path.join(output_directory, f"{video_name}.mp4")
    if os.path.exists(output_mp4_path):
        show_message(f"目录中存在同名文件 '{output_mp4_path}',请删除或重命名后重试。")
        return
    result=subprocess.run([ffmpeg_path, "-i",
                           input_video_path,
                           "-c:v", "libx264",
                           "-preset", "slow",
                           "-crf", "23",
                           "-c:a", "aac",
                           "-b:a", "192k",
                           "-strict",
                           "experimental", output_mp4_path])
    if result.returncode == 0:
        show_message("转码成功!")
    else:
        show_message(f"转码失败!错误代码:{result.returncode}")
#转mp3
def convert_to_mp3(input_video_path, output_directory):
    """
    将输入的视频文件转换为 MP3 格式。
    参数:
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    """
    video_name = os.path.splitext(os.path.basename(input_video_path))[0]
    output_mp3_path = os.path.join(output_directory, f"{video_name}.mp3")
    if os.path.exists(output_mp3_path):
        show_message(f"目录中存在同名文件 '{output_mp3_path}',请删除或重命名后重试。")
        return
    result=subprocess.run([ffmpeg_path, "-i", input_video_path, "-vn", "-acodec", "libmp3lame", output_mp3_path])
    if result.returncode == 0:
        show_message("转码成功!")
    else:
        show_message(f"转码失败!错误代码:{result.returncode}")
#转avi
def convert_to_avi(input_video_path, output_directory):
    """
    将输入的视频文件转换为 AVI 格式。
    参数:
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    """
    video_name = os.path.splitext(os.path.basename(input_video_path))[0]
    output_avi_path = os.path.join(output_directory, f"{video_name}.avi")
    if os.path.exists(output_avi_path):
        show_message(f"目录中存在同名文件 '{output_avi_path}',请删除或重命名后重试。")
        return
    result=subprocess.run([ffmpeg_path, "-i", input_video_path, "-c:v", "libx264", "-preset", "slow", "-crf", "23", "-c:a", "pcm_s16le", output_avi_path])
    if result.returncode == 0:
        show_message("转码成功!")
    else:
        show_message(f"转码失败!错误代码:{result.returncode}")
#转wmv
def convert_to_wmv(input_video_path, output_directory):
    """
    将输入的视频文件转换为 WMV 格式。
    参数:
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    """
    video_name = os.path.splitext(os.path.basename(input_video_path))[0]
    output_wmv_path = os.path.join(output_directory, f"{video_name}.wmv")
    if os.path.exists(output_wmv_path):
        show_message(f"目录中存在同名文件 '{output_wmv_path}',请删除或重命名后重试。")
        return
    result=subprocess.run([ffmpeg_path, "-i", input_video_path, "-c:v", "wmv2", "-b:v", "1024k", "-c:a", "wmav2", output_wmv_path])
    if result.returncode == 0:
        show_message("转码成功!")
    else:
        show_message(f"转码失败!错误代码:{result.returncode}")
#转gif
def convert_to_gif(input_video_path, output_directory, start_time, end_time, fps=10):
    """
    将输入的视频文件转换为 GIF 格式。
    参数:
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    start_time (int): 视频开始时间(以秒为单位)。
    end_time (int): 视频结束时间(以秒为单位)。
    fps (int): GIF 的帧率,默认为 10。
    """
    video_name = os.path.splitext(os.path.basename(input_video_path))[0]
    output_gif_path = os.path.join(output_directory, f"{video_name}_segment.gif")
    if os.path.exists(output_gif_path):
        show_message(f"目录中存在同名文件 '{output_gif_path}',请删除或重命名后重试。")
        return
    result=subprocess.run([ffmpeg_path, "-ss", str(start_time), "-i", input_video_path, "-t", str(end_time - start_time), "-vf", f"fps={fps}", output_gif_path])
    if result.returncode == 0:
        show_message("转码成功!")
    else:
        show_message(f"转码失败!错误代码:{result.returncode}")
#转mov
def convert_to_mov(input_video_path, output_directory):
    """
    将输入的视频文件转换为 MOV 格式。
    参数:
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    """
    video_name = os.path.splitext(os.path.basename(input_video_path))[0]
    output_mov_path = os.path.join(output_directory, f"{video_name}.mov")
    if os.path.exists(output_mov_path):
        show_message(f"目录中存在同名文件 '{output_mov_path}',请删除或重命名后重试。")
        return
    result=subprocess.run([ffmpeg_path, "-i", input_video_path, "-c:v", "libx264", "-preset", "slow", "-crf", "23", output_mov_path])
    if result.returncode == 0:
        show_message("转码成功!")
    else:
        show_message(f"转码失败!错误代码:{result.returncode}")
# 函数集合
def function_dispatcher(number,input_video_path,output_directory,start_time,end_time):
    """
    根据用户选择的数字执行相应的视频转换函数。
    参数:
    number (int): 用户选择的数字,代表要执行的转换函数。
    input_video_path (str): 输入视频文件的路径。
    output_directory (str): 输出目录的路径。
    start_time (int): 视频开始时间(以秒为单位)。
    end_time (int): 视频结束时间(以秒为单位)。
    """
    functions = {
        0: convert_to_mp4,
        1: convert_to_avi,
        2: convert_to_wmv,
        3: convert_to_mp3,
        4: convert_to_gif,
        5: convert_to_mov,
    }
    if number in functions:
        if number == 4:  # 如果选中的是转换为 GIF
            if start_time is not None and end_time is not None:
                functions[number](input_video_path, output_directory, start_time, end_time)
            else:
                # 在这种情况下,可以选择不传递 start_time 和 end_time 参数,或者传递默认值
                functions[number](input_video_path, output_directory, 0, None)
        else:
            functions[number](input_video_path, output_directory)
#gif所需视频长度格式
def convert_time_to_seconds(time_str):
    """
    将时间字符串转换为秒数。
    参数:
    time_str (str): 表示时间的字符串,可以是 HH:MM:SS 格式或整数表示的秒数。
    返回:
    int: 转换后的秒数。
    """
    if isinstance(time_str, int) or time_str.isdigit():
        # 如果输入是整数或者全是数字,则直接解析为整数,代表秒数
        return int(time_str)
    elif ':' in time_str:
        # 如果输入包含冒号,则尝试将其解析为时间格式
        parts = time_str.split(':')
        if len(parts) == 3:
            try:
                hours = int(parts[0])
                minutes = int(parts[1])
                seconds = int(parts[2])
                # 计算总秒数
                return hours * 3600 + minutes * 60 + seconds
            except ValueError:
                # 如果解析失败,则说明时间格式不正确
                raise ValueError("Invalid time format")
        else:
            # 如果冒号数量不是三个,则抛出异常
            raise ValueError("Invalid time format")
    else:
        # 如果既不是整数,也不是带冒号的时间格式,则抛出异常
        raise ValueError("Invalid time format")
#转换时间字符串
def format_duration(duration):
    """
    格式化视频持续时间为 HH:MM:SS 格式的字符串。
    参数:
    duration (float): 视频的持续时间(以秒为单位)。
    返回:
    str: 格式化后的时间字符串。
    """
    hours = int(duration // 3600)
    minutes = int((duration % 3600) // 60)
    seconds = int(duration % 60)
    return f'{hours:02d}:{minutes:02d}:{seconds:02d}'
#获取视频长度
def chixu_video(input_video_path):
    """
    获取视频文件的持续时间,并将其格式化为 HH:MM:SS 的字符串。
    参数:
    input_video_path (str): 视频文件的路径。
    返回:
    str: 格式化后的视频持续时间字符串。
    """
    if not input_video_path:
        return "00:00:00"
    # 运行 FFmpeg 命令获取视频时长信息
    command = [ffmpeg_path, '-i', input_video_path]
    result = subprocess.run(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, universal_newlines=True)
    output = result.stderr
    # 使用正则表达式从输出中提取视频时长信息
    duration_match = re.search(r'Duration:\s*(\d+):(\d+):(\d+)', output)
    if duration_match:
        hours = int(duration_match.group(1))
        minutes = int(duration_match.group(2))
        seconds = int(duration_match.group(3))
        formatted_duration = f"{hours:02d}:{minutes:02d}:{seconds:02d}"
        return formatted_duration
    else:
        return "00:00:00"
#视频合并
def merge_videos(video1_path, video2_path, video3_path, output_dir=None):
    # 如果未提供输出目录,则默认保存在视频文件所在目录
    if output_dir is None:
        output_dir = os.path.dirname(video1_path)
    # 输出文件路径
    output_path = os.path.join(output_dir, '合并视频.mp4')
    # 检查输出目录是否已存在合并视频文件
    if os.path.exists(output_path):
        show_message(f"合并视频文件 '{output_path}' 已存在,无法执行合并操作。")
        return
    # 生成concat.txt文件,包含要合并的视频文件路径
    concat_file = os.path.join(output_dir, 'concat.txt')
    with open(concat_file, 'w') as f:
        for path in [video1_path, video2_path, video3_path]:
            f.write(f"file '{path}'\n")
    # 使用FFmpeg执行合并命令
    log_file = os.path.join(output_dir, 'log.txt')
    with open(log_file, 'w') as log:
        command = [ffmpeg_path, '-n', '-f', 'concat', '-safe', '0', '-i', concat_file, '-c', 'copy', output_path]
        result = subprocess.run(command, stdout=log, stderr=subprocess.STDOUT)
    # 根据FFmpeg的返回代码显示消息
    if result.returncode == 0:
        show_message("合并成功!")
    else:
        show_message(f"合并失败!错误代码:{result.returncode},ffmpeg输出日志在同目录log.txt文件中")
    # 删除临时生成的concat.txt文件
    #-------------------------UI界面----------------------------
class DragButton(wx.Button):
    def __init__(self, parent, id=wx.ID_ANY, label="", pos=wx.DefaultPosition, size=wx.DefaultSize, style=0):
        super().__init__(parent, id, label=label, pos=pos, size=size, style=style)
        self.Bind(wx.EVT_LEFT_DOWN, self.OnLeftDown)
        self.Bind(wx.EVT_LEFT_UP, self.OnLeftUp)
        self.Bind(wx.EVT_MOTION, self.OnMotion)
        self.dragging = False
    def OnLeftDown(self, event):
        self.CaptureMouse()
        self.dragStartPos = event.GetPosition()
        self.dragging = True
        self.Raise()  # 当按钮被点击时,将其置于最顶层
    def OnLeftUp(self, event):
        if self.HasCapture():
            self.ReleaseMouse()
        self.dragging = False
    def OnMotion(self, event):
        if event.Dragging() and event.LeftIsDown() and self.dragging:
            newPos = event.GetPosition() - self.dragStartPos + self.GetPosition()
            self.Move(newPos)
class Frame(wx.Frame):
    def __init__(self):
        wx.Frame.__init__(self, None, title='原创:吾爱qianaonan', size=(437, 255), name='frame', style=541072384)
        self.启动窗口 = wx.Panel(self)
        self.Centre()
        self.编辑框1 = wx.TextCtrl(self.启动窗口, size=(293, 29), pos=(8, 5), value='', name='text', style=16)
        self.按钮1 = wx.Button(self.启动窗口, size=(80, 32), pos=(315, 3), label='选择文件', name='button')
        self.标签1 = wx.StaticText(self.启动窗口, size=(80, 24), pos=(10, 44), label='转换成', name='staticText',
                                   style=2321)
        self.单选框2 = wx.RadioButton(self.启动窗口, size=(58, 24), pos=(8, 72), name='radioButton', label='MP4')
        self.单选框3 = wx.RadioButton(self.启动窗口, size=(40, 24), pos=(73, 72), name='radioButton', label='AVI')
        self.单选框4 = wx.RadioButton(self.启动窗口, size=(58, 24), pos=(136, 72), name='radioButton', label='WMV')
        self.单选框5 = wx.RadioButton(self.启动窗口, size=(61, 24), pos=(209, 72), name='radioButton', label='MP3')
        self.单选框6 = wx.RadioButton(self.启动窗口, size=(42, 24), pos=(275, 72), name='radioButton', label='GIF')
        self.单选框7 = wx.RadioButton(self.启动窗口, size=(80, 24), pos=(327, 72), name='radioButton', label='MOV')
        self.按钮2 = wx.Button(self.启动窗口, size=(80, 32), pos=(317, 107), label='保存目录', name='button')
        self.编辑框2 = wx.TextCtrl(self.启动窗口, size=(293, 29), pos=(8, 107), value='', name='text', style=16)
        self.按钮3 = wx.Button(self.启动窗口, size=(80, 32), pos=(240, 157), label='开始转换', name='button')
        self.Bind(wx.EVT_BUTTON, self.OnSelectFile, self.按钮1)  # 绑定选择文件按钮的事件
        self.Bind(wx.EVT_BUTTON, self.OnSelectFolder, self.按钮2)
        self.按钮3.Bind(wx.EVT_BUTTON, self.按钮3_按钮被单击)
        self.标签2 = wx.StaticText(self.启动窗口, size=(195, 60), pos=(15, 143),
                                   label='请谨慎使用gif转换,一般是视频过\n长会导致程序卡死\n视频越长花费时间越长',
                                   name='staticText', style=17)
        self.标签2.SetForegroundColour((255, 0, 0, 255))
        self.标签3 = wx.StaticText(self.启动窗口, size=(80, 17), pos=(254, 56), label='10帧/秒', name='staticText',
                                   style=2321)
        self.编辑框3 = wx.TextCtrl(self.启动窗口, size=(80, 18), pos=(303, 35), value='', name='text', style=0)
        self.编辑框3.SetForegroundColour((128, 128, 128, 255))
        self.编辑框4 = wx.TextCtrl(self.启动窗口, size=(80, 18), pos=(304, 54), value='', name='text', style=0)
        self.编辑框4.SetForegroundColour((128, 128, 128, 255))
        self.编辑框4.Bind(wx.EVT_KILL_FOCUS, self.编辑框4_失去焦点)
        self.编辑框4.Bind(wx.EVT_SET_FOCUS, self.onSetFocus)
        self.标签5 = wx.StaticText(self.启动窗口, size=(26, 17), pos=(269, 38), label='开始', name='staticText',
                                   style=2321)
        self.标签6 = wx.StaticText(self.启动窗口, size=(28, 14), pos=(268, 56), label='结束', name='staticText',
                                   style=2321)
        self.按钮4 = wx.Button(self.启动窗口, size=(80, 32), pos=(330, 157), label='视频合并', name='button')
        self.按钮4.Bind(wx.EVT_BUTTON, self.按钮4_按钮被单击)
        self.编辑框3.Hide()
        self.编辑框4.Hide()
        self.标签5.Hide()
        self.标签6.Hide()
        self.单选框2.Bind(wx.EVT_RADIOBUTTON, self.onRadioButton)
        self.单选框3.Bind(wx.EVT_RADIOBUTTON, self.onRadioButton)
        self.单选框4.Bind(wx.EVT_RADIOBUTTON, self.onRadioButton)
        self.单选框5.Bind(wx.EVT_RADIOBUTTON, self.onRadioButton)
        self.单选框6.Bind(wx.EVT_RADIOBUTTON, self.onRadioButton)
        self.单选框7.Bind(wx.EVT_RADIOBUTTON, self.onRadioButton)
        #-----------------------------------------视频合并页面-------------------------------------------
        self.编辑框5 = wx.TextCtrl(self.启动窗口,size=(293, 29),pos=(8, 294),value='',name='text',style=0)
        self.按钮8 = wx.Button(self.启动窗口,size=(80, 32),pos=(318, 292),label='合成',name='button')
        self.按钮5 = DragButton(self.启动窗口, size=(80, 32), pos=(37, 225), label='视频1')
        self.按钮6 = DragButton(self.启动窗口, size=(80, 32), pos=(144, 225), label='视频2')
        self.按钮7 = DragButton(self.启动窗口, size=(80, 32), pos=(257, 225), label='视频3')
        self.按钮8.Bind(wx.EVT_BUTTON, self.OnButton4Click)
        self.标签7 = wx.StaticText(self.启动窗口, size=(401, 16), pos=(9, 272),
                                   label='以上按钮调整视频顺序,下面文本尽看选中文件的文件名',
                                   name='staticText', style=2321)
        self.标签7.SetForegroundColour((255, 0, 0, 255))
    def InitUI(self):
        # 初始化 UI
        # 添加按钮、编辑框等控件
        self.list_path = []  # 将 list_path 定义为类的成员变量
    def 按钮4_按钮被单击(self, event):
        self.SetSize(437, 371)
        wildcard = "Video Files (*.mp4;*.avi;*.wmv;*.mov)|*.mp4;*.avi;*.wmv;*.mov|All Files (*.*)|*.*"  # 文件类型过滤器
        dialog = wx.FileDialog(self, "选择文件", wildcard=wildcard,
                               style=wx.FD_OPEN | wx.FD_MULTIPLE)
        # dialog.SetMaxFiles(3)
        if dialog.ShowModal() != wx.ID_OK:
            dialog.Destroy()
            return
        paths = dialog.GetPaths()  # 获取所有选中文件的路径列表
        dialog.Destroy()
        if len(paths) >= 4:  # 假设我们只允许最多5个文件被选中
            wx.MessageBox("你只能选中3个视频", "Error")
            paths = paths[:3]  # 如果选择的文件超过3个,只保留前三个文件
        if len(paths) == 2:
            self.按钮7.Hide()
        if len(paths) == 3:
            self.按钮7.Show()
        # 处理文件,例如打开文件、读取内容等
        filenames = [os.path.basename(path) for path in paths]  # 提取文件名列表
        filenames_text = "\n".join([filename + "|" for filename in filenames])# 将文件名列表连接为单个字符串,用换行符分隔
        self.编辑框5.SetValue(filenames_text)  # 将连接后的文件名字符串设置为编辑框的值
        self.list_path = list(paths)
    def OnButton4Click(self, event):
        # 获取按钮5、按钮6和按钮7的位置信息
        pos5 = self.按钮5.GetPosition()
        pos6 = self.按钮6.GetPosition()
        pos7 = self.按钮7.GetPosition()
        #print(self.list_path[0],self.list_path[1],self.list_path[2])
        # 检查按钮5、6、7的 x 坐标值,以确定它们的从左到右位置顺序
        a = self.list_path[0]
        b = self.list_path[1]
        if len(self.list_path)>=3:
            c = self.list_path[2]
        else:
            c=None
        #print(self.list_path)
        if len(self.list_path)==3:
            if pos5.x
老规矩百度云链接:链接:https://pan.baidu.com/s/11g01x-vXlqIH-5rueezlGQ?pwd=0bdo 蓝奏云链接:https://wwm.lanzout.com/i1YKR1u3rm9i

按钮, 编辑

XTCharles   


wasm2023 发表于 2024-4-6 20:17
我是想用py判断,并且能过滤坏掉的视频

调用 ffprobe 查询媒体文件信息,报错就是坏的。
可以顺便捕捉并解析 stdout/stderr 的文本输出,得到媒体文件的宽度高度
inthelove95   

有点东西
ZCCHNS   

哇 感谢分享!
centenary   

谢谢楼主 但是我没办法使用
wasm2023   

感谢分享 辛苦了
qianaonan
OP
  

楼主,请问判断一个文件夹的视频是否为同一分辨率咋写呢
wasm2023   


wasm2023 发表于 2024-4-6 18:40
楼主,请问判断一个文件夹的视频是否为同一分辨率咋写呢

win11文件属性里面有啊
qianaonan
OP
  


qianaonan 发表于 2024-4-6 20:02
win11文件属性里面有啊

我是想用py判断,并且能过滤坏掉的视频
starryskyhello   


wasm2023 发表于 2024-4-6 20:17
我是想用py判断,并且能过滤坏掉的视频

放大图看,绝大部分坏掉的视频是无法显示缩略图
您需要登录后才可以回帖 登录 | 立即注册

返回顶部