# 格式转化文件,包含多种文件的互相转化,主要与视频相关
from pathlib import Path
import subprocess
import random
import os
import reclass Utils(object):@staticmethoddef get_decimal_part(x: float) -> float:s = format(x, '.15f')  # 格式化为15位小数字符串if '.' in s:_, decimal_part = s.split('.')decimal_str = decimal_part.rstrip('0')  # 移除末尾多余的0if not decimal_str:  # 如果小数部分全为0return 0.0result = float("0." + decimal_str)return -result if x < 0 else result  # 处理负数return 0.0@staticmethoddef file_path_processor(*file_paths: str) -> str | list[str]:if len(file_paths) == 1:return file_paths[0].replace("\\", "\\\\")return [file_path.replace("\\", "\\\\") for file_path in file_paths]class TimeConverter(object):def time_analysis(self, time_str, offset: float = 0.0) -> tuple[int, int, int, int]:pattern = r'^(\d{1,2})\D(\d{1,2})\D(\d{1,2})[.,](\d{1,3})$'match: re.Match = re.match(pattern, time_str)if not match:raise ValueError(f"无法解析的时间格式: {time_str}")results = list(match.groups())results[-1] = f"{results[-1]:0<3}"hours, minutes, seconds, milliseconds = map(int, results)if offset != 0:milliseconds += int(Utils.get_decimal_part(offset) * 1000)offset = int(offset)hours += (offset // 3600)minutes += (offset // 60)seconds += (offset % 60)# 注意进位seconds += milliseconds // 1000minutes += seconds // 60hours += minutes // 60seconds %= 60milliseconds %= 1000minutes %= 60return hours, minutes, seconds, millisecondsdef format_number(self, input_str: str, length: int) -> str:if not input_str.isdigit():raise ValueError("输入必须是数字字符串")if not isinstance(length, int) or length <= 0:raise ValueError("长度必须是正整数")if len(input_str) > length:return input_str[:length]  # 截断超长部分else:return '0' * (length - len(input_str))  + input_strdef format_copy(self, ref_time):match: re.Match = re.match("^(\d+)(\D)(\d+)(\D)(\d+)(\D)(\d+)$", ref_time)if not match:raise ValueError(f"无法复制的时间格式: {ref_time}")(h_str, sep1, m_str, sep2, s_str, sep3, ms_str) = match.groups()h_len, m_len, s_len, ms_len = map(len, (h_str, m_str, s_str, ms_str))def time_formater(hours: int, minutes: int, seconds: int, milliseconds: int) -> str:return (f"{self.format_number(str(hours), h_len)}{sep1}"f"{self.format_number(str(minutes), m_len)}{sep2}"f"{self.format_number(str(seconds), s_len)}{sep3}"f"{self.format_number(str(milliseconds), ms_len)}")return time_formaterclass SubtitleConverter(object):def ass_analyser(self, ass_file) -> list[dict]:ass_object = []with open(ass_file, "r", encoding="utf-8") as f:start_record = Falserecord_fields = Falsefor i in f:if i.strip() == "[Events]":record_fields = Truecontinueif re.match("\[.*?\]$", i.strip()):start_record = Falsecontinueif record_fields:record_fields = Falsestart_record = Truefields = [j.strip() for j in i.split(",")]continueif start_record:row = map(lambda j: j.strip(), i.split(",", maxsplit=len(fields) - 1))ass_object.append({k: v for k, v in zip(fields, row)})return ass_objectdef ass2srt(self, ass_file: str, srt_file: str, offset: float) -> None:if not ass_file.endswith(".ass"):returntime_analyser = TimeConverter()formatter = time_analyser.format_copy("00:00:00,000")srt_file_obj = open(srt_file, "w", encoding="utf-8")count = 1for row in self.ass_analyser(ass_file):if not re.match("标准", row["Style"]):continuestart_time = formatter(*time_analyser.time_analysis(row["Start"], offset))end_time = formatter(*time_analyser.time_analysis(row["End"], offset))subtitle = row["Text"]srt_file_obj.write(f"{count}\n{start_time} --> {end_time}\n{subtitle}\n\n")count += 1srt_file_obj.close()def ass2ass_eng(self, ass_file: str, output_file: str, remove_style='标准-Chi') -> None:"""这个方法是纯AI写的,与ass_analyser脱节。其作用是处理ass文件中的样式,去除其中的描边设置,同时移除掉指定字幕。字幕文件下载自网站:http://154.17.3.217:8888/sub/new, 原生字幕可能中英夹杂,但有时我们出于学习目的,可能只想要英文字幕;或纯粹出于娱乐目的,只想保留中文字幕等;"""with open(ass_file, 'r', encoding='utf-8-sig') as f:lines = f.readlines()output = []current_section = Nonefor line in lines:line = line.rstrip('\r\n')stripped = line.strip()if stripped.startswith('[') and stripped.endswith(']'):current_section = strippedoutput.append(line)continueif current_section == '[V4+ Styles]':if line.startswith('Style:'):parts = line[len('Style:'):].split(',')if len(parts) >= 17:parts[16] = '0'output.append('Style:' + ','.join(parts))else:output.append(line)else:output.append(line)elif current_section == '[Events]' and line.startswith('Dialogue:'):content = line[len('Dialogue:'):].strip()fields = content.split(',', 9)if len(fields) >= 4 and fields[3].strip() == remove_style:continueoutput.append(line)else:output.append(line)with open(output_file, 'w', encoding='utf-8') as f:f.write('\n'.join(output))def ass2ass_offset(self, ass_file1, ass_file2, offset: float) -> None:time_analyser = TimeConverter()ref_time = "0:00:00.00"time_formatter = time_analyser.format_copy(ref_time)ass_object = self.ass_analyser(ass_file1)for row in ass_object:offset_start = time_analyser.time_analysis(row["Start"], offset)offset_end = time_analyser.time_analysis(row["End"], offset)row["Start"] = time_formatter(*offset_start)row["End"] = time_formatter(*offset_end)fields = ",".join(ass_object[0].keys())rows = "\n".join(",".join(row[k] for k in ass_object[0]) for row in ass_object)rows = rows.replace("\\", "\\\\")new_body = f"[Events]\n{fields}\n{rows}"pattern = r'\[Events\]\nFormat:.*(?:\nDialogue:.*)*'with open(ass_file1, "r", encoding="utf-8") as f:content = f.read()new_content = re.sub(pattern, new_body, content, re.MULTILINE)with open(ass_file2, "w", encoding="utf-8") as f:f.write(new_content)def srt2srt_offset(self, srt_file1, srt_file2, offset: float) -> None:time_analyser = TimeConverter()ref_time = "00:00:00,000"time_formatter = time_analyser.format_copy(ref_time)with open(srt_file1, "r", encoding="utf-8") as f1:f2 = open(srt_file2, "w", encoding="utf-8")for i in f1:res = re.match("^(.*?) --> (.*)$", i.strip())if res:start_time, end_time = res.groups()offset_start = time_analyser.time_analysis(start_time, offset)offset_end = time_analyser.time_analysis(end_time, offset)start_time = time_formatter(*offset_start)end_time = time_formatter(*offset_end)i = f"{start_time} --> {end_time}\n"f2.write(i)f2.close()        class VedioConverter(object):def mkv2mp4(self, mkv_file: str, mp4_file: str) -> None:mkv_file, mp4_file = Utils.file_path_processor(mkv_file, mp4_file)command = [ffmpeg_file_path,'-i', Utils.file_path_processor(mkv_file),'-c:v', 'copy','-c:a', 'copy','-y',Utils.file_path_processor(mp4_file)]subprocess.run(command, check=True)def ass_embed_mp4(self, mp4_file: str, ass_file: str, output_file: str, itsoffset: float = 0.0) -> None:mp4_file, ass_file, output_file = Utils.file_path_processor(mp4_file, ass_file, output_file)converter = SubtitleConverter()random_name = f"{random.randint(0, 1000000)}.ass"converter.ass2ass_offset(ass_file, random_name, itsoffset)command = [ffmpeg_file_path,'-i', mp4_file,'-vf', f"subtitles={random_name}",'-c:v', 'libx264','-profile:v', 'high','-pix_fmt', 'yuv420p','-preset', 'fast','-b:a', '192k','-c:a', 'aac','-movflags', '+faststart','-y',output_file]subprocess.run(command, check=True)os.remove(random_name)def ass_embed_mkv(self, mkv_file: str, ass_file: str, output_file: str, itsoffset) -> None:mkv_file, ass_file, output_file = Utils.file_path_processor(mkv_file, ass_file, output_file)command = [ffmpeg_file_path,'-i', mkv_file,'-itsoffset', str(itsoffset),'-i', ass_file,'-map', '0','-map', '-0:s','-map', '1','-c', 'copy','-metadata:s:s:0', 'language=eng','-y',output_file]subprocess.run(command, check=True)ffmpeg_file_path = r"ffmpeg.exe"
vedio_converter = VedioConverter()
# 加入硬字幕(mp4)
vedio_converter.ass_embed_mp4(r"your_input_mp4_file.mp4",r"your_ass_subtitle_file.ass",r"your_output_mp4_file.mp4",itsoffset=6.3    # 指定的字幕偏移时间,让字幕对齐音频
)

