From f1f1036fdaa81e6463f24e53f6c9403bde00ea74 Mon Sep 17 00:00:00 2001 From: xunbu Date: Mon, 22 Sep 2025 16:00:13 +0800 Subject: [PATCH] =?UTF-8?q?=E5=A2=9E=E5=8A=A0ass=E6=94=AF=E6=8C=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- docutranslate/app.py | 103 +++++++++-- docutranslate/exporter/ass/__init__.py | 0 .../exporter/ass/ass2ass_exporter.py | 9 + .../exporter/ass/ass2html_exporter.py | 42 +++++ docutranslate/exporter/ass/base.py | 10 + docutranslate/static/i18nData.json | 172 ++++++++++-------- docutranslate/static/index.html | 2 +- docutranslate/template/ass.html | 16 ++ .../ai_translator/ass_translator.py | 136 ++++++++++++++ docutranslate/workflow/ass_workflow.py | 76 ++++++++ docutranslate/workflow/interfaces.py | 8 + pyproject.toml | 2 + uv.lock | 104 +++-------- 13 files changed, 505 insertions(+), 175 deletions(-) create mode 100644 docutranslate/exporter/ass/__init__.py create mode 100644 docutranslate/exporter/ass/ass2ass_exporter.py create mode 100644 docutranslate/exporter/ass/ass2html_exporter.py create mode 100644 docutranslate/exporter/ass/base.py create mode 100644 docutranslate/template/ass.html create mode 100644 docutranslate/translator/ai_translator/ass_translator.py create mode 100644 docutranslate/workflow/ass_workflow.py diff --git a/docutranslate/app.py b/docutranslate/app.py index 71e5527..9fd7a7f 100644 --- a/docutranslate/app.py +++ b/docutranslate/app.py @@ -31,12 +31,11 @@ from docutranslate.global_values.conditional_import import DOCLING_EXIST from docutranslate.workflow.base import Workflow from docutranslate.workflow.docx_workflow import DocxWorkflow, DocxWorkflowConfig from docutranslate.workflow.epub_workflow import EpubWorkflow, EpubWorkflowConfig -# --- HTML WORKFLOW IMPORT START --- from docutranslate.workflow.html_workflow import HtmlWorkflow, HtmlWorkflowConfig -# --- HTML WORKFLOW IMPORT END --- +from docutranslate.workflow.ass_workflow import AssWorkflow, AssWorkflowConfig from docutranslate.workflow.interfaces import DocxExportable, EpubExportable from docutranslate.workflow.interfaces import HTMLExportable, MDFormatsExportable, TXTExportable, JsonExportable, \ - XlsxExportable, SrtExportable, CsvExportable + XlsxExportable, SrtExportable, CsvExportable, AssExportable from docutranslate.workflow.json_workflow import JsonWorkflow, JsonWorkflowConfig from docutranslate.workflow.md_based_workflow import MarkdownBasedWorkflow, MarkdownBasedWorkflowConfig from docutranslate.workflow.srt_workflow import SrtWorkflow, SrtWorkflowConfig @@ -60,9 +59,9 @@ from docutranslate.translator.ai_translator.srt_translator import SrtTranslatorC from docutranslate.exporter.srt.srt2html_exporter import Srt2HTMLExporterConfig from docutranslate.translator.ai_translator.epub_translator import EpubTranslatorConfig from docutranslate.exporter.epub.epub2html_exporter import Epub2HTMLExporterConfig -# --- HTML TRANSLATOR IMPORT START --- from docutranslate.translator.ai_translator.html_translator import HtmlTranslatorConfig -# --- HTML TRANSLATOR IMPORT END --- +from docutranslate.translator.ai_translator.ass_translator import AssTranslatorConfig +from docutranslate.exporter.ass.ass2html_exporter import Ass2HTMLExporterConfig # ------------------------------------ from docutranslate.logger import global_logger @@ -86,6 +85,7 @@ WORKFLOW_DICT: Dict[str, Type[Workflow]] = { "srt": SrtWorkflow, "epub": EpubWorkflow, "html": HtmlWorkflow, + "ass": AssWorkflow, } # --- 媒体类型映射 --- @@ -100,6 +100,7 @@ MEDIA_TYPES = { "docx": "application/vnd.openxmlformats-officedocument.wordprocessingml.document", "srt": "text/plain; charset=utf-8", "epub": "application/epub+zip", + "ass": "text/plain; charset=utf-8", } @@ -159,7 +160,7 @@ async def lifespan(app: FastAPI): global_logger.propagate = False global_logger.setLevel(logging.INFO) print("应用启动完成,多任务状态已初始化。") - print(f"服务接口文档: http://127.0.0.1:{app.state.port_to_use}/docs") + print(f"服务接口文档: http://12ent.0.0.1:{app.state.port_to_use}/docs") print(f"请用浏览器访问 http://127.0.0.1:{app.state.port_to_use}\n") yield # 清理任何可能残留的临时目录 @@ -391,10 +392,26 @@ class HtmlWorkflowParams(BaseWorkflowParams): # --- HTML WORKFLOW PARAMS END --- +# --- ASS WORKFLOW PARAMS START --- +class AssWorkflowParams(BaseWorkflowParams): + workflow_type: Literal['ass'] = Field(..., description="指定使用ASS字幕的翻译工作流。") + insert_mode: Literal["replace", "append", "prepend"] = Field( + "replace", + description="翻译文本的插入模式。'replace':替换原文,'append':附加到原文后,'prepend':附加到原文前。" + ) + separator: str = Field( + "\\N", + description="当 insert_mode 为 'append' 或 'prepend' 时,用于分隔原文和译文的分隔符。ASS格式通常使用 \\N 作为换行符。" + ) + + +# --- ASS WORKFLOW PARAMS END --- + + # 3. 使用可辨识联合类型(Discriminated Union)将它们组合起来 TranslatePayload = Annotated[ Union[ - MarkdownWorkflowParams, TextWorkflowParams, JsonWorkflowParams, XlsxWorkflowParams, DocxWorkflowParams, SrtWorkflowParams, EpubWorkflowParams, HtmlWorkflowParams], + MarkdownWorkflowParams, TextWorkflowParams, JsonWorkflowParams, XlsxWorkflowParams, DocxWorkflowParams, SrtWorkflowParams, EpubWorkflowParams, HtmlWorkflowParams, AssWorkflowParams], Field(discriminator='workflow_type') ] @@ -403,7 +420,7 @@ TranslatePayload = Annotated[ class TranslateServiceRequest(BaseModel): file_name: str = Field(..., description="上传的原始文件名,含扩展名。", examples=["my_paper.pdf", "chapter1.txt", "data.xlsx", "video.srt", "my_book.epub", - "index.html"]) + "index.html", "dialogue.ass"]) file_content: str = Field(..., description="Base64编码的文件内容。", examples=["JVBERi0xLjQK..."]) payload: TranslatePayload = Field(..., description="包含工作流类型和相应参数的载荷。") @@ -582,6 +599,26 @@ class TranslateServiceRequest(BaseModel): "thinking": "default", "retry": default_params["retry"], } + }, + { + "file_name": "dialogue.ass", + "file_content": "U2NyaXB0IEluZm8NC...", + "payload": { + "workflow_type": "ass", + "skip_translate": False, + "base_url": "https://api.openai.com/v1", + "api_key": "sk-your-api-key-here", + "model_id": "gpt-4o", + "to_lang": "中文", + "insert_mode": "replace", + "separator": "\\N", + "chunk_size": default_params["chunk_size"], + "concurrent": default_params["concurrent"], + "temperature": default_params["temperature"], + "timeout": default_params["timeout"], + "thinking": "default", + "retry": default_params["retry"], + } } ] } @@ -787,6 +824,27 @@ async def _perform_translation( workflow = HtmlWorkflow(config=workflow_config) # --- HTML WORKFLOW LOGIC END --- + # --- ASS WORKFLOW LOGIC START --- + elif isinstance(payload, AssWorkflowParams): + task_logger.info("构建 AssWorkflow 配置。") + translator_args = payload.model_dump(include={ + 'skip_translate', 'base_url', 'api_key', 'model_id', 'to_lang', 'custom_prompt', + 'temperature', 'thinking', 'chunk_size', 'concurrent', + 'insert_mode', 'separator', 'glossary_dict', 'timeout', 'retry' + }, exclude_none=True) + translator_args['glossary_generate_enable'] = payload.glossary_generate_enable + translator_args['glossary_agent_config'] = build_glossary_agent_config() + translator_config = AssTranslatorConfig(**translator_args) + + html_exporter_config = Ass2HTMLExporterConfig(cdn=True) + workflow_config = AssWorkflowConfig( + translator_config=translator_config, + html_exporter_config=html_exporter_config, + logger=task_logger + ) + workflow = AssWorkflow(config=workflow_config) + # --- ASS WORKFLOW LOGIC END --- + else: raise TypeError(f"工作流类型 '{payload.workflow_type}' 的处理逻辑未实现。") @@ -832,6 +890,8 @@ async def _perform_translation( html_config = Srt2HTMLExporterConfig(cdn=is_cdn_available) elif isinstance(workflow, EpubWorkflow): html_config = Epub2HTMLExporterConfig(cdn=is_cdn_available) + elif isinstance(workflow, AssWorkflow): + html_config = Ass2HTMLExporterConfig(cdn=is_cdn_available) export_map['html'] = (lambda: workflow.export_to_html(html_config), f"{filename_stem}_translated.html", True) if isinstance(workflow, MDFormatsExportable): @@ -851,6 +911,8 @@ async def _perform_translation( export_map['srt'] = (workflow.export_to_srt, f"{filename_stem}_translated.srt", True) if isinstance(workflow, EpubExportable): export_map['epub'] = (workflow.export_to_epub, f"{filename_stem}_translated.epub", False) + if isinstance(workflow, AssExportable): + export_map['ass'] = (workflow.export_to_ass, f"{filename_stem}_translated.ass", True) # 循环生成文件 for file_type, (export_func, filename, is_string_output) in export_map.items(): @@ -1013,7 +1075,7 @@ def _cancel_translation_logic(task_id: str): description=""" 接收一个包含文件内容(Base64编码)和工作流参数的JSON请求,启动一个后台翻译任务。 -- **工作流选择**: 请求体中的 `payload.workflow_type` 字段决定了本次任务的类型(如 `markdown_based`, `txt`, `json`, `xlsx`, `docx`, `srt`, `epub`, `html`)。 +- **工作流选择**: 请求体中的 `payload.workflow_type` 字段决定了本次任务的类型(如 `markdown_based`, `txt`, `json`, `xlsx`, `docx`, `srt`, `epub`, `html`, `ass`)。 - **动态参数**: 根据所选工作流,API需要不同的参数集。请参考下面的Schema或示例。 - **异步处理**: 此端点会立即返回任务ID,客户端需轮询状态接口获取进度。 """, @@ -1220,6 +1282,23 @@ async def service_release_task(task_id: str): } }, # --- HTML STATUS EXAMPLE END --- + # --- ASS STATUS EXAMPLE START --- + "completed_ass": { + "summary": "已完成 (ASS)", + "value": { + "task_id": "a1b2c3d5", "is_processing": False, + "status_message": "翻译成功!用时 12.34 秒。", + "error_flag": False, "download_ready": True, "original_filename_stem": "dialogue", + "original_filename": "dialogue.ass", "task_start_time": 1678890200.0, + "task_end_time": 1678890212.34, + "downloads": { + "ass": "/service/download/a1b2c3d5/ass", + "html": "/service/download/a1b2c3d5/html" + }, + "attachment": {} + } + }, + # --- ASS STATUS EXAMPLE END --- "error": { "summary": "失败", "value": { @@ -1287,7 +1366,7 @@ async def service_get_logs(task_id: str): return JSONResponse(content={"logs": new_logs}) -FileType = Literal["markdown", "markdown_zip", "html", "txt", "json", "xlsx", "csv", "docx", "srt", "epub"] +FileType = Literal["markdown", "markdown_zip", "html", "txt", "json", "xlsx", "csv", "docx", "srt", "epub", "ass"] @service_router.get( @@ -1318,7 +1397,7 @@ FileType = Literal["markdown", "markdown_zip", "html", "txt", "json", "xlsx", "c async def service_download_file( task_id: str = FastApiPath(..., description="已完成任务的ID", examples=["b2865b93"]), file_type: FileType = FastApiPath(..., description="要下载的文件类型。", - examples=["html", "json", "csv", "docx", "srt", "epub"]) + examples=["html", "json", "csv", "docx", "srt", "epub", "ass"]) ): task_state = tasks_state.get(task_id) if not task_state: @@ -1418,7 +1497,7 @@ async def service_download_attachment( async def service_content( task_id: str = FastApiPath(..., description="已完成任务的ID", examples=["b2865b93"]), file_type: FileType = FastApiPath(..., description="要获取内容的文件类型。", - examples=["html", "json", "csv", "docx", "srt", "epub"]) + examples=["html", "json", "csv", "docx", "srt", "epub", "ass"]) ): task_state = tasks_state.get(task_id) if not task_state: diff --git a/docutranslate/exporter/ass/__init__.py b/docutranslate/exporter/ass/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/docutranslate/exporter/ass/ass2ass_exporter.py b/docutranslate/exporter/ass/ass2ass_exporter.py new file mode 100644 index 0000000..36093a4 --- /dev/null +++ b/docutranslate/exporter/ass/ass2ass_exporter.py @@ -0,0 +1,9 @@ +# SPDX-FileCopyrightText: 2025 QinHan +# SPDX-License-Identifier: MPL-2.0 +from docutranslate.exporter.ass.base import AssExporter +from docutranslate.ir.document import Document + + +class Ass2AssExporter(AssExporter): + def export(self, document: Document) -> Document: + return document.copy() diff --git a/docutranslate/exporter/ass/ass2html_exporter.py b/docutranslate/exporter/ass/ass2html_exporter.py new file mode 100644 index 0000000..7805ed4 --- /dev/null +++ b/docutranslate/exporter/ass/ass2html_exporter.py @@ -0,0 +1,42 @@ +# SPDX-FileCopyrightText: 2025 QinHan +# SPDX-License-Identifier: MPL-2.0 +from dataclasses import dataclass + +import jinja2 + +from docutranslate.exporter.ass.base import AssExporter +from docutranslate.exporter.base import ExporterConfig + +from docutranslate.ir.document import Document +from docutranslate.utils.resource_utils import resource_path + + +@dataclass +class Ass2HTMLExporterConfig(ExporterConfig): + cdn: bool = True + + +class Ass2HTMLExporter(AssExporter): + def __init__(self, config: Ass2HTMLExporterConfig = None): + config = config or Ass2HTMLExporterConfig() + super().__init__(config=config) + self.cdn = config.cdn + + def export(self, document: Document) -> Document: + cdn = self.cdn + + html_template = resource_path("template/ass.html").read_text(encoding="utf-8") + + render = jinja2.Template(html_template).render( + ass_data=document.content.decode("utf-8") + ) + return Document.from_bytes(content=render.encode("utf-8"), suffix=".html", stem=document.stem) + +if __name__ == '__main__': + from pathlib import Path + d=Document.from_path(r"C:\Users\jxgm\Desktop\testfiles\一个软件搞定文件翻译【DocuTranslate】.ass") + exporter=Ass2HTMLExporter() + d_html=exporter.export(d) + path=Path("./1.html") + path.write_text(d_html.content.decode("utf-8")) + diff --git a/docutranslate/exporter/ass/base.py b/docutranslate/exporter/ass/base.py new file mode 100644 index 0000000..a81a140 --- /dev/null +++ b/docutranslate/exporter/ass/base.py @@ -0,0 +1,10 @@ +# SPDX-FileCopyrightText: 2025 QinHan +# SPDX-License-Identifier: MPL-2.0 +from docutranslate.exporter.base import Exporter +from docutranslate.ir.document import Document + +#TODO:看情况是否需要为TXT单独写一个document类型 +class AssExporter(Exporter[Document]): + + def export(self,document:Document)->Document: + ... \ No newline at end of file diff --git a/docutranslate/static/i18nData.json b/docutranslate/static/i18nData.json index 59f7c20..9212f47 100644 --- a/docutranslate/static/i18nData.json +++ b/docutranslate/static/i18nData.json @@ -12,8 +12,8 @@ "workflowOptionSrt": "SRT字幕翻译 (.srt)", "workflowOptionEpub": "EPUB翻译 (.epub)", "workflowOptionHtml": "HTML翻译 (.html)", + "workflowOptionAss": "ASS字幕翻译 (.ass)", "autoWorkflowLabel": "自动选择工作流", - "txtSettingsTitleText": "TXT翻译选项", "insertModeLabel": "插入模式", "insertModeReplace": "替换原文 (Replace)", "insertModeAppend": "附加到原文后 (Append)", @@ -22,24 +22,20 @@ "separatorLabel": "分隔符", "separatorPlaceholderSimple": "例如: \\n---\\n", "separatorHelp": "当插入模式为附加或前置时,用于分隔原文和译文的字符。\\n 代表换行。", - "docxSettingsTitleText": "DOCX翻译选项", + "separatorHelpAss": "当插入模式为附加或前置时,用于分隔原文和译文的字符。\\N 是ASS格式的换行符。\n", "insertModeHelpDocx": "选择如何将翻译后的文本插入。", "separatorPlaceholder": "例如: \\n---翻译---\\n", - "xlsxSettingsTitleText": "XLSX翻译选项", "insertModeHelpXlsx": "选择如何将翻译后的文本插入到单元格中。", "xlsxTranslateRegionsLabel": "翻译区域 (可选)", "xlsxTranslateRegionsPlaceholder": "每行一个区域, 例如:Sheet1!A1:B10(不指定表名则对所有表生效)", - "srtSettingsTitleText": "SRT翻译选项", "insertModeHelpSrt": "选择如何将翻译后的文本插入。", - "epubSettingsTitleText": "EPUB翻译选项", "insertModeHelpEpub": "选择如何将翻译后的文本插入。", - "htmlSettingsTitleText": "HTML翻译选项", "insertModeHelpHtml": "选择如何将翻译后的文本插入。", - "jsonSettingsTitleText": "JSON路径配置", + "separatorPlaceholderAss": "例如: \\N (换行符)", + "insertModeHelpAss": "选择如何将翻译后的文本插入。", "jsonPathLabel": "需要翻译的JSON路径", "jsonPathPlaceholder": "每行一个路径, 例如:\n$.name\n$.*", - "jsonPathHelp": "采用jsonpath-ng的路径选择语法,每一行表示一个json路径。\n 将翻译路径匹配对象内的所有字符串", - "parsingSettingsTitleText": "解析配置", + "jsonPathHelp": "采用jsonpath-ng的路径选择语法,每一行表示一个json路径。 将翻译路径匹配对象内的所有字符串", "parsingEngineLabel": "解析引擎", "parsingEngineHelp": "如果上传的文件本身是.md格式,此项可不选。", "getMineruTokenTitle": "获取Mineru Token", @@ -50,7 +46,6 @@ "modelVersionHelp": "mineru VLM是更新的内测模型。", "formulaOcrLabel": "公式识别", "codeOcrLabel": "代码识别", - "aiSettingsTitleText": "翻译模型", "skipTranslationLabel": "跳过翻译", "platformLabel": "选择平台", "platformCustom": "自定义接口", @@ -60,7 +55,6 @@ "apiKeyPlaceholder": "请输入您的API Key", "modelIdLabel": "选择模型", "modelIdPlaceholder": "例如: gpt-4o, glm-4", - "translationSettingsTitleText": "翻译配置", "targetLanguageLabel": "目标语言", "targetLanguageCustom": "其它 (自定义)", "customLangPlaceholder": "请输入目标语言, 例如: Italian", @@ -84,7 +78,7 @@ "glossaryGenConfigLabel": "生成术语表配置", "glossaryGenConfigSame": "与翻译配置相同", "glossaryGenConfigCustom": "自定义", - "githubInfo": "GitHub主页(欢迎star❤):
https://github.com/xunbu/docutranslate", + "githubInfo": "GitHub主页(欢迎star❤):
https://github.com/xunbu/docutranslate", "qqGroupInfo": "交流QQ群: 1047781902", "taskListTitle": "任务列表", "newTaskBtn": "新建任务", @@ -102,6 +96,7 @@ "taskCardStartBtn": "开始翻译", "downloadMdEmbedded": "Markdown(嵌图)", "downloadMdZip": "Markdown压缩包", + "downloadAss": "ASS", "previewTitle": "预览", "previewBilingualBtn": "双语", "previewTranslatedOnlyBtn": "仅译文", @@ -110,7 +105,7 @@ "closeBtn": "关闭", "downloadBtn": "下载", "tutorialModalTitle": "使用教程", - "tutorialModalBody": "

