translate / static /index1.html
mistpe's picture
Rename static/index.html to static/index1.html
5eb14e1 verified
<!DOCTYPE html>
<html lang="zh">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>翻译助手</title>
<link href="https://cdnjs.cloudflare.com/ajax/libs/tailwindcss/2.2.19/tailwind.min.css" rel="stylesheet">
<script src="https://cdnjs.cloudflare.com/ajax/libs/axios/0.21.1/axios.min.js"></script>
</head>
<body class="bg-gray-50">
<!-- 导航栏 -->
<nav class="bg-white shadow-md fixed w-full top-0 z-50">
<div class="max-w-7xl mx-auto px-4">
<div class="flex justify-between h-16">
<div class="flex items-center">
<span class="text-xl font-bold text-purple-700">翻译助手</span>
</div>
<div class="hidden md:flex items-center space-x-8">
<button class="nav-btn text-gray-700 hover:text-purple-700" data-target="text-trans">文本翻译</button>
<button class="nav-btn text-gray-700 hover:text-purple-700" data-target="file-trans">文件翻译</button>
</div>
<button id="mobile-menu-btn" class="md:hidden flex items-center">
<svg class="w-6 h-6" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16"/>
</svg>
</button>
</div>
</div>
<!-- 移动端菜单 -->
<div id="mobile-menu" class="hidden md:hidden bg-white border-t">
<div class="px-2 pt-2 pb-3 space-y-1">
<button class="nav-btn block w-full text-left px-3 py-2 text-gray-700 hover:bg-purple-50" data-target="text-trans">文本翻译</button>
<button class="nav-btn block w-full text-left px-3 py-2 text-gray-700 hover:bg-purple-50" data-target="file-trans">文件翻译</button>
</div>
</div>
</nav>
<!-- 主要内容区域 -->
<main class="pt-20 px-4 max-w-7xl mx-auto">
<!-- 文本翻译页面 -->
<div id="text-trans" class="page">
<div class="flex flex-col lg:flex-row gap-6">
<!-- 左侧工具栏 -->
<div class="lg:w-64">
<div class="bg-white rounded-lg shadow p-4 space-y-4">
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">源语言</label>
<select id="source-lang" class="w-full border-gray-300 rounded-md shadow-sm">
<option value="AUTO">自动检测</option>
<option value="ZH">中文</option>
<option value="EN">英语</option>
<option value="JA">日语</option>
<option value="KO">韩语</option>
<option value="FR">法语</option>
<option value="DE">德语</option>
<option value="ES">西班牙语</option>
<option value="RU">俄语</option>
<option value="IT">意大利语</option>
<option value="PT">葡萄牙语</option>
<option value="VI">越南语</option>
<option value="ID">印尼语</option>
<option value="TH">泰语</option>
<option value="MS">马来语</option>
<option value="AR">阿拉伯语</option>
<option value="HI">印地语</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">目标语言</label>
<select id="target-lang" class="w-full border-gray-300 rounded-md shadow-sm">
<option value="ZH">中文</option>
<option value="EN">英语</option>
<option value="JA">日语</option>
<option value="KO">韩语</option>
<option value="FR">法语</option>
<option value="DE">德语</option>
<option value="ES">西班牙语</option>
<option value="RU">俄语</option>
<option value="IT">意大利语</option>
<option value="PT">葡萄牙语</option>
<option value="VI">越南语</option>
<option value="ID">印尼语</option>
<option value="TH">泰语</option>
<option value="MS">马来语</option>
<option value="AR">阿拉伯语</option>
<option value="HI">印地语</option>
</select>
</div>
<button id="switch-lang" class="w-full py-2 px-4 border border-purple-700 text-purple-700 rounded-md hover:bg-purple-50">
切换语言
</button>
</div>
</div>
<!-- 翻译区域 -->
<div class="flex-1">
<div class="grid md:grid-cols-2 gap-6">
<!-- 源文本 -->
<div class="bg-white rounded-lg shadow p-4">
<textarea
id="source-text"
class="w-full h-48 p-2 border-0 focus:ring-0 resize-none"
placeholder="请输入要翻译的文本"
></textarea>
<div class="flex justify-between mt-2 text-sm text-gray-500">
<span id="char-count">0/10000</span>
<button id="clear-text" class="hover:text-gray-700">清空</button>
</div>
</div>
<!-- 译文 -->
<div class="bg-white rounded-lg shadow p-4">
<div id="target-text" class="h-48 p-2"></div>
<div class="flex justify-end mt-2">
<button id="copy-result" class="text-sm text-gray-500 hover:text-gray-700">复制结果</button>
</div>
</div>
</div>
<button id="translate-btn" class="mt-6 mx-auto block py-3 px-8 bg-purple-600 text-white rounded-lg hover:bg-purple-700 disabled:opacity-50">
翻译
</button>
</div>
</div>
</div>
<!-- 文件翻译页面 -->
<div id="file-trans" class="page hidden">
<div class="flex flex-col lg:flex-row gap-6">
<!-- 左侧工具栏 -->
<div class="lg:w-64 space-y-6">
<div class="bg-white rounded-lg shadow p-4 space-y-4">
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">源语言</label>
<select id="file-source-lang" class="w-full border-gray-300 rounded-md shadow-sm">
<option value="AUTO">自动检测</option>
<option value="ZH">中文</option>
<option value="EN">英语</option>
<option value="JA">日语</option>
<option value="KO">韩语</option>
<option value="FR">法语</option>
<option value="DE">德语</option>
<option value="ES">西班牙语</option>
<option value="RU">俄语</option>
<option value="IT">意大利语</option>
<option value="PT">葡萄牙语</option>
<option value="VI">越南语</option>
<option value="ID">印尼语</option>
<option value="TH">泰语</option>
<option value="MS">马来语</option>
<option value="AR">阿拉伯语</option>
<option value="HI">印地语</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">目标语言</label>
<select id="file-target-lang" class="w-full border-gray-300 rounded-md shadow-sm">
<option value="ZH">中文</option>
<option value="EN">英语</option>
<option value="JA">日语</option>
<option value="KO">韩语</option>
<option value="FR">法语</option>
<option value="DE">德语</option>
<option value="ES">西班牙语</option>
<option value="RU">俄语</option>
<option value="IT">意大利语</option>
<option value="PT">葡萄牙语</option>
<option value="VI">越南语</option>
<option value="ID">印尼语</option>
<option value="TH">泰语</option>
<option value="MS">马来语</option>
<option value="AR">阿拉伯语</option>
<option value="HI">印地语</option>
</select>
</div>
</div>
<div class="bg-white rounded-lg shadow p-4 space-y-4">
<button id="upload-btn" class="w-full py-2 px-4 bg-purple-600 text-white rounded-md hover:bg-purple-700">
上传文件
</button>
<button id="file-translate-btn" class="w-full py-2 px-4 bg-purple-100 text-purple-700 rounded-md hover:bg-purple-200" disabled>
开始翻译
</button>
<button id="export-btn" class="w-full py-2 px-4 border border-gray-300 text-gray-700 rounded-md hover:bg-gray-50" disabled>
导出文档
</button>
</div>
<!-- 翻译进度 -->
<div id="progress-panel" class="bg-white rounded-lg shadow p-4 hidden">
<h3 class="text-sm font-medium text-gray-700 mb-4">翻译进度</h3>
<div class="space-y-2">
<div class="flex justify-between text-sm">
<span>总段落数</span>
<span id="total-segments">0</span>
</div>
<div class="flex justify-between text-sm">
<span>已翻译</span>
<span id="translated-segments">0</span>
</div>
<div class="h-2 bg-gray-200 rounded-full">
<div id="progress-bar" class="h-full bg-purple-600 rounded-full w-0 transition-all"></div>
</div>
</div>
</div>
</div>
<!-- 文件内容区域 -->
<div class="flex-1">
<!-- 上传区域 -->
<div id="upload-zone" class="h-64 border-2 border-dashed border-purple-400 rounded-lg flex flex-col items-center justify-center bg-white">
<svg class="w-12 h-12 text-purple-400 mb-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M7 16a4 4 0 01-.88-7.903A5 5 0 1115.9 6L16 6a5 5 0 011 9.9M15 13l-3-3m0 0l-3 3m3-3v12"/>
</svg>
<p class="text-gray-600">拖放文件到这里或点击上传</p>
<p class="text-sm text-gray-500 mt-2">支持 TXT、DOCX、PDF、Markdown 格式</p>
</div>
<!-- 文档预览 -->
<div id="file-preview" class="hidden">
<div class="bg-white rounded-lg shadow">
<div class="border-b border-gray-200 p-4 flex justify-between items-center">
<span id="file-name" class="font-medium"></span>
<button id="close-preview" class="text-gray-500 hover:text-gray-700">
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"/>
</svg>
</button>
</div>
<div class="p-6">
<div class="grid md:grid-cols-2 gap-6">
<div>
<h3 class="text-sm font-medium text-gray-700 mb-4">原文</h3>
<div id="file-content" class="space-y-4"></div>
</div>
<div>
<h3 class="text-sm font-medium text-gray-700 mb-4">译文</h3>
<div id="file-translation" class="space-y-4"></div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</main>
<!-- 导出设置模态框 -->
<div id="export-modal" class="fixed inset-0 z-50 hidden">
<div class="absolute inset-0 bg-gray-500 bg-opacity-75"></div>
<div class="fixed inset-0 z-10 overflow-y-auto">
<div class="flex min-h-full items-end justify-center p-4 sm:items-center">
<div class="relative bg-white rounded-lg shadow-xl w-full max-w-md p-6">
<h3 class="text-lg font-medium text-gray-900 mb-4">导出设置(暂时不支持导出为PDF)</h3>
<div class="space-y-4">
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">导出格式</label>
<select id="export-format" class="w-full rounded-md border-gray-300">
<option value="auto">自动 (与源文件相同)</option>
<option value="txt">纯文本 (TXT)</option>
<option value="docx">Word文档 (DOCX)</option>
<option value="md">Markdown (MD)</option>
<option value="html">网页 (HTML)</option>
</select>
</div>
<div>
<label class="block text-sm font-medium text-gray-700 mb-1">导出模式</label>
<div class="space-y-2">
<label class="inline-flex items-center">
<input type="radio" name="export-mode" value="translated" class="text-purple-600" checked>
<span class="ml-2">仅译文</span>
</label>
<label class="inline-flex items-center ml-4">
<input type="radio" name="export-mode" value="parallel" class="text-purple-600">
<span class="ml-2">对照模式</span>
</label>
</div>
</div>
</div>
<div class="mt-6 flex justify-end space-x-3">
<button id="cancel-export" class="px-4 py-2 border border-gray-300 rounded-md text-gray-700 hover:bg-gray-50">
取消
</button>
<button id="confirm-export" class="px-4 py-2 bg-purple-600 text-white rounded-md hover:bg-purple-700">
导出
</button>
</div>
</div>
</div>
</div>
</div>
<script>
// 常量定义
const API_ENDPOINTS = {
TRANSLATE_TEXT: '/translate_text',
UPLOAD_FILE: '/upload',
TRANSLATE_FILE: '/translate',
EXPORT: '/export'
};
// 工具函数
const utils = {
showError(message) {
// 简单的错误提示
alert(message);
},
async copyToClipboard(text) {
try {
await navigator.clipboard.writeText(text);
return true;
} catch (err) {
console.error('Copy failed:', err);
return false;
}
},
formatBytes(bytes) {
if (bytes === 0) return '0 B';
const k = 1024;
const sizes = ['B', 'KB', 'MB', 'GB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return `${parseFloat((bytes / Math.pow(k, i)).toFixed(2))} ${sizes[i]}`;
}
};
// API封装
const api = {
async translateText(text, sourceLang, targetLang) {
try {
const response = await fetch(API_ENDPOINTS.TRANSLATE_TEXT, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
text,
source_lang: sourceLang,
target_lang: targetLang
})
});
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
const data = await response.json();
return data;
} catch (error) {
console.error('Translation failed:', error);
throw error;
}
},
async uploadFile(file, sourceLang, targetLang) {
const formData = new FormData();
formData.append('file', file);
formData.append('source_lang', sourceLang);
formData.append('target_lang', targetLang);
try {
const response = await fetch(API_ENDPOINTS.UPLOAD_FILE, {
method: 'POST',
body: formData
});
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return await response.json();
} catch (error) {
console.error('File upload failed:', error);
throw error;
}
},
async translateFile(segments, sourceLang, targetLang) {
try {
const response = await fetch(API_ENDPOINTS.TRANSLATE_FILE, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
segments,
source_lang: sourceLang,
target_lang: targetLang
})
});
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return await response.json();
} catch (error) {
console.error('File translation failed:', error);
throw error;
}
},
async exportDocument(segments, format, mode, sourceFileType) {
try {
const response = await fetch(API_ENDPOINTS.EXPORT, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
segments,
format,
mode,
source_file_type: sourceFileType
})
});
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return await response.blob();
} catch (error) {
console.error('Export failed:', error);
throw error;
}
}
};
// 文本翻译页面类
class TextTranslation {
constructor() {
this.sourceText = document.getElementById('source-text');
this.targetText = document.getElementById('target-text');
this.translateBtn = document.getElementById('translate-btn');
this.sourceLang = document.getElementById('source-lang');
this.targetLang = document.getElementById('target-lang');
this.charCount = document.getElementById('char-count');
this.clearBtn = document.getElementById('clear-text');
this.copyBtn = document.getElementById('copy-result');
this.switchBtn = document.getElementById('switch-lang');
this.init();
}
init() {
this.bindEvents();
this.updateCharCount();
}
bindEvents() {
this.sourceText.addEventListener('input', () => this.updateCharCount());
this.translateBtn.addEventListener('click', () => this.translate());
this.clearBtn.addEventListener('click', () => this.clearText());
this.copyBtn.addEventListener('click', () => this.copyResult());
this.switchBtn.addEventListener('click', () => this.switchLanguages());
}
updateCharCount() {
const count = this.sourceText.value.length;
this.charCount.textContent = `${count}/10000`;
this.translateBtn.disabled = count === 0;
}
clearText() {
this.sourceText.value = '';
this.targetText.textContent = '';
this.updateCharCount();
}
async copyResult() {
const success = await utils.copyToClipboard(this.targetText.textContent);
if (success) {
this.copyBtn.textContent = '已复制';
setTimeout(() => {
this.copyBtn.textContent = '复制结果';
}, 2000);
}
}
switchLanguages() {
if (this.sourceLang.value === 'auto') return;
[this.sourceLang.value, this.targetLang.value] =
[this.targetLang.value, this.sourceLang.value];
}
async translate() {
if (!this.sourceText.value.trim()) return;
this.translateBtn.disabled = true;
try {
const result = await api.translateText(
this.sourceText.value,
this.sourceLang.value,
this.targetLang.value
);
this.targetText.textContent = result.translated;
} catch (error) {
utils.showError('翻译失败,请稍后重试');
} finally {
this.translateBtn.disabled = false;
}
}
}
// 文件翻译页面类
class FileTranslation {
constructor() {
this.uploadZone = document.getElementById('upload-zone');
this.filePreview = document.getElementById('file-preview');
this.uploadBtn = document.getElementById('upload-btn');
this.translateBtn = document.getElementById('file-translate-btn');
this.exportBtn = document.getElementById('export-btn');
this.progressPanel = document.getElementById('progress-panel');
this.closePreviewBtn = document.getElementById('close-preview');
this.sourceLang = document.getElementById('file-source-lang');
this.targetLang = document.getElementById('file-target-lang');
this.currentFile = null;
this.segments = [];
this.init();
}
init() {
this.bindEvents();
this.initDragDrop();
}
bindEvents() {
this.uploadBtn.addEventListener('click', () => this.triggerFileInput());
this.translateBtn.addEventListener('click', () => this.translateFile());
this.exportBtn.addEventListener('click', () => this.showExportModal());
this.closePreviewBtn.addEventListener('click', () => this.closePreview());
// 导出模态框事件
document.getElementById('confirm-export').addEventListener('click', () => this.exportFile());
document.getElementById('cancel-export').addEventListener('click', () => this.hideExportModal());
}
initDragDrop() {
this.uploadZone.addEventListener('dragover', (e) => {
e.preventDefault();
e.stopPropagation();
this.uploadZone.classList.add('border-purple-600');
});
this.uploadZone.addEventListener('dragleave', (e) => {
e.preventDefault();
e.stopPropagation();
this.uploadZone.classList.remove('border-purple-600');
});
this.uploadZone.addEventListener('drop', async (e) => {
e.preventDefault();
e.stopPropagation();
this.uploadZone.classList.remove('border-purple-600');
const file = e.dataTransfer.files[0];
await this.handleFile(file);
});
}
triggerFileInput() {
const input = document.createElement('input');
input.type = 'file';
input.accept = '.txt,.docx,.pdf,.md';
input.onchange = (e) => this.handleFile(e.target.files[0]);
input.click();
}
async handleFile(file) {
if (!file) return;
const validTypes = [
'text/plain',
'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
'application/pdf',
'text/markdown'
];
if (!validTypes.includes(file.type)) {
utils.showError('不支持的文件格式');
return;
}
this.currentFile = file;
try {
const result = await api.uploadFile(
file,
this.sourceLang.value,
this.targetLang.value
);
this.segments = result.segments;
this.updatePreview();
this.translateBtn.disabled = false;
this.showProgressPanel();
} catch (error) {
utils.showError('文件上传失败,请重试');
}
}
updatePreview() {
this.uploadZone.classList.add('hidden');
this.filePreview.classList.remove('hidden');
document.getElementById('file-name').textContent = this.currentFile.name;
const contentDiv = document.getElementById('file-content');
contentDiv.innerHTML = this.segments.map((segment, index) => `
<div class="p-4 bg-gray-50 rounded">
<div class="text-gray-900">${segment.text}</div>
${segment.type === 'heading' ? '<div class="text-xs text-gray-500 mt-1">标题</div>' : ''}
</div>
`).join('');
}
async translateFile() {
if (!this.segments.length) return;
this.translateBtn.disabled = true;
try {
const result = await api.translateFile(
this.segments,
this.sourceLang.value,
this.targetLang.value
);
this.segments = result.segments;
this.updateTranslations();
this.exportBtn.disabled = false;
this.updateProgress();
} catch (error) {
utils.showError('翻译失败,请重试');
} finally {
this.translateBtn.disabled = false;
}
}
updateTranslations() {
const translationDiv = document.getElementById('file-translation');
translationDiv.innerHTML = this.segments.map((segment, index) => `
<div class="p-4 bg-gray-50 rounded">
<div class="text-gray-900">${segment.translated || ''}</div>
${segment.confidence ? `
<div class="mt-2 h-1 bg-gray-200 rounded-full overflow-hidden">
<div class="h-full bg-purple-600" style="width: ${segment.confidence * 100}%"></div>
</div>
` : ''}
</div>
`).join('');
}
showProgressPanel() {
this.progressPanel.classList.remove('hidden');
this.updateProgress();
}
updateProgress() {
const total = this.segments.length;
const translated = this.segments.filter(s => s.translated).length;
document.getElementById('total-segments').textContent = total;
document.getElementById('translated-segments').textContent = translated;
document.getElementById('progress-bar').style.width = `${(translated / total * 100)}%`;
}
closePreview() {
this.uploadZone.classList.remove('hidden');
this.filePreview.classList.add('hidden');
this.progressPanel.classList.add('hidden');
this.translateBtn.disabled = true;
this.exportBtn.disabled = true;
this.currentFile = null;
this.segments = [];
}
showExportModal() {
document.getElementById('export-modal').classList.remove('hidden');
}
hideExportModal() {
document.getElementById('export-modal').classList.add('hidden');
}
async exportFile() {
const formatSelect = document.getElementById('export-format');
const selectedFormat = formatSelect.value;
const mode = document.querySelector('input[name="export-mode"]:checked').value;
// 确定最终使用的导出格式
let exportFormat;
if (selectedFormat === 'auto') {
// 如果选择"自动",则使用原文件格式
exportFormat = this.currentFile.name.split('.').pop();
} else {
// 否则使用用户选择的格式
exportFormat = selectedFormat;
}
try {
const blob = await api.exportDocument(
this.segments,
exportFormat,
mode,
this.currentFile.name.split('.').pop() // 原文件类型仍然需要传递给后端
);
// 根据选择的格式确定文件扩展名
let fileExtension = exportFormat;
if (exportFormat === 'auto') {
fileExtension = this.currentFile.name.split('.').pop();
}
// 创建新的文件名
const originalName = this.currentFile.name.split('.')[0];
const newFileName = `translated_${originalName}.${fileExtension}`;
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = newFileName;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
URL.revokeObjectURL(url);
this.hideExportModal();
} catch (error) {
utils.showError('导出失败,请重试');
}
}
}
// 页面导航控制类
class Navigation {
constructor() {
this.mobileMenuBtn = document.getElementById('mobile-menu-btn');
this.mobileMenu = document.getElementById('mobile-menu');
this.navButtons = document.querySelectorAll('.nav-btn');
this.init();
}
init() {
this.bindEvents();
}
bindEvents() {
this.mobileMenuBtn.addEventListener('click', () => this.toggleMobileMenu());
this.navButtons.forEach(btn => {
btn.addEventListener('click', () => {
this.switchPage(btn.dataset.target);
this.mobileMenu.classList.add('hidden');
});
});
}
toggleMobileMenu() {
this.mobileMenu.classList.toggle('hidden');
}
switchPage(pageId) {
document.querySelectorAll('.page').forEach(page => {
page.classList.add('hidden');
});
document.getElementById(pageId).classList.remove('hidden');
this.navButtons.forEach(btn => {
if (btn.dataset.target === pageId) {
btn.classList.add('text-purple-700');
} else {
btn.classList.remove('text-purple-700');
}
});
}
}
// 初始化应用
document.addEventListener('DOMContentLoaded', () => {
new Navigation();
new TextTranslation();
new FileTranslation();
});
</script>
</body>
</html>