为了方便字幕的视频嵌入,上述代码实现了一些较为重要的功能,部分如下:

ass文件转srt文件,见SubtitleConverter的ass2srt方法;

根据srt文件和ass文件的时间偏移指定时间生成新的srt文件和ass文件的方法,见SubtitleConverter的ass2ass_offset和srt2srt_offset

ass字幕文件嵌入mp4视频的方法(可指定时间偏移),见VedioConverter的ass_embed_mp4方法,注意是硬字幕嵌入,相对耗时。

这些方法的实现都不是很难,不过并不能保证没有Bug;经过简单而基本测试,暂时没有出现问题

在使用之前,请确保找到你的ffmpeg路径。如果已经添加到环境变量,可以直接使用;否则请自行修改里面ffmpeg的路径为绝对路径。此处提供ffmpeg的下载路径:

FFmpeg 最新 Windows 64 位 GPL 版本下载

关于为什么mp4视频嵌入硬字幕,而mkv视频却是嵌入软字幕(在代码中),其实是有原因的:

mp4被更加广泛的兼容,几乎所有视频播放器都可以正确播放。但如果是嵌入软字幕,mp4视频的字幕则不一定能被播放器支持。硬字幕嵌入能保证字幕一定显示,更凸显mp4兼容的优势。缺陷就是硬字幕不能选择隐藏,同时需要重新编码,比较耗时;