视频教程可以在B站搜索 docutranslate 获取。

欢迎使用 DocuTranslate!请按照以下步骤完成文档翻译:

  1. 选择工作流

    首先,在配置面板顶部选择您需要的翻译流程。不同的工作流适用于不同类型的文件:

    • 转Markdown再翻译: 适用于翻译PDF、markdown、图片等文件。
    • 纯文本翻译: 用于翻译 .txt 等纯文本文件。
    • JSON翻译: 用于翻译 .json 文件中的特定字段。
    • DOCX翻译: 用于翻译 .docx 文件。
    • XLSX翻译: 用于翻译 .xlsx 电子表格、 .csv 文件。
    • SRT字幕翻译: 用于翻译 .srt 字幕文件。
    • EPUB翻译: 用于翻译 .epub 电子书文件。
    • HTML翻译: 用于翻译 .html 文件。
    新增功能: \"自动选择工作流\"开关已默认开启。您只需上传文件,系统会自动为您匹配合适的工作流,简化操作。

  2. 配置参数

    根据您选择的工作流,完成相应的配置。所有配置项都会自动保存在您的浏览器中。

    • 解析配置 (仅在“转Markdown再翻译”工作流下显示):
      • 解析引擎: 选择一个引擎将您的文件(如PDF)转换为适合翻译的Markdown格式。如果您的文件已经是Markdown格式,则无需选择。
      • Mineru Token: 如果您选择 minerU 引擎,需要在此处填入您的Token。
    • TXT/DOCX/XLSX/SRT/EPUB/HTML翻译选项 (在对应工作流下显示):
      • 插入模式: 定义翻译结果如何放入文档或字幕。您可以选择直接“替换”原文,或是在原文之后“附加”,或是在原文之前“前置”。
      • 分隔符: 当选择“附加”或“前置”模式时,此项用于在原文和译文之间插入分隔符。
    • JSON路径配置 (仅在“JSON翻译”工作流下显示):
      • 需要翻译的JSON路径: 每行输入一个 JSONPath 表达式,指定需要翻译的字段。
      • 例如:$..description翻译所有键为description的值。$.items[0].name翻译第一个item的name值。$.*翻译所有字符串。
    • 翻译模型:
      • 跳过翻译: 勾选此项后,将只执行文档解析和格式转换,不调用AI进行翻译。
      • 选择平台/API 地址/API Key/模型 ID: 配置您希望使用的AI翻译服务。
      • 模型ID参考平台文档,建议使用非推理模型或混合推理模型(关闭思考)。
      • 模型能力指令遵循越强,出错漏翻的概率越低。
    • 翻译配置:
      • 目标语言/自定义Prompt/术语表: 指定翻译的目标语言、附加指令以及用于保证特定名词翻译准确性的术语表。
      • 思考模式:设置混合推理模型是否进行思考,目前支持智谱的glm4.5系列、火山引擎的seed1.6系列、硅基流动平台、google的gemini系列,建议选择禁用思考。
      • 分块大小/并发数/Temperature/重试次数: 发给AI的分块大小、并发请求数、温度和失败重试次数,通常保持默认即可。
  3. 上传文件

    在右侧的任务列表中,点击或拖拽您的文档到文件上传区域。

  4. 开始翻译

    文件选择成功后,点击任务卡片右下角的 开始翻译 按钮。系统将开始处理任务,您可以在日志区域查看实时进度。

  5. 查看与下载

    翻译完成后,任务卡片下方会出现操作按钮:

    • 预览: 在右侧滑出的面板中进行原文和译文的对照预览(仅作参考)。
    • 下载: 下载包括 PDF, DOCX, XLSX, HTML, Markdown 等多种格式的译文。
    • 附件: 如果翻译过程中生成了附加文件(如术语表),可在此处下载。
