feat: 添加 OCR 词汇提取功能

新增 OCR 页面,用户可上传教材词汇表截图,使用 GLM-4.6V 视觉模型
提取单词-释义对并保存到指定文件夹。

- AI 管道: src/lib/bigmodel/ocr/ (orchestrator, types)
- 后端模块: src/modules/ocr/ (action-service-repository 架构)
- 前端页面: src/app/(features)/ocr/ (拖拽上传、folder 选择)
- i18n: 8 种语言翻译支持
This commit is contained in:
2026-03-10 15:21:45 +08:00
parent 683a4104ec
commit 9b78fd5215
18 changed files with 868 additions and 0 deletions

View File

@@ -197,6 +197,35 @@
"favorites": "Favoriten",
"settings": "Einstellungen"
},
"ocr": {
"title": "OCR Vokabel-Extraktion",
"description": "Laden Sie Screenshots von Vokabeltabellen aus Lehrbüchern hoch, um Wort-Definition-Paare zu extrahieren",
"uploadImage": "Bild hochladen",
"dragDropHint": "Ziehen Sie ein Bild hierher oder klicken Sie zum Auswählen",
"supportedFormats": "Unterstützt: JPG, PNG, WebP",
"selectFolder": "Ordner auswählen",
"chooseFolder": "Wählen Sie einen Ordner zum Speichern der extrahierten Paare",
"noFolders": "Keine Ordner verfügbar. Bitte erstellen Sie zuerst einen Ordner.",
"languageHints": "Sprachhinweise (Optional)",
"sourceLanguageHint": "Quellsprache (z.B. Englisch)",
"targetLanguageHint": "Ziel-/Übersetzungssprache (z.B. Chinesisch)",
"process": "Bild verarbeiten",
"processing": "Verarbeitung...",
"preview": "Vorschau",
"extractedPairs": "Extrahierte Paare",
"word": "Wort",
"definition": "Definition",
"pairsCount": "{count} Paare extrahiert",
"savePairs": "In Ordner speichern",
"saving": "Speichern...",
"saved": "{count} Paare erfolgreich in {folder} gespeichert",
"saveFailed": "Speichern fehlgeschlagen",
"noImage": "Bitte laden Sie zuerst ein Bild hoch",
"noFolder": "Bitte wählen Sie einen Ordner",
"processingFailed": "OCR-Verarbeitung fehlgeschlagen",
"tryAgain": "Bitte versuchen Sie es mit einem klareren Bild",
"detectedLanguages": "Erkannt: {source} → {target}"
},
"profile": {
"myProfile": "Mein Profil",
"email": "E-Mail: {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "Favorites",
"settings": "Settings"
},
"ocr": {
"title": "OCR Vocabulary Extractor",
"description": "Upload vocabulary table screenshots from textbooks to extract word-definition pairs",
"uploadImage": "Upload Image",
"dragDropHint": "Drag and drop an image here, or click to select",
"supportedFormats": "Supports: JPG, PNG, WebP",
"selectFolder": "Select Folder",
"chooseFolder": "Choose a folder to save extracted pairs",
"noFolders": "No folders available. Please create a folder first.",
"languageHints": "Language Hints (Optional)",
"sourceLanguageHint": "Source language (e.g., English)",
"targetLanguageHint": "Target/Translation language (e.g., Chinese)",
"process": "Process Image",
"processing": "Processing...",
"preview": "Preview",
"extractedPairs": "Extracted Pairs",
"word": "Word",
"definition": "Definition",
"pairsCount": "{count} pairs extracted",
"savePairs": "Save to Folder",
"saving": "Saving...",
"saved": "Successfully saved {count} pairs to {folder}",
"saveFailed": "Failed to save pairs",
"noImage": "Please upload an image first",
"noFolder": "Please select a folder",
"processingFailed": "OCR processing failed",
"tryAgain": "Please try again with a clearer image",
"detectedLanguages": "Detected: {source} → {target}"
},
"profile": {
"myProfile": "My Profile",
"email": "Email: {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "Favoris",
"settings": "Paramètres"
},
"ocr": {
"title": "Extraction OCR de vocabulaire",
"description": "Téléchargez des captures d'écran de tableaux de vocabulaire pour extraire les paires mot-définition",
"uploadImage": "Télécharger une image",
"dragDropHint": "Glissez-déposez une image ici, ou cliquez pour sélectionner",
"supportedFormats": "Supportés : JPG, PNG, WebP",
"selectFolder": "Sélectionner un dossier",
"chooseFolder": "Choisissez un dossier pour sauvegarder les paires extraites",
"noFolders": "Aucun dossier disponible. Veuillez d'abord créer un dossier.",
"languageHints": "Indices de langue (Optionnel)",
"sourceLanguageHint": "Langue source (ex : Anglais)",
"targetLanguageHint": "Langue cible/traduction (ex : Chinois)",
"process": "Traiter l'image",
"processing": "Traitement...",
"preview": "Aperçu",
"extractedPairs": "Paires extraites",
"word": "Mot",
"definition": "Définition",
"pairsCount": "{count} paires extraites",
"savePairs": "Sauvegarder dans le dossier",
"saving": "Sauvegarde...",
"saved": "{count} paires sauvegardées dans {folder}",
"saveFailed": "Échec de la sauvegarde",
"noImage": "Veuillez d'abord télécharger une image",
"noFolder": "Veuillez sélectionner un dossier",
"processingFailed": "Échec du traitement OCR",
"tryAgain": "Veuillez réessayer avec une image plus claire",
"detectedLanguages": "Détecté : {source} → {target}"
},
"profile": {
"myProfile": "Mon profil",
"email": "E-mail : {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "Preferiti",
"settings": "Impostazioni"
},
"ocr": {
"title": "Estrazione vocaboli OCR",
"description": "Carica screenshot di tabelle di vocaboli per estrarre coppie parola-definizione",
"uploadImage": "Carica immagine",
"dragDropHint": "Trascina e rilascia un'immagine qui, o clicca per selezionare",
"supportedFormats": "Supportati: JPG, PNG, WebP",
"selectFolder": "Seleziona cartella",
"chooseFolder": "Scegli una cartella per salvare le coppie estratte",
"noFolders": "Nessuna cartella disponibile. Crea prima una cartella.",
"languageHints": "Suggerimenti lingua (Opzionale)",
"sourceLanguageHint": "Lingua sorgente (es: Inglese)",
"targetLanguageHint": "Lingua target/traduzione (es: Cinese)",
"process": "Elabora immagine",
"processing": "Elaborazione...",
"preview": "Anteprima",
"extractedPairs": "Coppie estratte",
"word": "Parola",
"definition": "Definizione",
"pairsCount": "{count} coppie estratte",
"savePairs": "Salva nella cartella",
"saving": "Salvataggio...",
"saved": "{count} coppie salvate in {folder}",
"saveFailed": "Salvataggio fallito",
"noImage": "Carica prima un'immagine",
"noFolder": "Seleziona una cartella",
"processingFailed": "Elaborazione OCR fallita",
"tryAgain": "Riprova con un'immagine più chiara",
"detectedLanguages": "Rilevato: {source} → {target}"
},
"profile": {
"myProfile": "Il Mio Profilo",
"email": "Email: {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "お気に入り",
"settings": "設定"
},
"ocr": {
"title": "OCR語彙抽出",
"description": "教科書の語彙表のスクリーンショットをアップロードして単語と定義のペアを抽出",
"uploadImage": "画像をアップロード",
"dragDropHint": "ここに画像をドラッグ&ドロップ、またはクリックして選択",
"supportedFormats": "対応形式JPG、PNG、WebP",
"selectFolder": "フォルダを選択",
"chooseFolder": "抽出したペアを保存するフォルダを選択",
"noFolders": "フォルダがありません。まずフォルダを作成してください。",
"languageHints": "言語ヒント(オプション)",
"sourceLanguageHint": "ソース言語(例:英語)",
"targetLanguageHint": "ターゲット/翻訳言語(例:中国語)",
"process": "画像を処理",
"processing": "処理中...",
"preview": "プレビュー",
"extractedPairs": "抽出されたペア",
"word": "単語",
"definition": "定義",
"pairsCount": "{count} ペアを抽出",
"savePairs": "フォルダに保存",
"saving": "保存中...",
"saved": "{count} ペアを {folder} に保存しました",
"saveFailed": "保存に失敗しました",
"noImage": "先に画像をアップロードしてください",
"noFolder": "フォルダを選択してください",
"processingFailed": "OCR処理に失敗しました",
"tryAgain": "より鮮明な画像でお試しください",
"detectedLanguages": "検出:{source} → {target}"
},
"profile": {
"myProfile": "マイプロフィール",
"email": "メール: {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "즐겨찾기",
"settings": "설정"
},
"ocr": {
"title": "OCR 어휘 추출",
"description": "교과서 어휘표 스크린샷을 업로드하여 단어-정의 쌍 추출",
"uploadImage": "이미지 업로드",
"dragDropHint": "이미지를 여기에 끌어다 놓거나 클릭하여 선택",
"supportedFormats": "지원 형식: JPG, PNG, WebP",
"selectFolder": "폴더 선택",
"chooseFolder": "추출된 쌍을 저장할 폴더 선택",
"noFolders": "폴더가 없습니다. 먼저 폴더를 만드세요.",
"languageHints": "언어 힌트 (선택사항)",
"sourceLanguageHint": "소스 언어 (예: 영어)",
"targetLanguageHint": "대상/번역 언어 (예: 중국어)",
"process": "이미지 처리",
"processing": "처리 중...",
"preview": "미리보기",
"extractedPairs": "추출된 쌍",
"word": "단어",
"definition": "정의",
"pairsCount": "{count} 쌍 추출됨",
"savePairs": "폴더에 저장",
"saving": "저장 중...",
"saved": "{folder}에 {count} 쌍 저장 완료",
"saveFailed": "저장 실패",
"noImage": "먼저 이미지를 업로드하세요",
"noFolder": "폴더를 선택하세요",
"processingFailed": "OCR 처리 실패",
"tryAgain": "더 선명한 이미지로 다시 시도하세요",
"detectedLanguages": "감지됨: {source} → {target}"
},
"profile": {
"myProfile": "내 프로필",
"email": "이메일: {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "يىغىپ ساقلاش",
"settings": "تەڭشەكلەر"
},
"ocr": {
"title": "OCR سۆز ئاستىرىش",
"description": "دەرىسلىك كىتابىدىكى سۆز جەدۋىلى سۈرەتلىرىنى يۈكلەپ سۆز-مەنا جۈپلىرىنى ئاستىرىڭ",
"uploadImage": "سۈرەت يۈكلەش",
"dragDropHint": "سۈرەتنى بۇ يەرگە سۆرەڭ ياكى چېكىپ تاللاڭ",
"supportedFormats": "قوللايدىغان فورماتلار: JPG، PNG، WebP",
"selectFolder": "قىسقۇچ تاللاش",
"chooseFolder": "ئاستىرىلغان جۈپلەرنى ساقلاش ئۈچۈن قىسقۇچ تاللاڭ",
"noFolders": "قىسقۇچ يوق. ئاۋۋال قىسقۇچ قۇرۇڭ.",
"languageHints": "تىل ئۇچۇرلىرى (ئىختىيارىي)",
"sourceLanguageHint": "مەنبە تىلى (مىسال: ئىنگىلىزچە)",
"targetLanguageHint": "نىشان/تەرجىمە تىلى (مىسال: خەنزۇچە)",
"process": "سۈرەتنى بىر تەرەپ قىلىش",
"processing": "بىر تەرەپ قىلىۋاتىدۇ...",
"preview": "ئالدىن كۆرۈش",
"extractedPairs": "ئاستىرىلغان جۈپلەر",
"word": "سۆز",
"definition": "مەنا",
"pairsCount": "{count} جۈپ ئاستىرىلدى",
"savePairs": "قىسقۇچقا ساقلاش",
"saving": "ساقلاۋاتىدۇ...",
"saved": "{folder} غا {count} جۈپ ساقلاندى",
"saveFailed": "ساقلاش مەغلۇپ بولدى",
"noImage": "ئاۋۋال سۈرەت يۈكلەڭ",
"noFolder": "قىسقۇچ تاللاڭ",
"processingFailed": "OCR بىر تەرەپ قىلىش مەغلۇپ بولدى",
"tryAgain": "تېخىمۇ ئېنىق سۈرەت بىلەن قايتا سىناڭ",
"detectedLanguages": "بايقالدى: {source} → {target}"
},
"profile": {
"myProfile": "شەخسىي ئۇچۇرۇم",
"email": "ئېلخەت: {email}",

View File

@@ -197,6 +197,35 @@
"favorites": "收藏",
"settings": "设置"
},
"ocr": {
"title": "OCR 词汇提取",
"description": "上传教材词汇表截图,提取单词-释义对",
"uploadImage": "上传图片",
"dragDropHint": "拖放图片到此处,或点击选择",
"supportedFormats": "支持格式JPG、PNG、WebP",
"selectFolder": "选择文件夹",
"chooseFolder": "选择保存提取词汇的文件夹",
"noFolders": "暂无文件夹,请先创建文件夹",
"languageHints": "语言提示(可选)",
"sourceLanguageHint": "源语言(如:英语)",
"targetLanguageHint": "目标/翻译语言(如:中文)",
"process": "处理图片",
"processing": "处理中...",
"preview": "预览",
"extractedPairs": "提取的词汇对",
"word": "单词",
"definition": "释义",
"pairsCount": "已提取 {count} 个词汇对",
"savePairs": "保存到文件夹",
"saving": "保存中...",
"saved": "成功将 {count} 个词汇对保存到 {folder}",
"saveFailed": "保存失败",
"noImage": "请先上传图片",
"noFolder": "请选择文件夹",
"processingFailed": "OCR 处理失败",
"tryAgain": "请尝试上传更清晰的图片",
"detectedLanguages": "检测到:{source} → {target}"
},
"profile": {
"myProfile": "我的个人资料",
"email": "邮箱:{email}",