mkv视频的支持性就明显要差不少,如手机上很多视频播放器就对mkv视频的支持不好,或视频变形,或音频丢失;但是软字幕的显示往往不是问题,这得益于mkv视频是字幕的天然容器,所以只要能找到合适的mkv视频播放器,几乎就能正常的显示软字幕。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/85746.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/85746.shtml
英文地址,请注明出处:http://en.pswp.cn/diannao/85746.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

05 APP 自动化- Appium 单点触控 多点触控

文章目录 一、单点触控查看指针的指针位置实现手势密码&#xff1a; 二、多点触控 一、单点触控 查看指针的指针位置 方便查看手势密码-九宫格每个点的坐标 实现手势密码&#xff1a; 执行手势操作&#xff1a; 按压起点 -> 移动到下一点 -> 依次移动 -> 释放&am…

【软件】在 macOS 上安装 MySQL

在 macOS 上安装 MySQL 有多种方法&#xff0c;以下是两种常见的安装方式&#xff1a;通过 Homebrew 安装和通过安装包安装。以下是详细的步骤&#xff1a; 一、通过 Homebrew 安装 MySQL Homebrew 是 macOS 的包管理器&#xff0c;使用它安装 MySQL 非常方便。 1.安装 Home…

第11节 Node.js 模块系统

为了让Node.js的文件可以相互调用&#xff0c;Node.js提供了一个简单的模块系统。 模块是Node.js 应用程序的基本组成部分&#xff0c;文件和模块是一一对应的。换言之&#xff0c;一个 Node.js 文件就是一个模块&#xff0c;这个文件可能是JavaScript 代码、JSON 或者编译过的…

力扣热题100之二叉树的直径

题目 给你一棵二叉树的根节点&#xff0c;返回该树的 直径 。 二叉树的 直径 是指树中任意两个节点之间最长路径的 长度 。这条路径可能经过也可能不经过根节点 root 。 两节点之间路径的 长度 由它们之间边数表示。 代码 方法&#xff1a;递归 计算二叉树的直径可以理解…

OpenCV CUDA模块图像处理------创建CUDA加速的Canny边缘检测器对象createCannyEdgeDetector()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 该函数用于创建一个 CUDA 加速的 Canny 边缘检测器对象&#xff08;CannyEdgeDetector&#xff09;&#xff0c;可以在 GPU 上高效执行 Canny 边…

unix/linux,sudo,其内部结构机制

我们现在深入sudo的“引擎室”,探究其内部的结构和运作机制。这就像我们从观察行星运动,到深入研究万有引力定律的数学表达和物理内涵一样,是理解事物本质的关键一步。 sudo 的内部结构与机制详解 sudo 的执行流程可以看作是一系列精心设计的步骤,确保了授权的准确性和安…

什么是 TOML?

&#x1f6e0; Rust 配置文件实战&#xff1a;TOML 语法详解与结构体映射&#xff08; 在 Rust 中&#xff0c;Cargo.toml 是每个项目的心脏。它不仅定义了项目的名称、版本和依赖项&#xff0c;还使用了一种轻巧易读的配置语言&#xff1a;TOML。 本文将深入解析 TOML 的语法…

react native webview加载本地HTML,解决iOS无法加载成功问题

在react native中使用 “react-native-webview”: “^13.13.5”,加载HTML文件 Android: 将HTML文件放置到android/src/main/assets目录&#xff0c;访问 {uri: file:///android_asset/markmap/index.html}ios: 在IOS中可以直接可以直接放在react native项目下&#xff0c;访问…

数据结构(JAVA版)练习题