提示: 所有配置都会自动保存在您的浏览器本地,方便下次使用。
", + "tutorialModalBody": "

视频教程可以在B站搜索 docutranslate 获取。

欢迎使用 DocuTranslate!请按照以下步骤完成文档翻译:

  1. 选择工作流

    首先,在配置面板顶部选择您需要的翻译流程。不同的工作流适用于不同类型的文件:

    • 转Markdown再翻译: 适用于翻译PDF、markdown、图片等文件。
    • 纯文本翻译: 用于翻译 .txt 等纯文本文件。
    • JSON翻译: 用于翻译 .json 文件中的特定字段。
    • DOCX翻译: 用于翻译 .docx 文件。
    • XLSX翻译: 用于翻译 .xlsx 电子表格、 .csv 文件。
    • SRT字幕翻译: 用于翻译 .srt 字幕文件。
    • EPUB翻译: 用于翻译 .epub 电子书文件。
    • HTML翻译: 用于翻译 .html 文件。
    新增功能: \"自动选择工作流\"开关已默认开启。您只需上传文件,系统会自动为您匹配合适的工作流,简化操作。
  2. 配置参数

    根据您选择的工作流,完成相应的配置。所有配置项都会自动保存在您的浏览器中。

    • 解析配置 (仅在“转Markdown再翻译”工作流下显示):
      • 解析引擎: 选择一个引擎将您的文件(如PDF)转换为适合翻译的Markdown格式。如果您的文件已经是Markdown格式,则无需选择。
      • Mineru Token: 如果您选择 minerU 引擎,需要在此处填入您的Token。
    • TXT/DOCX/XLSX/SRT/EPUB/HTML翻译选项 (在对应工作流下显示):
      • 插入模式: 定义翻译结果如何放入文档或字幕。您可以选择直接“替换”原文,或是在原文之后“附加”,或是在原文之前“前置”。
      • 分隔符: 当选择“附加”或“前置”模式时,此项用于在原文和译文之间插入分隔符。
    • JSON路径配置 (仅在“JSON翻译”工作流下显示):
      • 需要翻译的JSON路径: 每行输入一个 JSONPath 表达式,指定需要翻译的字段。
      • 例如:$..description翻译所有键为description的值。$.items[0].name翻译第一个item的name值。 $.*翻译所有字符串。
    • 翻译模型:
      • 跳过翻译: 勾选此项后,将只执行文档解析和格式转换,不调用AI进行翻译。
      • 选择平台/API 地址/API Key/模型 ID: 配置您希望使用的AI翻译服务。
      • 模型ID参考平台文档,建议使用非推理模型或混合推理模型(关闭思考)。
      • 模型能力指令遵循越强,出错漏翻的概率越低。
    • 翻译配置:
      • 目标语言/自定义Prompt/术语表: 指定翻译的目标语言、附加指令以及用于保证特定名词翻译准确性的术语表。
      • 思考模式:设置混合推理模型是否进行思考,目前支持智谱的glm4.5系列、火山引擎的seed1.6系列、硅基流动平台、google的gemini系列,建议选择禁用思考。
      • 分块大小/并发数/Temperature/重试次数: 发给AI的分块大小、并发请求数、温度和失败重试次数,通常保持默认即可。
  3. 上传文件

    在右侧的任务列表中,点击或拖拽您的文档到文件上传区域。

  4. 开始翻译

    文件选择成功后,点击任务卡片右下角的 开始翻译 按钮。系统将开始处理任务,您可以在日志区域查看实时进度。

  5. 查看与下载

    翻译完成后,任务卡片下方会出现操作按钮:

    • 预览: 在右侧滑出的面板中进行原文和译文的对照预览(仅作参考)。
    • 下载: 下载包括 PDF, DOCX, XLSX, HTML, Markdown 等多种格式的译文。
    • 附件: 如果翻译过程中生成了附加文件(如术语表),可在此处下载。
