当前位置: 首页 > article >正文

ass字幕嵌入mp4带偏移

# 格式转化文件,包含多种文件的互相转化,主要与视频相关
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视频播放器,几乎就能正常的显示软字幕。

相关文章:

ass字幕嵌入mp4带偏移

# 格式转化文件&#xff0c;包含多种文件的互相转化&#xff0c;主要与视频相关 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) # 格式化为…...

WPF响应式UI的基础:INotifyPropertyChanged

INotifyPropertyChanged 1 实现基础接口2 CallerMemberName优化3 数据更新触发策略4 高级应用技巧4.1 表达式树优化4.2 性能优化模式4.3 跨平台兼容实现 5 常见错误排查 在WPF的MVVM架构中&#xff0c; INotifyPropertyChanged是实现数据驱动界面的核心机制。本章将深入解析属…...

JavaScript字符串方法全面指南:从基础到高级应用

在JavaScript开发中&#xff0c;字符串(String)是最常用的数据类型之一&#xff0c;用于存储和操作文本数据。JavaScript提供了丰富的内置方法来处理字符串&#xff0c;掌握这些方法能极大提高开发效率。本文将全面介绍JavaScript中的字符串方法&#xff0c;按照"先总后分…...

浅谈 JavaScript 性能优化

文章目录 概要一、代码执行优化1. 减少全局变量访问2. 避免不必要的计算3. 优化循环操作 二、内存管理优化1. 减少内存泄漏2. 对象池与内存复用 三、渲染性能优化1. 避免强制同步布局2. 减少 DOM 操作3. 优化动画与合成 四、网络加载优化1. 代码压缩与 Tree Shaking2. 按需加载…...

React从基础入门到高级实战:React 生态与工具 - 构建与部署

React 构建与部署 引言 在现代Web开发中&#xff0c;构建与部署是项目从开发到上线的关键环节。对于React开发者而言&#xff0c;掌握构建优化和部署策略不仅能提升应用的性能&#xff0c;还能确保项目的稳定性和安全性。随着React应用的复杂性不断增加&#xff0c;合理的构建…...

Kafka性能调优三剑客:深度解析buffer_memory、linger_ms和batch_size

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 持续学习&#xff0c;不断…...

5分钟学会网络服务搭建,飞凌i.MX9352 + Linux 6.1实战示例

在“万物互联”的技术浪潮下&#xff0c;网络服务已成为连接物理世界与数字世界的核心纽带&#xff0c;它不仅赋予了终端设备“开口说话”的能力&#xff0c;更构建了智能设备的开发范式。 本文就将以飞凌嵌入式OK-MX9352-C开发板&#xff08;搭载了在工业物联网领域广泛应用的…...

网络安全-等级保护(等保) 3-2-2 GB/T 28449-2019 第7章 现场测评活动/第8章 报告编制活动

################################################################################ GB/T 28449-2019《信息安全技术 网络安全等级保护测评过程指南》是规定了等级测评过程&#xff0c;是纵向的流程&#xff0c;包括&#xff1a;四个基本测评活动:测评准备活动、方案编制活…...

74道TypeScript高频题整理(附答案背诵版)

1.简述什么是TypeScript &#xff1f; TypeScript是一种由Microsoft开发和维护的开源编程语言。它是JavaScript的一个超集&#xff0c;意味着它扩展了JavaScript的功能&#xff0c;包括添加了类型系统和对ES6的新特性的支持。TypeScript的设计目标是帮助开发者捕捉代码中的错误…...

PostgreSQL 临时表空间

PostgreSQL 临时表空间 PostgreSQL 使用临时表空间来存储查询执行过程中产生的临时数据&#xff0c;与 Oracle 类似但实现方式有所不同。 一、临时表空间基本概念 PostgreSQL 的临时表空间主要用于存储&#xff1a; 排序操作&#xff08;ORDER BY、GROUP BY、DISTINCT&…...

N2语法 状態

1&#xff0c;&#xff5e;てならない  接続&#xff1a;て型  意味&#xff1a;…得不得了(强调自然产生的情感&#xff0c;可接自发动词)  例文&#xff1a;     お腹が痛くてならない。     心配でならない。     両親に会いたくてならない。&#xff08;非常…...