&#xff08;题目难易程度与题号顺序无关哦&#xff09; 目录 1、多关键字排序 2、集合类的综合应用问题 3、数组排序 4、球的相关计算问题 5、利用类对象计算日期 6、日期计算问题 7、星期日期的计算 8、计算坐标平面上两点距离 9、异常处理设计问题 10、Java源文件…

04-redis-分布式锁-redisson

1 基本概念 百度百科&#xff1a;控制分布式系统之间同步访问共享资源方式。 在分布式系统中&#xff0c;常常需要协调他们的动作。如果不同的系统或是同一个系统的不同主机之间共享了一个或一组资源&#xff0c;那么访问这些资源的时候&#xff0c;往往需要互斥来防止…

性能优化 - 案例篇:缓存_Guava#LoadingCache设计

文章目录 Pre引言1. 缓存基本概念2. Guava 的 LoadingCache2.1 引入依赖与初始化2.2 手动 put 与自动加载&#xff08;CacheLoader&#xff09;2.2.1 示例代码 2.3 缓存移除与监听&#xff08;invalidate removalListener&#xff09; 3. 缓存回收策略3.1 基于容量的回收&…

使用jstack排查CPU飙升的问题记录

最近&#xff0c;看到短视频传播了一个使用jstack来协助排查CPU飙升的案例。我也是比较感兴趣&#xff0c;参考了视频博主的流程&#xff0c;自己做了下对应案例的实战演练&#xff0c;在此&#xff0c;想做一下&#xff0c;针对相关问题模拟与排查演练的实战过程记录。 案例中…

Sql Server 中常用语句

1.创建用户数据库 --创建数据库 use master --切换到master数据库 go-- 终止所有与SaleManagerDB数据库的连接 alter database SaleManagerDB set single_user with rollback immediate goif exists (select * from sysdatabases where nameSaleManagerDB) drop database Sal…

联通专线赋能,亿林网络裸金属服务器:中小企业 IT 架构升级优选方案

在当今数字化飞速发展的时代&#xff0c;中小企业面临着日益增长的业务需求与复杂多变的市场竞争环境。如何构建高效、稳定且具性价比的 IT 架构&#xff0c;成为众多企业突破发展瓶颈的关键所在。而亿林网络推出的 24 核 32G 裸金属服务器&#xff0c;搭配联通专线的千兆共享带…

LangChain核心之Runnable接口底层实现

导读&#xff1a;作为LangChain框架的核心抽象层&#xff0c;Runnable接口正在重新定义AI应用开发的标准模式。这一统一接口设计将模型调用、数据处理和API集成等功能封装为可复用的逻辑单元&#xff0c;通过简洁的管道符语法实现复杂任务的声明式编排。 对于面临AI应用架构选择…

CSP严格模式返回不存在的爬虫相关文件

文章目录 说明示例&#xff08;返回404&#xff09;示例&#xff08;创建CSP例外&#xff09; 说明 日期&#xff1a;2025年6月4日。 CSP严格模式是default-src none&#xff0c;但有些web应用中&#xff0c;在爬虫相关文件不存在的情况下&#xff0c;依旧返回了对应文件&…

DeviceNET从站转EtherNET/IP主站在盐化工行业的创新应用

在工业自动化飞速发展的今天&#xff0c;盐化工行业也在积极探索智能化升级的路径。其中&#xff0c;设备之间的高效通信与协同工作成为了提升生产效率和质量的关键。而JH-DVN-EIP疆鸿智能DeviceNET从站转EtherNET/IP主站的技术应用&#xff0c;为盐化工行业带来了全新的解决方…

安装 Nginx

个人博客地址&#xff1a;安装 Nginx | 一张假钞的真实世界 对于 Linux 平台&#xff0c;Nginx 安装包 可以从 nginx.org 下载。 Ubuntu: 版本Codename支持平台12.04precisex86_64, i38614.04trustyx86_64, i386, aarch64/arm6415.10wilyx86_64, i386 在 Debian/Ubuntu 系统…

默认网关 -- 负责转发数据包到其他网络的设备(通常是路由器)

✅ 默认网关概括说明&#xff1a; 默认网关&#xff08;Default Gateway&#xff09;是网络中一台负责转发数据包到其他网络的设备&#xff08;通常是路由器&#xff09;。当一台主机要访问不在本地子网内的设备时&#xff0c;会将数据包发给默认网关&#xff0c;由它继续转发…

cv::FileStorage用法

cv::FileStorage 是 OpenCV 中的一个类&#xff0c;用于读取和写入结构化数据&#xff08;如 YAML、XML、JSON&#xff09;。它非常适合保存和加载诸如&#xff1a; 相机内参&#xff08;K、D&#xff09; 位姿&#xff08;R、T&#xff09; IMU 数据 配置参数 向量、矩阵、…