提示: 所有配置都会自动保存在您的浏览器本地,方便下次使用。
", "tutorialUnderstandBtn": "我明白了", "contributorsModalTitle": "感谢贡献", "contributorsPara1": "DocuTranslate是一个开源项目!大家的需求与使用是项目进步的动力。", @@ -125,42 +120,53 @@ "glossaryTableDestination": "译文 (dst)", "init_i18n_failed_alert": "加载界面翻译资源失败,请检查网络连接或联系管理员。", "init_failed_alert": "初始化失败,无法连接到后端服务。请检查服务是否运行或刷新页面。", - "engineOptionIdentity": "已经是markdown格式", - "engineOptionMineru": "Mineru", - "engineOptionDocling": "Docling", "glossaryEmpty": "术语表为空。", + "parsingSettingsTitleText": "解析配置", + "txtSettingsTitleText": "TXT翻译选项", + "jsonSettingsTitleText": "JSON路径配置", + "xlsxSettingsTitleText": "XLSX翻译选项", + "docxSettingsTitleText": "DOCX翻译选项", + "srtSettingsTitleText": "SRT翻译选项", + "epubSettingsTitleText": "EPUB翻译选项", + "htmlSettingsTitleText": "HTML翻译选项", + "assSettingsTitleText": "ASS翻译选项", + "aiSettingsTitleText": "翻译模型", + "translationSettingsTitleText": "翻译配置", + "engineOptionIdentity": "已经是markdown格式", + "engineOptionMineru": "minerU", + "engineOptionDocling": "docling", "status_selectFileFirst": "请先选择文件!", "status_fillRequired": "请填写所有必填项!", "btn_initializing": "初始化中...", - "status_encodingAndSubmitting": "文件编码和提交中...", - "status_requestOk": "请求成功,任务已开始。", + "status_encodingAndSubmitting": "文件编码与提交中...", + "status_requestOk": "请求成功,任务已开始", "btn_cancelTranslation": "取消翻译", "status_requestFail": "请求失败", "status_initFail": "初始化失败", "status_cancelling": "取消中...", - "status_cancelSent": "取消请求已发送。", + "status_cancelSent": "取消请求已发送", "status_cancelFail": "取消失败", - "btn_reTranslate": "重新翻译", "status_gettingStatus": "获取状态中...", - "status_updateError": "状态更新出错。", - "preview_loading": "预览加载中...", - "preview_cantReadOriginal": "无法读取原始文件内容。", - "preview_cantPreviewType": "无法预览此文件类型", - "preview_noOriginalCache": "无原始文件缓存,无法预览。", - "preview_loadFailed": "预览加载失败。", - "pdf_preparing": "PDF生成准备中...", - "pdf_print_failed": "调用打印失败,请尝试手动右键打印。", + "btn_reTranslate": "重新翻译", + "status_updateError": "更新状态失败", + "pdf_preparing": "正在准备PDF...", + "pdf_print_failed": "打印PDF失败,请尝试手动下载HTML并使用浏览器打印。", "pdf_fetch_failed": "获取翻译内容失败,无法生成PDF。", - "preview_bilingual": "双语对照预览", + "preview_bilingual": "双语预览", "preview_translatedOnly": "仅译文预览", - "admin_tasklist_failed": "管理员模式:加载任务列表失败" + "preview_loading": "加载预览中...", + "preview_cantReadOriginal": "无法读取原文文件内容进行预览。", + "preview_cantPreviewType": "无法预览此文件类型", + "preview_noOriginalCache": "页面刷新后无法获取原文,请重新上传文件以预览。", + "preview_loadFailed": "加载预览失败", + "admin_tasklist_failed": "管理员模式:加载任务列表失败。" }, "en": { "pageTitle": "DocuTranslate - Interactive Document Translation", "tutorialBtn": "Tutorial", - "projectContributeBtn": "Project Contribution", + "projectContributeBtn": "Contribute", "workflowTitle": "Select Workflow", - "workflowOptionMarkdown": "Convert to Markdown then Translate (.pdf/.md/.png, etc.)", + "workflowOptionMarkdown": "Convert to Markdown then Translate (.pdf/.md/.png etc.)", "workflowOptionTxt": "Plain Text Translation (.txt)", "workflowOptionJson": "JSON Translation (.json)", "workflowOptionDocx": "DOCX Translation (.docx)", @@ -168,8 +174,8 @@ "workflowOptionSrt": "SRT Subtitle Translation (.srt)", "workflowOptionEpub": "EPUB Translation (.epub)", "workflowOptionHtml": "HTML Translation (.html)", + "workflowOptionAss": "ASS Subtitle Translation (.ass)", "autoWorkflowLabel": "Auto-select Workflow", - "txtSettingsTitleText": "TXT Translation Options", "insertModeLabel": "Insert Mode", "insertModeReplace": "Replace Original (Replace)", "insertModeAppend": "Append to Original (Append)", @@ -177,51 +183,45 @@ "insertModeHelpTxt": "Choose how to insert the translated text.", "separatorLabel": "Separator", "separatorPlaceholderSimple": "e.g., \\n---\\n", - "separatorHelp": "When the insert mode is append or prepend, these characters are used to separate the original and translated text. \\n represents a newline.", - "docxSettingsTitleText": "DOCX Translation Options", + "separatorHelp": "Separator between original and translated text when using Append or Prepend mode. \\n represents a newline.", + "separatorHelpAss": "Separator between original and translated text when using Append or Prepend mode. \\N represents a newline in ass.", "insertModeHelpDocx": "Choose how to insert the translated text.", - "separatorPlaceholder": "e.g., \\n---Translation---\\n", - "xlsxSettingsTitleText": "XLSX Translation Options", + "separatorPlaceholder": "e.g., \\n---Translated---\\n", "insertModeHelpXlsx": "Choose how to insert the translated text into cells.", - "xlsxTranslateRegionsLabel": "Translation Regions (Optional)", + "xlsxTranslateRegionsLabel": "Translate Regions (Optional)", "xlsxTranslateRegionsPlaceholder": "One region per line, e.g., Sheet1!A1:B10 (applies to all sheets if sheet name is omitted)", - "srtSettingsTitleText": "SRT Translation Options", "insertModeHelpSrt": "Choose how to insert the translated text.", - "epubSettingsTitleText": "EPUB Translation Options", "insertModeHelpEpub": "Choose how to insert the translated text.", - "htmlSettingsTitleText": "HTML Translation Options", "insertModeHelpHtml": "Choose how to insert the translated text.", - "jsonSettingsTitleText": "JSON Path Configuration", + "separatorPlaceholderAss": "e.g., \\N (newline character)", + "insertModeHelpAss": "Choose how to insert the translated text.", "jsonPathLabel": "JSON Paths to Translate", "jsonPathPlaceholder": "One path per line, e.g.:\n$.name\n$.*", - "jsonPathHelp": "Uses jsonpath-ng path selection syntax. Each line represents a JSON path.\n All strings within the matched objects will be translated.", - "parsingSettingsTitleText": "Parsing Configuration", + "jsonPathHelp": "Uses jsonpath-ng syntax. Each line represents a JSON path. All strings within the matched objects will be translated.", "parsingEngineLabel": "Parsing Engine", - "parsingEngineHelp": "If the uploaded file is already in .md format, this option can be skipped.", + "parsingEngineHelp": "If the uploaded file is already in .md format, this can be skipped.", "getMineruTokenTitle": "Get Mineru Token", - "mineruTokenPlaceholder": "Required when using Mineru engine", + "mineruTokenPlaceholder": "Required when using the Mineru engine", "modelVersionLabel": "Mineru Model Version", "modelVersionVlm": "VLM", "modelVersionPipline": "Pipeline", - "modelVersionHelp": "Mineru VLM is a newer internal testing model.", + "modelVersionHelp": "Mineru VLM is a newer model in internal testing.", "formulaOcrLabel": "Formula Recognition", "codeOcrLabel": "Code Recognition", - "aiSettingsTitleText": "Translation Model", "skipTranslationLabel": "Skip Translation", "platformLabel": "Select Platform", - "platformCustom": "Custom API", + "platformCustom": "Custom Endpoint", "baseUrlLabel": "API Address (Base URL)", - "baseUrlPlaceholder": "OpenAI-compatible address", + "baseUrlPlaceholder": "OpenAI-Compatible Address", "getApiKeyTitle": "Get API Key", - "apiKeyPlaceholder": "Please enter your API Key", + "apiKeyPlaceholder": "Enter your API Key", "modelIdLabel": "Select Model", "modelIdPlaceholder": "e.g., gpt-4o, glm-4", - "translationSettingsTitleText": "Translation Configuration", "targetLanguageLabel": "Target Language", "targetLanguageCustom": "Other (Custom)", "customLangPlaceholder": "Enter target language, e.g., Italian", "thinkingModeLabel": "Thinking Mode", - "thinkingModeTooltip": "Set whether the hybrid inference model should think. Currently supports Zhipu's glm4.5 series, Volcengine's seed1.6 series, SiliconFlow platform, and Google's Gemini series. It is recommended to disable thinking.", + "thinkingModeTooltip": "Sets whether the mixed-inference model should 'think'. Currently supported by Zhipu's glm4.5 series, Volcengine's seed1.6 series, SiliconFlow platform, and Google's Gemini series. Disabling 'think' is recommended.", "thinkingModeEnable": "Enable", "thinkingModeDisable": "Disable (Recommended)", "thinkingModeDefault": "Default", @@ -233,18 +233,18 @@ "retryLabel": "Retry Count", "glossaryGenTitle": "Glossary", "glossaryLabel": "Glossary (Optional)", - "glossaryHelp": "Select one or more CSV files. Files must contain 'src' and 'dst' column headers, representing the source and destination text respectively.", + "glossaryHelp": "Select one or more CSV files. Files must contain 'src' and 'dst' headers, representing source and destination text respectively.", "viewGlossaryBtn": "View Glossary", "clearGlossaryBtn": "Clear", "glossaryGenEnableLabel": "Auto-generate Glossary", "glossaryGenConfigLabel": "Glossary Generation Config", "glossaryGenConfigSame": "Same as Translation Config", "glossaryGenConfigCustom": "Custom", - "githubInfo": "GitHub Homepage (Stars are welcome❤):
https://github.com/xunbu/docutranslate", - "qqGroupInfo": "QQ Group for discussion: 1047781902", + "githubInfo": "GitHub Homepage (Stars are welcome❤):
https://github.com/xunbu/docutranslate", + "qqGroupInfo": "QQ Group: 1047781902", "taskListTitle": "Task List", "newTaskBtn": "New Task", - "noTaskPlaceholder": "No tasks currently. Click 'New Task' to get started!", + "noTaskPlaceholder": "No tasks yet. Click 'New Task' to get started!", "taskCardIdLabel": "Task ID", "taskCardIdPlaceholder": "Waiting for submission...", "taskCardFileDrop": "Click or drag file here", @@ -258,6 +258,7 @@ "taskCardStartBtn": "Start Translation", "downloadMdEmbedded": "Markdown (Embedded Images)", "downloadMdZip": "Markdown (Zip)", + "downloadAss": "ASS", "previewTitle": "Preview", "previewBilingualBtn": "Bilingual", "previewTranslatedOnlyBtn": "Translated Only", @@ -266,49 +267,60 @@ "closeBtn": "Close", "downloadBtn": "Download", "tutorialModalTitle": "Tutorial", - "tutorialModalBody": "