从Node.js到Go:如何从NestJS丝滑切换并爱上Sponge框架

引言 各位 NestJS 老司机们&#xff0c; 不得不说&#xff0c;用装饰器开发 API 简直像在键盘上跳华尔兹——Controller 转个圈&#xff0c;Get 踮个脚&#xff0c;Injectable 优雅谢幕&#xff0c;三下五除二就能搭出个像模像样的后端服务。TypeScript 的类型检查就像个贴心管…...

海思 35XX MIPI读取YUV422

1.项目背景&#xff1a; 使用海思芯片&#xff0c;接收FPGA发送的MIPI数据&#xff0c;不需要ISP处理&#xff0c;YUV图像格式为YUV422。 2.移植MIPI驱动 修改IMX347的驱动远吗&#xff0c;将I2C读写的部分注释&#xff0c;其他的不用再做修改。 int imx347_slave_i2c_init(ot…...

sass三大循环语法

for for 指令可以在限制的范围内重复输出格式&#xff0c;每次按要求&#xff08;变量的值&#xff09;对输出结果做出变动。这个指令包含两种格式&#xff1a;for $var from through &#xff0c;或者 for v a r f r o m < s t a r t > t o < e n d > &#xff…...

第1章 Redis 概述

一、Redis 简介 Redis,Remote Dictionary Server,远程字典服务,由意大利人Salvatore Sanfilippo(又名Antirez)开发,是一个使用ANSI C 语言编写&#xff64;支持网络&#xff64; 可基于内存亦可持久化的日志型&#xff64;NoSQL 开源内存数据库,其提供多种语言的API&#xff61…...

硬件工程师笔记——二极管Multisim电路仿真实验汇总

目录 1 二极管基础知识 1.1 工作原理 1.2 二极管的结构 1.3 PN结的形成 1.4 二极管的工作原理详解 正向偏置 反向偏置 multisim使用说明链接 2 二极管特性实验 2.1 二极管加正向电压 2.2 二极管加反向电压 2.3 二极管两端的电阻 2.4 交流电下二级管工作 2.5 二极…...

30V/3A,云岑CP8335B,完美替换EUP3484

1 FEATURES ● Wide Input Voltage Range: 6V ~ 30V ● Low RDS(ON) for Internal Switches (Top/Bottom): 90mΩ/65 mΩ ● 3A output current capability ● 500kHz Switching Frequency Minimize the External Components ● Internal 1.5-ms Soft-Start ● 0.6V/0.8V/0.925…...

基于大模型预测的FicatIII-IV期股骨头坏死综合治疗研究报告

目录 一、引言 1.1 研究背景与目的 1.2 国内外研究现状 1.3 研究意义和创新点 二、FicatIII-IV 期股骨头坏死概述 2.1 疾病定义与分期 2.2 病因与病理机制 2.3 临床症状与诊断方法 三、大模型预测原理与方法 3.1 大模型简介 3.2 数据收集与预处理 3.3 模型训练与优…...

promptfoo:让语言模型评测不再“靠感觉”——一站式 LLM 自动化测评神器深度解读

大家好&#xff0c;这里是你们喜闻乐见、永远不低调的 AI 技术博主。这篇分享&#xff0c;我要隆重介绍一个我愿称之为“LLM 测试自动化福音”的神器——promptfoo。 如果你做 LLM&#xff08;大模型&#xff09;落地开发&#xff0c;调教 prompt 拼死拼活&#xff0c;一上线用…...

LINUX安装运行jeelowcode后端项目(idea启动)

参考 LINUX安装运行jeelowcode后端项目&#xff08;命令行&#xff09;-CSDN博客 IntelliJ IDEA下载地址&#xff08;社区版、付费版&#xff09;-CSDN博客 软件已安装好&#xff0c;数据库也初始化完毕。 步骤1&#xff1a;打开项目目录步骤2&#xff1a;配置JDK步骤3&…...

硬件I2C和软件I2C的区别