Video tutorials can be found by searching for docutranslate on Bilibili.

Welcome to DocuTranslate! Please follow the steps below to complete your document translation:

  1. Select Workflow

    First, select your desired translation process from the top of the settings panel. Different workflows are suited for different file types:

    • Convert to Markdown then Translate: Suitable for translating files like PDF, Markdown, and images.
    • Plain Text Translation: For translating plain text files like .txt.
    • JSON Translation: For translating specific fields within .json files.
    • DOCX Translation: For translating .docx files.
    • XLSX Translation: For translating .xlsx spreadsheets and .csv files.
    • SRT Subtitle Translation: For translating .srt subtitle files.
    • EPUB Translation: For translating .epub e-book files.
    • HTML Translation: For translating .html files.
    New Feature: The \"Auto-select Workflow\" switch is now on by default. Simply upload your file, and the system will automatically match it with the appropriate workflow, simplifying the process.

  2. Configure Parameters

    Based on your selected workflow, complete the corresponding configurations. All settings are automatically saved in your browser.

    • Parsing Configuration (Only shown for 'Convert to Markdown' workflow):
      • Parsing Engine: Choose an engine to convert your file (like a PDF) into a translation-friendly Markdown format. If your file is already in Markdown, no selection is needed.
      • Mineru Token: If you choose the minerU engine, you need to enter your token here.
    • TXT/DOCX/XLSX/SRT/EPUB/HTML Translation Options (Shown for corresponding workflows):
      • Insert Mode: Defines how the translation result is placed in the document or subtitle. You can choose to 'Replace' the original, 'Append' after it, or 'Prepend' before it.
      • Separator: When 'Append' or 'Prepend' mode is selected, this is used to insert a separator between the original and translated text.
    • JSON Path Configuration (Only shown for 'JSON Translation' workflow):
      • JSON Paths to Translate: Enter one JSONPath expression per line to specify the fields to be translated.
      • For example: $..description translates all values for keys named 'description'. $.items[0].name translates the name of the first item. $.* translates all strings.
    • Translation Model:
      • Skip Translation: If checked, only document parsing and format conversion will be performed, without calling an AI for translation.
      • Select Platform/API Address/API Key/Model ID: Configure the AI translation service you wish to use.
      • Refer to the platform's documentation for Model IDs. It's recommended to use non-inference models or hybrid-inference models (with thinking turned off).
      • The better a model follows instructions, the lower the probability of errors and missed translations.
    • Translation Configuration:
      • Target Language/Custom Prompt/Glossary: Specify the target language for translation, add any additional instructions, and use a glossary to ensure the accuracy of specific term translations.
      • Thinking Mode: Set whether the hybrid inference model should think. Currently supports Zhipu's glm4.5 series, Volcengine's seed1.6 series, SiliconFlow platform, and Google's Gemini series. It is recommended to disable thinking.
      • Chunk Size/Concurrency/Temperature/Retry Count: The size of text chunks sent to the AI, number of concurrent requests, temperature, and retry attempts on failure. Default values are usually fine.
  3. Upload File

    In the task list on the right, click or drag your document into the file upload area.

  4. Start Translation

    Once the file is selected, click the Start Translation button on the bottom right of the task card. The system will begin processing the task, and you can view real-time progress in the log area.

  5. View & Download

    After the translation is complete, action buttons will appear on the task card:

    • Preview: Compare the original and translated text in a slide-out panel on the right (for reference only).
    • Download: Download the translated document in various formats, including PDF, DOCX, XLSX, HTML, and Markdown.
    • Attachments: If any additional files (like a glossary) were generated during the translation, they can be downloaded here.
Tip: All your settings are automatically saved locally in your browser for future use.
", + "tutorialModalBody": "

Video tutorials can be found by searching for docutranslate on Bilibili.

Welcome to DocuTranslate! Please follow these steps to translate your documents:

  1. Select Workflow

    First, select your desired translation process at the top of the settings panel. Different workflows are suitable for different file types:

    • Convert to Markdown then Translate: Suitable for translating PDFs, markdown files, images, etc.
    • Plain Text Translation: For translating plain text files like .txt.
    • JSON Translation: For translating specific fields within .json files.
    • DOCX Translation: For translating .docx files.
    • XLSX Translation: For translating .xlsx spreadsheets or .csv files.
    • SRT Subtitle Translation: For translating .srt subtitle files.
    • EPUB Translation: For translating .epub e-book files.
    • HTML Translation: For translating .html files.
    New Feature: The 'Auto-select Workflow' switch is now enabled by default. Simply upload your file, and the system will automatically match it with the appropriate workflow to simplify the process.
  2. Configure Parameters

    Configure the settings according to your chosen workflow. All settings are automatically saved in your browser.

    • Parsing Config (Only shown for 'Convert to Markdown' workflow):
      • Parsing Engine: Choose an engine to convert your file (e.g., PDF) into a translation-friendly Markdown format. If your file is already in Markdown, no selection is needed.
      • Mineru Token: If you select the minerU engine, you need to enter your token here.
    • TXT/DOCX/XLSX/SRT/EPUB/HTML Translation Options (Shown for corresponding workflows):
      • Insert Mode: Defines how the translation result is placed in the document or subtitle. You can choose to 'Replace' the original, 'Append' after it, or 'Prepend' before it.
      • Separator: When 'Append' or 'Prepend' mode is selected, this is used to insert a separator between the original and translated text.
    • JSON Path Config (Only shown for 'JSON Translation' workflow):
      • JSON Paths to Translate: Enter one JSONPath expression per line to specify the fields to be translated.
      • For example: $..description translates all values for the key 'description'. $.items[0].name translates the name of the first item. $.* translates all strings.
    • Translation Model:
      • Skip Translation: If checked, only document parsing and format conversion will be performed, without calling an AI for translation.
      • Select Platform/API Address/API Key/Model ID: Configure the AI translation service you wish to use.
      • Refer to the platform's documentation for Model IDs. It's recommended to use non-inference or mixed-inference models (with 'thinking' turned off).
      • The better the model's ability to follow instructions, the lower the probability of errors and missed translations.
    • Translation Config:
      • Target Language/Custom Prompt/Glossary: Specify the target language for the translation, add any additional instructions, and use a glossary to ensure the accuracy of specific term translations.
      • Thinking Mode: Sets whether the mixed-inference model should 'think'. Currently supported by Zhipu's glm4.5 series, Volcengine's seed1.6 series, SiliconFlow platform, and Google's Gemini series. Disabling 'think' is recommended.
      • Chunk Size/Concurrency/Temperature/Retry Count: The size of chunks sent to the AI, number of concurrent requests, temperature, and retry attempts on failure. The default values are usually sufficient.
  3. Upload File

    In the task list on the right, click or drag your document into the file upload area.

  4. Start Translation

    Once the file is successfully selected, click the Start Translation button on the bottom right of the task card. The system will begin processing the task, and you can view real-time progress in the log area.

  5. View & Download

    After the translation is complete, action buttons will appear on the task card:

    • Preview: Compare the original and translated text in a side-panel (for reference only).
    • Download: Download the translated document in various formats, including PDF, DOCX, XLSX, HTML, and Markdown.
    • Attachments: If any additional files were generated during the translation (like a glossary), you can download them here.
Tip: All settings are automatically saved in your browser's local storage for future use.
", "tutorialUnderstandBtn": "I Understand", "contributorsModalTitle": "Thanks for Contributing", "contributorsPara1": "DocuTranslate is an open-source project! The community's needs and usage are the driving force behind its progress.", - "contributorsPara2": "Thanks to all the friends who have sponsored the project, submitted code, provided valuable suggestions, and starred the project!", - "contributorsWelcome": "We welcome contributions in the following ways:", + "contributorsPara2": "Thank you to everyone who has sponsored the project, submitted code, provided valuable suggestions, and starred the project!", + "contributorsWelcome": "You are welcome to contribute in the following ways:", "contributorsGithub": "GitHub Homepage", "contributorsPR": "Submit Pull Request", - "contributorsIssue": "Report Issue", + "contributorsIssue": "Report an Issue", "contributorsQQ": "Or contact the author via QQ group: 1047781902", "glossaryModalTitle": "Current Glossary", "glossaryTableSource": "Source (src)", "glossaryTableDestination": "Destination (dst)", "init_i18n_failed_alert": "Failed to load interface translations. Please check your network connection or contact an administrator.", "init_failed_alert": "Initialization failed, could not connect to the backend service. Please ensure the service is running and refresh the page.", - "engineOptionIdentity": "Already in Markdown format", - "engineOptionMineru": "Mineru", - "engineOptionDocling": "Docling", "glossaryEmpty": "Glossary is empty.", + "parsingSettingsTitleText": "Parsing Config", + "txtSettingsTitleText": "TXT Translation Options", + "jsonSettingsTitleText": "JSON Path Config", + "xlsxSettingsTitleText": "XLSX Translation Options", + "docxSettingsTitleText": "DOCX Translation Options", + "srtSettingsTitleText": "SRT Translation Options", + "epubSettingsTitleText": "EPUB Translation Options", + "htmlSettingsTitleText": "HTML Translation Options", + "assSettingsTitleText": "ASS Translation Options", + "aiSettingsTitleText": "Translation Model", + "translationSettingsTitleText": "Translation Config", + "engineOptionIdentity": "Already in Markdown format", + "engineOptionMineru": "minerU", + "engineOptionDocling": "docling", "status_selectFileFirst": "Please select a file first!", "status_fillRequired": "Please fill in all required fields!", "btn_initializing": "Initializing...", "status_encodingAndSubmitting": "Encoding and submitting file...", - "status_requestOk": "Request successful, task has started.", + "status_requestOk": "Request successful, task has started", "btn_cancelTranslation": "Cancel Translation", - "status_requestFail": "Request failed", - "status_initFail": "Initialization failed", + "status_requestFail": "Request Failed", + "status_initFail": "Initialization Failed", "status_cancelling": "Cancelling...", - "status_cancelSent": "Cancel request sent.", - "status_cancelFail": "Cancel failed.", - "btn_reTranslate": "Re-translate", + "status_cancelSent": "Cancellation request sent", + "status_cancelFail": "Cancellation failed", "status_gettingStatus": "Getting status...", - "status_updateError": "Status update error.", - "preview_loading": "Loading preview...", - "preview_cantReadOriginal": "Cannot read original file content.", - "preview_cantPreviewType": "Cannot preview this file type", - "preview_noOriginalCache": "No original file cache, cannot preview.", - "preview_loadFailed": "Preview failed to load.", - "pdf_preparing": "Preparing PDF generation...", - "pdf_print_failed": "Print command failed. Please try right-clicking to print manually.", + "btn_reTranslate": "Re-translate", + "status_updateError": "Failed to update status", + "pdf_preparing": "Preparing PDF...", + "pdf_print_failed": "Failed to print PDF. Please try downloading the HTML manually and printing it from your browser.", "pdf_fetch_failed": "Failed to fetch translated content, cannot generate PDF.", "preview_bilingual": "Bilingual Preview", - "preview_translatedOnly": "Translated Only Preview", - "admin_tasklist_failed": "Admin Mode: Failed to load task list" + "preview_translatedOnly": "Translated-Only Preview", + "preview_loading": "Loading preview...", + "preview_cantReadOriginal": "Could not read original file content for preview.", + "preview_cantPreviewType": "Cannot preview this file type", + "preview_noOriginalCache": "Original file is not available after page refresh. Please re-upload to preview.", + "preview_loadFailed": "Failed to load preview.", + "admin_tasklist_failed": "Admin Mode: Failed to load task list." } } \ No newline at end of file diff --git a/docutranslate/static/index.html b/docutranslate/static/index.html index f09f15d..ca79d4a 100644 --- a/docutranslate/static/index.html +++ b/docutranslate/static/index.html @@ -1 +1 @@ - DocuTranslate - 交互式文档翻译

DocuTranslate

如果上传的文件本身是.md格式,此项可不选。
mineru VLM是更新的内测模型。

Base URL:

选择一个或多个CSV文件。文件需包含'src'和'dst'两列标题,分别代表原文和译文。

GitHub主页(欢迎star❤):
https://github.com/xunbu/docutranslate

交流QQ群: 1047781902

version:

任务列表

LOGO

当前没有任务,点击“新建任务”开始吧!

预览
原文
译文
\ No newline at end of file + DocuTranslate - 交互式文档翻译

DocuTranslate

如果上传的文件本身是.md格式,此项可不选。
mineru VLM是更新的内测模型。

Base URL:

选择一个或多个CSV文件。文件需包含'src'和'dst'两列标题,分别代表原文和译文。

GitHub主页(欢迎star❤):
https://github.com/xunbu/docutranslate

交流QQ群: 1047781902

version:

任务列表

LOGO

当前没有任务,点击“新建任务”开始吧!