硬件I2C和软件I2C的区别 一、硬件I2C 1、硬件IC的局限性及学习意义 尽管硬件IC外设在STM32等微控制器中提供了标准化的通信支持&#xff0c;但在实际应用中&#xff0c;其稳定性可能存在问题。例如&#xff0c;某些情况下外设会因事件检测异常而进入死锁状态&#xff0c;仅能…...

单元测试报错

报错信息如下所示&#xff1a; 五月 30, 2025 5:35:44 下午 org.junit.vintage.engine.descriptor.RunnerTestDescriptor warnAboutUnfilterableRunner 警告: Runner org.junit.internal.runners.ErrorReportingRunner (used on class redis.demo.RedisTemplateTest) does not…...

AWS WAF设置IP白名单

目标 设置一个组白名单IP地址&#xff0c;当发现是这些IP地址发过来的请求后&#xff0c;WAF自动放行。 创建IP集 打开WAF页面&#xff0c;开始IP集创建如下图&#xff1a; 设置ip集&#xff0c;如下图&#xff1a; aws waf acl配置白名单 找到Web ACL&#xff0c;开始在…...

智能门禁的项目

项目需求 矩阵键盘输入密码&#xff0c;正确开锁&#xff0c;错误提示&#xff0c;三次错误后蜂鸣器响三秒&#xff1b;按下#号确认输入&#xff0c;按下*号修改密码&#xff1b;密码保存在W25Q128里&#xff1b;OLED屏幕显示信息。 硬件清单 矩阵键盘OLED显示屏继电器蜂鸣器…...

《Google I/O 2025:AI浪潮下的科技革新风暴》

Google I/O 2025 盛大开幕 在科技飞速发展的时代&#xff0c;Google I/O 开发者大会一直是全球科技爱好者和开发者瞩目的焦点&#xff0c;堪称科技领域的年度盛宴。2025 年 5 月 20 日至 21 日&#xff0c;Google I/O 2025 在美国加州山景城的 Shoreline Amphitheatre 盛大举行…...

职坐标IT培训:硬件嵌入式与AI芯片开发实战

课程体系以硬件嵌入式开发与AI芯片技术融合为核心&#xff0c;构建模块化知识框架。从硬件设计规范切入&#xff0c;系统讲解PCB Layout设计中的信号完整性控制、电磁兼容性&#xff08;EMC&#xff09;优化等关键要素&#xff0c;延伸至高速电路设计中阻抗匹配与电源完整性&am…...

一句话开发Chrome摸鱼插件

本文所使用的 CodeBuddy 免费下载链接&#xff1a;腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴&#xfeff;。 CodeBuddy 一、CodeBuddy新功能特色 Craft智能体&#xff1a;自然语言驱动的全栈开发引擎Craft开发智能体的核心突破在于实现需求理解-任务拆解-代码生成的…...

Spring Boot + OpenCSV 数据清洗实战:CSV 结构化处理与可视化

目录 摘要 演示 一、背景&#xff1a;为什么需要自动化数据清洗&#xff1f; 二、技术选型&#xff1a;为什么选择这三个工具&#xff1f; 三、核心功能实现&#xff1a;从数据读取到智能清洗 1. 配置控制器 2. 文件上传控制器 3. CSV数据处理服务接口 4. CSV数据处理…...

Cmake编译glog成功并在QT中测试成功步骤

glog是开源的日志记录系统&#xff0c;下载地址GitHub - google/glog: C implementation of the Google logging module 跟gflags有点相似&#xff0c;编译和测试过程比较周折&#xff0c;所以记录下来具体的编译和测试步骤。 编译环境&#xff1a;WindowsCmakeVs2022Qt5.14.…...

AI绘画提示词:从零开始掌握Prompt Engineering的艺术

文章目录 什么是AI绘画提示词&#xff1f;提示词的基本结构主体描述场景/背景风格指定技术参数负面提示人物肖像模板风景模板 高级技巧权重调整混合风格颜色控制情绪氛围 常见问题与解决方法手部变形问题构图不理想风格不够突出 提示词示例库科幻场景奇幻人物静物画 结语 在当今…...