预览
原文
译文
\ No newline at end of file diff --git a/docutranslate/template/ass.html b/docutranslate/template/ass.html new file mode 100644 index 0000000..6ea0af1 --- /dev/null +++ b/docutranslate/template/ass.html @@ -0,0 +1,16 @@ + + + + + ass subtitle + + + +
{{ ass_data }}
+ + \ No newline at end of file diff --git a/docutranslate/translator/ai_translator/ass_translator.py b/docutranslate/translator/ai_translator/ass_translator.py new file mode 100644 index 0000000..d5a8c3d --- /dev/null +++ b/docutranslate/translator/ai_translator/ass_translator.py @@ -0,0 +1,136 @@ +# SPDX-FileCopyrightText: 2025 QinHan +# SPDX-License-Identifier: MPL-2.0 + +import asyncio +from dataclasses import dataclass +from typing import Self, Literal, List, Optional + +import pysubs2 + +from docutranslate.agents.segments_agent import SegmentsTranslateAgentConfig, SegmentsTranslateAgent +from docutranslate.ir.document import Document +from docutranslate.translator.ai_translator.base import AiTranslatorConfig, AiTranslator + + +@dataclass +class AssTranslatorConfig(AiTranslatorConfig): + insert_mode: Literal["replace", "append", "prepend"] = "replace" + separator: str = "\\N" # ASS 中换行符是 \N + # 未来可扩展:指定样式名或时间范围,当前暂不实现,翻译所有 Dialogue + translate_regions: Optional[List[str]] = None # 暂保留接口,但当前忽略 + + +class AssTranslator(AiTranslator): + def __init__(self, config: AssTranslatorConfig): + super().__init__(config=config) + self.chunk_size = config.chunk_size + self.translate_agent = None + if not self.skip_translate: + agent_config = SegmentsTranslateAgentConfig( + custom_prompt=config.custom_prompt, + to_lang=config.to_lang, + base_url=config.base_url, + api_key=config.api_key, + model_id=config.model_id, + temperature=config.temperature, + thinking=config.thinking, + concurrent=config.concurrent, + timeout=config.timeout, + logger=self.logger, + glossary_dict=config.glossary_dict, + retry=config.retry + ) + self.translate_agent = SegmentsTranslateAgent(agent_config) + self.insert_mode = config.insert_mode + self.separator = config.separator + self.translate_regions = config.translate_regions # 暂不处理,保留接口 + + def _pre_translate(self, document: Document): + """ + 解析 ASS 文件,提取所有 Dialogue 行的文本。 + 返回:subs 对象、待翻译条目列表、原文列表 + """ + try: + content_str = document.content.decode('utf-8-sig') # ASS 通常带 BOM + except UnicodeDecodeError: + content_str = document.content.decode('utf-8') + + subs = pysubs2.SSAFile.from_string(content_str) + lines_to_translate = [] + + for i, line in enumerate(subs): + if line.type == "Dialogue": + # 仅翻译文本部分,保留样式、时间等 + if isinstance(line.text, str) and line.text.strip(): + lines_to_translate.append({ + "index": i, # 记录在 subs 中的位置 + "original_text": line.text, + "line": line # 保留引用,便于后续修改 + }) + + original_texts = [item["original_text"] for item in lines_to_translate] + return subs, lines_to_translate, original_texts + + def _after_translate(self, subs, lines_to_translate, translated_texts, original_texts): + """ + 将翻译结果写回 ASS 对象,根据 insert_mode 处理。 + """ + for i, item in enumerate(lines_to_translate): + line = item["line"] + translated_text = translated_texts[i] + original_text = original_texts[i] + + if self.insert_mode == "replace": + line.text = translated_text + elif self.insert_mode == "append": + line.text = original_text + self.separator + translated_text + elif self.insert_mode == "prepend": + line.text = translated_text + self.separator + original_text + else: + self.logger.error(f"不支持的插入模式: {self.insert_mode}") + + # 输出为字符串,再编码为 bytes + output_str = subs.to_string(format_="ass") + return output_str.encode('utf-8-sig') # 带 BOM,兼容播放器 + + def translate(self, document: Document) -> Self: + subs, lines_to_translate, original_texts = self._pre_translate(document) + + if not lines_to_translate: + print("\n未找到需要翻译的字幕行。") + return self + + if self.glossary_agent: + self.glossary_dict_gen = self.glossary_agent.send_segments(original_texts, self.chunk_size) + if self.translate_agent: + self.translate_agent.update_glossary_dict(self.glossary_dict_gen) + + if self.translate_agent: + translated_texts = self.translate_agent.send_segments(original_texts, self.chunk_size) + else: + translated_texts = original_texts + + document.content = self._after_translate(subs, lines_to_translate, translated_texts, original_texts) + return self + + async def translate_async(self, document: Document) -> Self: + subs, lines_to_translate, original_texts = await asyncio.to_thread(self._pre_translate, document) + + if not lines_to_translate: + print("\n未找到需要翻译的字幕行。") + return self + + if self.glossary_agent: + self.glossary_dict_gen = await self.glossary_agent.send_segments_async(original_texts, self.chunk_size) + if self.translate_agent: + self.translate_agent.update_glossary_dict(self.glossary_dict_gen) + + if self.translate_agent: + translated_texts = await self.translate_agent.send_segments_async(original_texts, self.chunk_size) + else: + translated_texts = original_texts + + document.content = await asyncio.to_thread( + self._after_translate, subs, lines_to_translate, translated_texts, original_texts + ) + return self diff --git a/docutranslate/workflow/ass_workflow.py b/docutranslate/workflow/ass_workflow.py new file mode 100644 index 0000000..5dcd68a --- /dev/null +++ b/docutranslate/workflow/ass_workflow.py @@ -0,0 +1,76 @@ +# SPDX-FileCopyrightText: 2025 QinHan +# SPDX-License-Identifier: MPL-2.0 +from dataclasses import dataclass +from pathlib import Path +from typing import Self + +from docutranslate.exporter.ass.ass2ass_exporter import Ass2AssExporter +from docutranslate.exporter.ass.ass2html_exporter import Ass2HTMLExporterConfig, Ass2HTMLExporter +from docutranslate.exporter.base import ExporterConfig +from docutranslate.glossary.glossary import Glossary +from docutranslate.ir.document import Document +from docutranslate.translator.ai_translator.ass_translator import AssTranslatorConfig, AssTranslator +from docutranslate.workflow.base import WorkflowConfig, Workflow +from docutranslate.workflow.interfaces import HTMLExportable, AssExportable + + + + + +@dataclass(kw_only=True) +class AssWorkflowConfig(WorkflowConfig): + translator_config: AssTranslatorConfig + html_exporter_config: Ass2HTMLExporterConfig + + +class AssWorkflow(Workflow[AssWorkflowConfig, Document, Document], HTMLExportable[Ass2HTMLExporterConfig], + AssExportable[ExporterConfig]): + def __init__(self, config: AssWorkflowConfig): + super().__init__(config=config) + if config.logger: + for sub_config in [self.config.translator_config]: + if sub_config: + sub_config.logger = config.logger + + def _pre_translate(self,document_original:Document): + document = document_original.copy() + translate_config = self.config.translator_config + translator = AssTranslator(translate_config) + return document,translator + + + def translate(self) -> Self: + document, translator=self._pre_translate(self.document_original) + translator.translate(document) + if translator.glossary_dict_gen: + self.attachment.add_document("glossary", Glossary.glossary_dict2csv(translator.glossary_dict_gen)) + self.document_translated = document + return self + + async def translate_async(self) -> Self: + document, translator = self._pre_translate(self.document_original) + await translator.translate_async(document) + if translator.glossary_dict_gen: + self.attachment.add_document("glossary", Glossary.glossary_dict2csv(translator.glossary_dict_gen)) + self.document_translated = document + return self + + def export_to_html(self, config: Ass2HTMLExporterConfig = None) -> str: + config = config or self.config.html_exporter_config + docu = self._export(Ass2HTMLExporter(config)) + return docu.content.decode() + + def export_to_ass(self, _: ExporterConfig | None = None) -> str: + docu = self._export(Ass2AssExporter()) + return docu.content.decode() + + def save_as_html(self, name: str = None, output_dir: Path | str = "./output", + config: Ass2HTMLExporterConfig | None = None) -> Self: + config = config or self.config.html_exporter_config + self._save(exporter=Ass2HTMLExporter(config), name=name, output_dir=output_dir) + return self + + def save_as_ass(self, name: str = None, output_dir: Path | str = "./output", + _: ExporterConfig | None = None) -> Self: + self._save(exporter=Ass2AssExporter(), name=name, output_dir=output_dir) + return self diff --git a/docutranslate/workflow/interfaces.py b/docutranslate/workflow/interfaces.py index ca0d4cd..28311fd 100644 --- a/docutranslate/workflow/interfaces.py +++ b/docutranslate/workflow/interfaces.py @@ -103,3 +103,11 @@ class EpubExportable(Protocol[T_ExporterConfig]): def save_as_epub(self, name: str, output_dir: Path | str, config: T_ExporterConfig | None = None) -> Self: ... + +@runtime_checkable +class AssExportable(Protocol[T_ExporterConfig]): + def export_to_ass(self, config: T_ExporterConfig | None = None) -> str: + ... + + def save_as_ass(self, name: str, output_dir: Path | str, config: T_ExporterConfig | None = None) -> Self: + ... diff --git a/pyproject.toml b/pyproject.toml index c056036..41152d9 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -18,6 +18,8 @@ dependencies = [ "markdown>=3.8.2", "pymdown-extensions>=10.16.1", "chardet>=5.2.0", + "py>=1.11.0", + "pysubs2>=1.8.0", ] dynamic = ["version"] diff --git a/uv.lock b/uv.lock index 3ec8f0b..f4335e0 100644 --- a/uv.lock +++ b/uv.lock @@ -24,15 +24,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/9f/1c/a17fb513aeb684fb83bef5f395910f53103ab30308bbdd77fd66d6698c46/accelerate-1.9.0-py3-none-any.whl", hash = "sha256:c24739a97ade1d54af4549a65f8b6b046adc87e2b3e4d6c66516e32c53d5a8f1", size = 367073 }, ] -[[package]] -name = "altgraph" -version = "0.17.4" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/de/a8/7145824cf0b9e3c28046520480f207df47e927df83aa9555fb47f8505922/altgraph-0.17.4.tar.gz", hash = "sha256:1b5afbb98f6c4dcadb2e2ae6ab9fa994bbb8c1d75f4fa96d340f9437ae454406", size = 48418 } -wheels = [ - { url = "https://files.pythonhosted.org/packages/4d/3f/3bc3f1d83f6e4a7fcb834d3720544ca597590425be5ba9db032b2bf322a2/altgraph-0.17.4-py2.py3-none-any.whl", hash = "sha256:642743b4750de17e655e6711601b077bc6598dbfa3ba5fa2b2a35ce12b508dff", size = 21212 }, -] - [[package]] name = "annotated-types" version = "0.7.0" @@ -334,7 +325,9 @@ dependencies = [ { name = "mammoth" }, { name = "markdown" }, { name = "openpyxl" }, + { name = "py" }, { name = "pymdown-extensions" }, + { name = "pysubs2" }, { name = "python-docx" }, { name = "srt" }, { name = "xlsx2html" }, @@ -350,7 +343,6 @@ docling = [ dev = [ { name = "docling" }, { name = "opencv-python" }, - { name = "pyinstaller" }, ] [package.metadata] @@ -367,7 +359,9 @@ requires-dist = [ { name = "markdown", specifier = ">=3.8.2" }, { name = "opencv-python", marker = "extra == 'docling'", specifier = ">=4.11.0.86" }, { name = "openpyxl", specifier = ">=3.1.5" }, + { name = "py", specifier = ">=1.11.0" }, { name = "pymdown-extensions", specifier = ">=10.16.1" }, + { name = "pysubs2", specifier = ">=1.8.0" }, { name = "python-docx", specifier = ">=1.2.0" }, { name = "srt", specifier = ">=3.5.3" }, { name = "xlsx2html", specifier = ">=0.6.2" }, @@ -378,7 +372,6 @@ provides-extras = ["docling"] dev = [ { name = "docling", specifier = ">=2.40.0" }, { name = "opencv-python", specifier = ">=4.11.0.86" }, - { name = "pyinstaller", specifier = ">=6.14.2" }, ] [[package]] @@ -798,18 +791,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/fc/14/c115516c62a7d2499781d2d3d7215218c0731b2c940753bf9f9b7b73924d/lxml-5.4.0-cp313-cp313-win_amd64.whl", hash = "sha256:bcb7a1096b4b6b24ce1ac24d4942ad98f983cd3810f9711bcd0293f43a9d8b9f", size = 3814606 }, ] -[[package]] -name = "macholib" -version = "1.16.3" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "altgraph" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/95/ee/af1a3842bdd5902ce133bd246eb7ffd4375c38642aeb5dc0ae3a0329dfa2/macholib-1.16.3.tar.gz", hash = "sha256:07ae9e15e8e4cd9a788013d81f5908b3609aa76f9b1421bae9c4d7606ec86a30", size = 59309 } -wheels = [ - { url = "https://files.pythonhosted.org/packages/d1/5d/c059c180c84f7962db0aeae7c3b9303ed1d73d76f2bfbc32bc231c8be314/macholib-1.16.3-py2.py3-none-any.whl", hash = "sha256:0e315d7583d38b8c77e815b1ecbdbf504a8258d8b3e17b61165c6feb60d18f2c", size = 38094 }, -] - [[package]] name = "mammoth" version = "1.10.0" @@ -1267,15 +1248,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/d5/f9/07086f5b0f2a19872554abeea7658200824f5835c58a106fa8f2ae96a46c/pandas-2.3.1-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:5db9637dbc24b631ff3707269ae4559bce4b7fd75c1c4d7e13f40edc42df4444", size = 13189044 }, ] -[[package]] -name = "pefile" -version = "2023.2.7" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/78/c5/3b3c62223f72e2360737fd2a57c30e5b2adecd85e70276879609a7403334/pefile-2023.2.7.tar.gz", hash = "sha256:82e6114004b3d6911c77c3953e3838654b04511b8b66e8583db70c65998017dc", size = 74854 } -wheels = [ - { url = "https://files.pythonhosted.org/packages/55/26/d0ad8b448476d0a1e8d3ea5622dc77b916db84c6aa3cb1e1c0965af948fc/pefile-2023.2.7-py3-none-any.whl", hash = "sha256:da185cd2af68c08a6cd4481f7325ed600a88f6a813bad9dea07ab3ef73d8d8d6", size = 71791 }, -] - [[package]] name = "pillow" version = "11.3.0" @@ -1393,6 +1365,15 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/50/1b/6921afe68c74868b4c9fa424dad3be35b095e16687989ebbb50ce4fceb7c/psutil-7.0.0-cp37-abi3-win_amd64.whl", hash = "sha256:4cf3d4eb1aa9b348dec30105c55cd9b7d4629285735a102beb4441e38db90553", size = 244885 }, ] +[[package]] +name = "py" +version = "1.11.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/98/ff/fec109ceb715d2a6b4c4a85a61af3b40c723a961e8828319fbcb15b868dc/py-1.11.0.tar.gz", hash = "sha256:51c75c4126074b472f746a24399ad32f6053d1b34b68d2fa41e558e6f4a98719", size = 207796 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f6/f0/10642828a8dfb741e5f3fbaac830550a518a775c7fff6f04a007259b0548/py-1.11.0-py2.py3-none-any.whl", hash = "sha256:607c53218732647dff4acdfcd50cb62615cedf612e72d1724fb1a0cc6405b378", size = 98708 }, +] + [[package]] name = "pyclipper" version = "1.3.0.post6" @@ -1527,47 +1508,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/c7/21/705964c7812476f378728bdf590ca4b771ec72385c533964653c68e86bdc/pygments-2.19.2-py3-none-any.whl", hash = "sha256:86540386c03d588bb81d44bc3928634ff26449851e99741617ecb9037ee5ec0b", size = 1225217 }, ] -[[package]] -name = "pyinstaller" -version = "6.15.0" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "altgraph" }, - { name = "macholib", marker = "sys_platform == 'darwin'" }, - { name = "packaging" }, - { name = "pefile", marker = "sys_platform == 'win32'" }, - { name = "pyinstaller-hooks-contrib" }, - { name = "pywin32-ctypes", marker = "sys_platform == 'win32'" }, - { name = "setuptools" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/64/17/b2bb4de22650adbeef401fa82a1b43028976547a8728602e4d29735b455e/pyinstaller-6.15.0.tar.gz", hash = "sha256:a48fc4644ee4aa2aa2a35e7b51f496f8fbd7eecf6a2150646bbf1613ad07bc2d", size = 4331521 } -wheels = [ - { url = "https://files.pythonhosted.org/packages/24/dd/d5c8a127446adda954f68ea7fac22772f7ab8656ad4b06df396d82574ca9/pyinstaller-6.15.0-py3-none-macosx_10_13_universal2.whl", hash = "sha256:9f00c71c40148cd1e61695b2c6f1e086693d3bcf9bfa22ab513aa4254c3b966f", size = 1016981 }, - { url = "https://files.pythonhosted.org/packages/2d/2a/7b50593b419db43e48d9bdeebaac0ff92a5fe035f3c30f87ca3e1650d7e2/pyinstaller-6.15.0-py3-none-manylinux2014_aarch64.whl", hash = "sha256:cbcc8eb77320c60722030ac875883b564e00768fe3ff1721c7ba3ad0e0a277e9", size = 726337 }, - { url = "https://files.pythonhosted.org/packages/77/83/7f498fba0154c57eb5fc93eb9680a2dbadb9f780a3389fb85b8d79683378/pyinstaller-6.15.0-py3-none-manylinux2014_i686.whl", hash = "sha256:c33e6302bc53db2df1104ed5566bd980b3e0ee7f18416a6e3caa908c12a54542", size = 737539 }, - { url = "https://files.pythonhosted.org/packages/09/d6/e4477feab7c8379fb49e7ec95c82d0a69ad88f6ccc247f76bef3cb0e3432/pyinstaller-6.15.0-py3-none-manylinux2014_ppc64le.whl", hash = "sha256:eb902d0fed3bb1f8b7190dc4df5c11f3b59505767e0d56d1ed782b853938bbf3", size = 735426 }, - { url = "https://files.pythonhosted.org/packages/32/7e/ff25648276f15e2e77fc563d36d8cfcd917e077bf2a172420df3588601b4/pyinstaller-6.15.0-py3-none-manylinux2014_s390x.whl", hash = "sha256:b4df862adae7cf1f08eff53c43ace283822447f7f528f72e4f94749062712f15", size = 732210 }, - { url = "https://files.pythonhosted.org/packages/db/3d/267a7dddd0647de95d260780050ccd8228ab29d2b9edea54ed1f56800967/pyinstaller-6.15.0-py3-none-manylinux2014_x86_64.whl", hash = "sha256:b9ebf16ed0f99016ae8ae5746dee4cb244848a12941539e62ce2eea1df5a3f95", size = 732194 }, - { url = "https://files.pythonhosted.org/packages/4d/61/962b2eb79ef225233e2d6e04600e998935328011dfb2fa775b1dd16b943a/pyinstaller-6.15.0-py3-none-musllinux_1_1_aarch64.whl", hash = "sha256:22193489e6a22435417103f61e7950363bba600ef36ec3ab1487303668c81092", size = 731256 }, - { url = "https://files.pythonhosted.org/packages/67/5e/4e20e1c0e5791b09b69bef3ac921fd0cd25551b56879324ad999b92fa045/pyinstaller-6.15.0-py3-none-musllinux_1_1_x86_64.whl", hash = "sha256:18f743069849dbaee3e10900385f35795a5743eabab55e99dcc42f204e40a0db", size = 731148 }, - { url = "https://files.pythonhosted.org/packages/88/31/28956c534991f289e2f981c715730b6241e75dc6295737a8cbd050a0cc8c/pyinstaller-6.15.0-py3-none-win32.whl", hash = "sha256:60da8f1b5071766b45c0f607d8bc3d7e59ba2c3b262d08f2e4066ba65f3544a2", size = 1312297 }, - { url = "https://files.pythonhosted.org/packages/09/ab/6a45186c7f8e34c422faecd72580116a67d068158c57faa2d2f6d01faa7f/pyinstaller-6.15.0-py3-none-win_amd64.whl", hash = "sha256:cbea297e16eeda30b41c300d6ec2fd2abea4dbd8d8a32650eeec36431c94fcd9", size = 1373091 }, - { url = "https://files.pythonhosted.org/packages/5b/86/72159af032b9db36f2470a3b085f79277ec1c38e7e48f8c5dc1ed16dc4e1/pyinstaller-6.15.0-py3-none-win_arm64.whl", hash = "sha256:f43c035621742cf2d19b84308c60e4e44e72c94786d176b8f6adcde351b5bd98", size = 1314305 }, -] - -[[package]] -name = "pyinstaller-hooks-contrib" -version = "2025.8" -source = { registry = "https://pypi.org/simple" } -dependencies = [ - { name = "packaging" }, - { name = "setuptools" }, -] -sdist = { url = "https://files.pythonhosted.org/packages/71/d6/e5b378b7d4add8c879295c531309b0320e9c07a70458665d091760ffdc87/pyinstaller_hooks_contrib-2025.8.tar.gz", hash = "sha256:3402ad41dfe9b5110af134422e37fc5d421ba342c6cb980bd67cb30b7415641c", size = 164214 } -wheels = [ - { url = "https://files.pythonhosted.org/packages/48/34/1d973d0dae849683e53fbcda84443ce016f315e6f4dc7605ede4f56a28c3/pyinstaller_hooks_contrib-2025.8-py3-none-any.whl", hash = "sha256:8d0b8cfa0cb689a619294ae200497374234bd4e3994b3ace2a4442274c899064", size = 442346 }, -] - [[package]] name = "pylatexenc" version = "2.10" @@ -1607,6 +1547,15 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/be/7a/097801205b991bc3115e8af1edb850d30aeaf0118520b016354cf5ccd3f6/pypdfium2-4.30.0-py3-none-win_arm64.whl", hash = "sha256:119b2969a6d6b1e8d55e99caaf05290294f2d0fe49c12a3f17102d01c441bd29", size = 2752118 }, ] +[[package]] +name = "pysubs2" +version = "1.8.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/31/4a/becf78d9d3df56e6c4a9c50b83794e5436b6c5ab6dd8a3f934e94c89338c/pysubs2-1.8.0.tar.gz", hash = "sha256:3397bb58a4a15b1325ba2ae3fd4d7c214e2c0ddb9f33190d6280d783bb433b20", size = 1130048 } +wheels = [ + { url = "https://files.pythonhosted.org/packages/99/09/0fc0719162e5ad723f71d41cf336f18b6b5054d70dc0fe42ace6b4d2bdc9/pysubs2-1.8.0-py3-none-any.whl", hash = "sha256:05716f5039a9ebe32cd4d7673f923cf36204f3a3e99987f823ab83610b7035a0", size = 43516 }, +] + [[package]] name = "python-bidi" version = "0.6.6" @@ -1743,15 +1692,6 @@ wheels = [ { url = "https://files.pythonhosted.org/packages/c0/d2/21af5c535501a7233e734b8af901574572da66fcc254cb35d0609c9080dd/pywin32-311-cp314-cp314-win_arm64.whl", hash = "sha256:a508e2d9025764a8270f93111a970e1d0fbfc33f4153b388bb649b7eec4f9b42", size = 8932540 }, ] -[[package]] -name = "pywin32-ctypes" -version = "0.2.3" -source = { registry = "https://pypi.org/simple" } -sdist = { url = "https://files.pythonhosted.org/packages/85/9f/01a1a99704853cb63f253eea009390c88e7131c67e66a0a02099a8c917cb/pywin32-ctypes-0.2.3.tar.gz", hash = "sha256:d162dc04946d704503b2edc4d55f3dba5c1d539ead017afa00142c38b9885755", size = 29471 } -wheels = [ - { url = "https://files.pythonhosted.org/packages/de/3d/8161f7711c017e01ac9f008dfddd9410dff3674334c233bde66e7ba65bbf/pywin32_ctypes-0.2.3-py3-none-any.whl", hash = "sha256:8a1513379d709975552d202d942d9837758905c8d01eb82b8bcc30918929e7b8", size = 30756 }, -] - [[package]] name = "pyyaml" version = "6.0.2"