4 Commits

Author SHA1 Message Date
44077796f8 feat: 翻译记录添加不共享开关功能
- Translation 模型新增 no_share 字段
- 管理后台翻译记录页面添加共享状态列和切换按钮
- 不共享的翻译不会被其他用户使用缓存
- 缓存匹配时检查是否有 no_share 标记
2026-04-16 19:06:43 +08:00
504fed6c3e fix: 修复网站配置保存问题
- 保存配置时 key 不再加 site_ 前缀,与读取时一致
- 修正:site_site_name -> site_name
- 修正:site_site_footer -> site_footer
2026-04-16 18:49:03 +08:00
69e4ca4d64 feat: 前端页面使用网站基础配置
- 使用 Flask context_processor 自动注入 site_config
- 所有页面标题使用 site_name 配置
- 所有页面导航栏品牌使用 site_name 配置
- 所有页面底部使用 site_footer 配置
- 文件上传时使用 max_file_size 配置验证文件大小
- 显示最大文件限制提示
2026-04-16 18:44:57 +08:00
aa8526035b feat: 系统配置增加网站基础配置
- 网站名称设置
- 底部信息设置(支持HTML)
- 最大上传文件大小设置(MB)
- 缓存有效期设置(天)
- 默认源语言/目标语言设置
- 翻译缓存开关
- 访客翻译开关
- 邮件通知开关
- 新增 get_site_config() 函数供其他模块使用
2026-04-16 18:36:21 +08:00
13 changed files with 355 additions and 94 deletions

105
admin.py
View File

@@ -255,6 +255,28 @@ def delete_translation(trans_id):
return jsonify({'success': True})
@admin_bp.route('/translation/<int:trans_id>/toggle-share', methods=['POST'])
@admin_required
def toggle_translation_share(trans_id):
"""切换翻译共享状态"""
translation = Translation.query.get_or_404(trans_id)
translation.no_share = not translation.no_share
db.session.commit()
# 记录日志
log = OperationLog(
user_id=session.get('user_id'),
username='admin',
action='toggle_translation_share',
target=f'翻译#{trans_id}',
detail=f'设置共享状态为: {not translation.no_share}'
)
db.session.add(log)
db.session.commit()
return jsonify({'success': True, 'no_share': translation.no_share})
# ==================== 缓存管理 ====================
@admin_bp.route('/cache')
@admin_required
@@ -313,35 +335,56 @@ def clear_cache():
@admin_required
def settings():
"""系统配置"""
if request.method == 'POST':
data = request.json if request.is_json else request.form
for key, value in data.items():
SystemConfig.set(key, value)
if request.is_json:
return jsonify({'success': True})
flash('配置已保存', 'success')
# 获取所有配置
configs = SystemConfig.query.all()
config_dict = {c.key: c.value for c in configs}
# 获取动态配置
dynamic_configs = DynamicConfig.query.all()
# 获取LLM动态配置
llm_config = get_llm_config()
# 获取网站基础配置
site_config = {
'site_name': DynamicConfig.get('site_name', 'PDF翻译助手'),
'site_footer': DynamicConfig.get('site_footer', '© 2026 PDF翻译助手'),
'max_file_size': DynamicConfig.get('max_file_size', 50),
'cache_expire_days': DynamicConfig.get('cache_expire_days', 30),
'enable_email_notify': DynamicConfig.get('enable_email_notify', True),
'enable_cache': DynamicConfig.get('enable_cache', True),
'enable_guest': DynamicConfig.get('enable_guest', True),
'default_source_lang': DynamicConfig.get('default_source_lang', 'en'),
'default_target_lang': DynamicConfig.get('default_target_lang', 'zh'),
}
return render_template('admin/settings.html',
configs=config_dict,
dynamic_configs=dynamic_configs,
user_limits=USER_LIMITS,
membership_plans=MEMBERSHIP_PLANS,
llm_config=llm_config
llm_config=llm_config,
site_config=site_config
)
@admin_bp.route('/settings/site', methods=['POST'])
@admin_required
def save_site_settings():
"""保存网站基础配置"""
data = request.json
# 保存每个配置项key 直接使用,不带 site_ 前缀)
for key, value in data.items():
if key in ['max_file_size', 'cache_expire_days']:
DynamicConfig.set(key, int(value), category='site', value_type='int', user_id=session.get('user_id'))
elif key in ['enable_email_notify', 'enable_cache', 'enable_guest']:
DynamicConfig.set(key, bool(value), category='site', value_type='bool', user_id=session.get('user_id'))
else:
DynamicConfig.set(key, value, category='site', user_id=session.get('user_id'))
# 记录日志
log = OperationLog(
user_id=session.get('user_id'),
username='admin',
action='save_site_settings',
detail='保存网站基础配置'
)
db.session.add(log)
db.session.commit()
return jsonify({'success': True})
# ==================== 用户权限配置 ====================
@admin_bp.route('/settings/user-limits', methods=['GET', 'POST'])
@admin_required
@@ -1450,4 +1493,20 @@ def init_default_backup_llm():
def get_backup_llm_configs():
"""获取所有备用大模型配置(供其他模块使用)"""
configs = BackupLLMConfig.query.filter_by(is_active=True).order_by(BackupLLMConfig.sort_order).all()
return [c.to_dict() for c in configs]
return [c.to_dict() for c in configs]
# ==================== 获取网站基础配置(供其他模块使用) ====================
def get_site_config():
"""获取网站基础配置"""
return {
'site_name': DynamicConfig.get('site_name', 'PDF翻译助手'),
'site_footer': DynamicConfig.get('site_footer', '© 2026 PDF翻译助手'),
'max_file_size': DynamicConfig.get('max_file_size', 50),
'cache_expire_days': DynamicConfig.get('cache_expire_days', 30),
'enable_email_notify': DynamicConfig.get('enable_email_notify', True),
'enable_cache': DynamicConfig.get('enable_cache', True),
'enable_guest': DynamicConfig.get('enable_guest', True),
'default_source_lang': DynamicConfig.get('default_source_lang', 'en'),
'default_target_lang': DynamicConfig.get('default_target_lang', 'zh'),
}

15
app.py
View File

@@ -34,6 +34,14 @@ db.init_app(app)
# 注册后台管理蓝图
app.register_blueprint(admin_bp)
# Context processor - 所有模板自动获得 site_config
@app.context_processor
def inject_site_config():
def get_config():
from admin import get_site_config
return get_site_config()
return {'site_config': get_config()}
# 初始化服务
cache_service = CacheService(CACHE_DIR, CACHE_EXPIRE_DAYS)
@@ -357,8 +365,11 @@ def upload_pdf():
cache_path = cache_service.get_cache(file_hash)
from_cache = False
if cache_path and ENABLE_CACHE and not instruction:
# 有缓存且无特殊翻译要求,直接使用缓存
# 检查是否有用户设置了不共享此文件
no_share_check = Translation.query.filter_by(file_hash=file_hash, no_share=True).first()
if cache_path and ENABLE_CACHE and not instruction and not no_share_check:
# 有缓存且无特殊翻译要求且无不共享标记,直接使用缓存
from_cache = True
output_path = cache_path
else:

View File

@@ -178,6 +178,9 @@ class Translation(db.Model):
# 是否来自缓存
from_cache = db.Column(db.Boolean, default=False)
# 不共享缓存
no_share = db.Column(db.Boolean, default=False) # 不共享此翻译给其他用户
# 重译信息
retranslate_request = db.Column(db.Text, nullable=True) # 重译要求
parent_id = db.Column(db.Integer, db.ForeignKey('translations.id'), nullable=True) # 原翻译ID
@@ -190,6 +193,7 @@ class Translation(db.Model):
'status': self.status,
'progress': self.progress,
'from_cache': self.from_cache,
'no_share': self.no_share,
'file_size': self.file_size,
'created_at': self.created_at.isoformat() if self.created_at else None,
'completed_at': self.completed_at.isoformat() if self.completed_at else None,

View File

@@ -23,6 +23,14 @@ document.getElementById('uploadForm').addEventListener('submit', async function(
return;
}
// 检查文件大小
const maxSizeMB = parseInt(document.getElementById('submitBtn').dataset.maxSize) || 50;
const fileSizeMB = file.size / (1024 * 1024);
if (fileSizeMB > maxSizeMB) {
alert(`文件大小超出限制(最大${maxSizeMB}MB当前${fileSizeMB.toFixed(1)}MB`);
return;
}
// 显示进度区域
document.getElementById('progressSection').style.display = 'block';
document.getElementById('resultSection').style.display = 'none';

View File

@@ -22,16 +22,16 @@
<h5 class="text-white mb-0"><i class="bi bi-gear-fill"></i> 后台管理</h5>
</div>
<ul class="nav flex-column">
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.dashboard') }}"><i class="bi bi-speedometer2"></i> 数据概览</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.users') }}"><i class="bi bi-people"></i> 用户管理</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.translations') }}"><i class="bi bi-file-text"></i> 翻译记录</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.cache_list') }}"><i class="bi bi-database"></i> 缓存管理</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.packages') }}"><i class="bi bi-box-seam"></i> 数据包套餐</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.stats') }}"><i class="bi bi-bar-chart"></i> 统计报表</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.logs') }}"><i class="bi bi-list-check"></i> 操作日志</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.llm_config') }}"><i class="bi bi-cpu"></i> 大模型配置</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.user_types') }}"><i class="bi bi-person-badge"></i> 用户类型</a></li>
<li class="nav-item"><a class="nav-link " href="{{ url_for('admin.membership_plans') }}"><i class="bi bi-credit-card"></i> 会员套餐</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.dashboard') }}"><i class="bi bi-speedometer2"></i> 数据概览</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.users') }}"><i class="bi bi-people"></i> 用户管理</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.translations') }}"><i class="bi bi-file-text"></i> 翻译记录</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.cache_list') }}"><i class="bi bi-database"></i> 缓存管理</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.packages') }}"><i class="bi bi-box-seam"></i> 数据包套餐</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.stats') }}"><i class="bi bi-bar-chart"></i> 统计报表</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.logs') }}"><i class="bi bi-list-check"></i> 操作日志</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.llm_config') }}"><i class="bi bi-cpu"></i> 大模型配置</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.user_types') }}"><i class="bi bi-person-badge"></i> 用户类型</a></li>
<li class="nav-item"><a class="nav-link" href="{{ url_for('admin.membership_plans') }}"><i class="bi bi-credit-card"></i> 会员套餐</a></li>
<li class="nav-item"><a class="nav-link active" href="{{ url_for('admin.settings') }}"><i class="bi bi-sliders"></i> 系统配置</a></li>
</ul>
<div class="position-absolute bottom-0 w-100 p-3 border-top border-secondary">
@@ -42,6 +42,92 @@
<main class="main-content">
<h4 class="mb-4"><i class="bi bi-sliders"></i> 系统配置</h4>
<!-- 网站基础配置 -->
<div class="card mb-4">
<div class="card-header">
<h6 class="mb-0"><i class="bi bi-building"></i> 网站基础配置</h6>
</div>
<div class="card-body">
<form id="siteConfigForm">
<div class="row">
<div class="col-md-6">
<div class="mb-3">
<label class="form-label">网站名称</label>
<input type="text" class="form-control" name="site_name" value="{{ site_config.site_name }}" placeholder="PDF翻译助手">
<small class="text-muted">显示在页面标题和Logo处</small>
</div>
<div class="mb-3">
<label class="form-label">底部信息</label>
<textarea class="form-control" name="site_footer" rows="2" placeholder="© 2026 PDF翻译助手">{{ site_config.site_footer }}</textarea>
<small class="text-muted">显示在页面底部支持HTML</small>
</div>
<div class="mb-3">
<label class="form-label">默认源语言</label>
<select class="form-select" name="default_source_lang">
<option value="en" {% if site_config.default_source_lang == 'en' %}selected{% endif %}>英语 (en)</option>
<option value="ja" {% if site_config.default_source_lang == 'ja' %}selected{% endif %}>日语 (ja)</option>
<option value="ko" {% if site_config.default_source_lang == 'ko' %}selected{% endif %}>韩语 (ko)</option>
<option value="fr" {% if site_config.default_source_lang == 'fr' %}selected{% endif %}>法语 (fr)</option>
<option value="de" {% if site_config.default_source_lang == 'de' %}selected{% endif %}>德语 (de)</option>
<option value="es" {% if site_config.default_source_lang == 'es' %}selected{% endif %}>西班牙语 (es)</option>
<option value="ru" {% if site_config.default_source_lang == 'ru' %}selected{% endif %}>俄语 (ru)</option>
<option value="pt" {% if site_config.default_source_lang == 'pt' %}selected{% endif %}>葡萄牙语 (pt)</option>
<option value="zh" {% if site_config.default_source_lang == 'zh' %}selected{% endif %}>中文 (zh)</option>
</select>
</div>
<div class="mb-3">
<label class="form-label">默认目标语言</label>
<select class="form-select" name="default_target_lang">
<option value="zh" {% if site_config.default_target_lang == 'zh' %}selected{% endif %}>中文 (zh)</option>
<option value="en" {% if site_config.default_target_lang == 'en' %}selected{% endif %}>英语 (en)</option>
<option value="ja" {% if site_config.default_target_lang == 'ja' %}selected{% endif %}>日语 (ja)</option>
<option value="ko" {% if site_config.default_target_lang == 'ko' %}selected{% endif %}>韩语 (ko)</option>
</select>
</div>
</div>
<div class="col-md-6">
<div class="mb-3">
<label class="form-label">最大上传文件大小 (MB)</label>
<input type="number" class="form-control" name="max_file_size" value="{{ site_config.max_file_size }}" min="1" max="500">
<small class="text-muted">用户上传PDF文件的最大大小限制</small>
</div>
<div class="mb-3">
<label class="form-label">缓存有效期 (天)</label>
<input type="number" class="form-control" name="cache_expire_days" value="{{ site_config.cache_expire_days }}" min="1" max="365">
<small class="text-muted">翻译结果缓存的保留天数</small>
</div>
<div class="mb-3">
<div class="form-check form-switch">
<input class="form-check-input" type="checkbox" name="enable_cache" id="enable_cache" {% if site_config.enable_cache %}checked{% endif %}>
<label class="form-check-label" for="enable_cache">启用翻译缓存</label>
</div>
<small class="text-muted">相同文件翻译时直接返回缓存结果</small>
</div>
<div class="mb-3">
<div class="form-check form-switch">
<input class="form-check-input" type="checkbox" name="enable_guest" id="enable_guest" {% if site_config.enable_guest %}checked{% endif %}>
<label class="form-check-label" for="enable_guest">允许访客翻译</label>
</div>
<small class="text-muted">未登录用户可以使用翻译服务</small>
</div>
<div class="mb-3">
<div class="form-check form-switch">
<input class="form-check-input" type="checkbox" name="enable_email_notify" id="enable_email_notify" {% if site_config.enable_email_notify %}checked{% endif %}>
<label class="form-check-label" for="enable_email_notify">邮件通知</label>
</div>
<small class="text-muted">翻译完成后发送邮件通知用户</small>
</div>
</div>
</div>
<button type="submit" class="btn btn-primary"><i class="bi bi-check-lg"></i> 保存配置</button>
</form>
<div id="saveResult" class="mt-3" style="display:none;"></div>
</div>
</div>
<!-- 其他配置入口 -->
<div class="row mb-4">
<div class="col-md-6">
<div class="card config-card h-100">
@@ -50,14 +136,7 @@
</div>
<div class="card-body">
<p class="text-muted">配置不同用户类型的权限限制,包括翻译次数、页数限制、功能权限等。</p>
<p><strong>支持操作:</strong></p>
<ul class="small">
<li>添加新的用户类型</li>
<li>编辑现有类型的权限</li>
<li>删除自定义类型(系统类型不可删除)</li>
<li>启用/禁用用户类型</li>
</ul>
<a href="{{ url_for('admin.user_types') }}" class="btn btn-primary">
<a href="{{ url_for('admin.user_types') }}" class="btn btn-outline-primary">
<i class="bi bi-gear"></i> 管理用户类型
</a>
</div>
@@ -71,15 +150,7 @@
</div>
<div class="card-body">
<p class="text-muted">配置会员套餐的价格、周期、描述等,用户购买后可升级用户类型。</p>
<p><strong>支持操作:</strong></p>
<ul class="small">
<li>添加新的会员套餐</li>
<li>编辑套餐价格和描述</li>
<li>删除自定义套餐(系统套餐不可删除)</li>
<li>上架/下架套餐</li>
<li>设置推荐套餐</li>
</ul>
<a href="{{ url_for('admin.membership_plans') }}" class="btn btn-primary">
<a href="{{ url_for('admin.membership_plans') }}" class="btn btn-outline-primary">
<i class="bi bi-credit-card"></i> 管理会员套餐
</a>
</div>
@@ -88,6 +159,21 @@
</div>
<div class="row mb-4">
<div class="col-md-6">
<div class="card config-card h-100">
<div class="card-header">
<h6 class="mb-0"><i class="bi bi-cpu"></i> 大模型配置</h6>
</div>
<div class="card-body">
<p class="text-muted">配置翻译使用的LLM大模型API地址、模型名称等参数。</p>
<p><strong>当前模型:</strong> {{ llm_config.get('model', '未设置') }}</p>
<a href="{{ url_for('admin.llm_config') }}" class="btn btn-outline-primary">
<i class="bi bi-cpu"></i> 配置大模型
</a>
</div>
</div>
</div>
<div class="col-md-6">
<div class="card config-card h-100">
<div class="card-header">
@@ -101,44 +187,69 @@
</div>
</div>
</div>
<div class="col-md-6">
<div class="card config-card h-100">
<div class="card-header">
<h6 class="mb-0"><i class="bi bi-cpu"></i> 大模型配置</h6>
</div>
<div class="card-body">
<p class="text-muted">配置翻译使用的LLM大模型API地址、模型名称等参数。</p>
<p><strong>当前模型:</strong> {{ llm_config.model }}</p>
<a href="{{ url_for('admin.llm_config') }}" class="btn btn-outline-primary">
<i class="bi bi-cpu"></i> 配置大模型
</a>
</div>
</div>
</div>
</div>
<!-- 系统信息 -->
<div class="card">
<div class="card-header"><h6 class="mb-0"><i class="bi bi-info-circle"></i> 系统信息</h6></div>
<div class="card-body">
<div class="row">
<div class="col-md-4">
<p><strong>应用名称:</strong> PDF翻译助手</p>
<p><strong>版本:</strong> 2.0.0</p>
<p><strong>应用名称:</strong> {{ site_config.site_name }}</p>
<p><strong>版本:</strong> 2.3.1</p>
<p><strong>框架:</strong> Flask + SQLAlchemy</p>
</div>
<div class="col-md-4">
<p><strong>缓存有效期:</strong> 30天</p>
<p><strong>默认最大文件:</strong> 50MB</p>
<p><strong>最大文件:</strong> {{ site_config.max_file_size }}MB</p>
<p><strong>缓存有效期:</strong> {{ site_config.cache_expire_days }}天</p>
<p><strong>数据库:</strong> SQLite</p>
</div>
<div class="col-md-4">
<p><strong>API地址:</strong> {{ llm_config.api_base }}</p>
<p><strong>超时时间:</strong> {{ llm_config.timeout }}秒</p>
<p><strong>API地址:</strong> {{ llm_config.get('api_base', '未设置') }}</p>
<p><strong>超时时间:</strong> {{ llm_config.get('timeout', 180) }}秒</p>
</div>
</div>
</div>
</div>
</main>
<script>
document.getElementById('siteConfigForm').addEventListener('submit', function(e) {
e.preventDefault();
const formData = new FormData(this);
const data = {};
formData.forEach((value, key) => {
if (key === 'max_file_size' || key === 'cache_expire_days') {
data[key] = parseInt(value) || 0;
} else if (key === 'enable_cache' || key === 'enable_guest' || key === 'enable_email_notify') {
data[key] = document.getElementById(key).checked;
} else {
data[key] = value;
}
});
fetch('/admin/settings/site', {
method: 'POST',
headers: {'Content-Type': 'application/json'},
body: JSON.stringify(data)
})
.then(r => r.json())
.then(res => {
const div = document.getElementById('saveResult');
div.style.display = 'block';
if (res.success) {
div.innerHTML = '<div class="alert alert-success"><i class="bi bi-check-circle"></i> 配置已保存!</div>';
setTimeout(() => div.style.display = 'none', 2000);
} else {
div.innerHTML = '<div class="alert alert-danger"><i class="bi bi-x-circle"></i> 保存失败: ' + (res.error || '未知错误') + '</div>';
}
})
.catch(err => {
const div = document.getElementById('saveResult');
div.style.display = 'block';
div.innerHTML = '<div class="alert alert-danger">请求失败: ' + err + '</div>';
});
});
</script>
</body>
</html>

View File

@@ -69,13 +69,14 @@
<th>大小</th>
<th>状态</th>
<th>缓存</th>
<th>共享</th>
<th>时间</th>
<th>操作</th>
</tr>
</thead>
<tbody>
{% for t in translations.items %}
<tr>
<tr id="trans-row-{{ t.id }}">
<td>{{ t.id }}</td>
<td>{{ t.original_filename[:25] }}{% if t.original_filename|length > 25 %}...{% endif %}</td>
<td>{% if t.user_id %}ID:{{ t.user_id }}{% else %}访客{% endif %}</td>
@@ -87,14 +88,26 @@
</span>
</td>
<td>{% if t.from_cache %}<i class="bi bi-check-circle text-success"></i>{% else %}-{% endif %}</td>
<td>
{% if t.no_share %}
<span class="badge bg-secondary" id="share-badge-{{ t.id }}">不共享</span>
{% else %}
<span class="badge bg-success" id="share-badge-{{ t.id }}">共享</span>
{% endif %}
</td>
<td>{{ t.created_at.strftime('%m-%d %H:%M') }}</td>
<td>
<a href="{{ url_for('admin.translation_detail', trans_id=t.id) }}" class="btn btn-sm btn-outline-primary"><i class="bi bi-eye"></i></a>
<button class="btn btn-sm btn-outline-danger" onclick="deleteTrans({{ t.id }})"><i class="bi bi-trash"></i></button>
<div class="d-flex gap-1">
<button class="btn btn-sm btn-outline-{% if t.no_share %}success{% else %}warning{% endif %}" onclick="toggleShare({{ t.id }})" title="切换共享状态">
<i class="bi bi-share{% if t.no_share %}-fill{% endif %}"></i>
</button>
<a href="{{ url_for('admin.translation_detail', trans_id=t.id) }}" class="btn btn-sm btn-outline-primary"><i class="bi bi-eye"></i></a>
<button class="btn btn-sm btn-outline-danger" onclick="deleteTrans({{ t.id }})"><i class="bi bi-trash"></i></button>
</div>
</td>
</tr>
{% else %}
<tr><td colspan="9" class="text-center text-muted py-4">暂无数据</td></tr>
<tr><td colspan="10" class="text-center text-muted py-4">暂无数据</td></tr>
{% endfor %}
</tbody>
</table>
@@ -122,6 +135,26 @@
.then(r => r.json())
.then(d => { if (d.success) location.reload(); });
}
function toggleShare(id) {
fetch(`/admin/translation/${id}/toggle-share`, { method: 'POST' })
.then(r => r.json())
.then(d => {
if (d.success) {
// 更新显示状态
const badge = document.getElementById(`share-badge-${id}`);
if (d.no_share) {
badge.className = 'badge bg-secondary';
badge.textContent = '不共享';
} else {
badge.className = 'badge bg-success';
badge.textContent = '共享';
}
// 更新按钮样式
location.reload(); // 简化处理,刷新页面
}
});
}
</script>
</body>
</html>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>翻译历史 - PDF翻译助手</title>
<title>翻译历史 - {{ site_config.site_name }}</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -11,7 +11,7 @@
<body>
<nav class="navbar navbar-expand-lg navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
<div class="navbar-nav ms-auto">
<span class="nav-link text-light">👋 {{ user.username }}</span>
<a class="nav-link" href="/logout">退出</a>
@@ -59,5 +59,12 @@
</main>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
<!-- 页脚 -->
<footer class="bg-light py-4 mt-5">
<div class="container text-center">
{{ site_config.site_footer | safe }}
</div>
</footer>
</body>
</html>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>PDF翻译助手 - 英文PDF翻译中文</title>
<title>{{ site_config.site_name }} - 英文PDF翻译中文</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -12,7 +12,7 @@
<!-- 导航栏 -->
<nav class="navbar navbar-expand-lg navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav">
<span class="navbar-toggler-icon"></span>
</button>
@@ -132,7 +132,7 @@
<div class="card-body">
<form id="uploadForm">
<div class="mb-3">
<label class="form-label">选择PDF文件最大{{ max_pages }}页)</label>
<label class="form-label">选择PDF文件最大{{ max_pages }}页{{ site_config.max_file_size }}MB</label>
<input type="file" class="form-control" id="pdfFile" name="file" accept=".pdf" required>
<div class="form-text">支持英文PDF翻译为中文</div>
</div>
@@ -145,7 +145,7 @@
</div>
{% endif %}
<button type="submit" class="btn btn-primary btn-lg w-100" id="submitBtn">
<button type="submit" class="btn btn-primary btn-lg w-100" id="submitBtn" data-max-size="{{ site_config.max_file_size }}">
<span id="btnText">开始翻译</span>
<span id="btnSpinner" class="spinner-border spinner-border-sm" style="display:none"></span>
</button>
@@ -196,7 +196,7 @@
<!-- 页脚 -->
<footer class="bg-light py-4 mt-5">
<div class="container text-center">
<p class="text-muted">PDF翻译助手 v1.0.0 | 基于本地LLM服务</p>
{{ site_config.site_footer | safe }}
</div>
</footer>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>登录 - PDF翻译助手</title>
<title>登录 - {{ site_config.site_name }}</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -11,7 +11,7 @@
<body class="bg-light">
<nav class="navbar navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
</div>
</nav>
@@ -77,5 +77,12 @@
}
});
</script>
<!-- 页脚 -->
<footer class="bg-light py-4 mt-5">
<div class="container text-center">
{{ site_config.site_footer | safe }}
</div>
</footer>
</body>
</html>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>会员套餐 - PDF翻译助手</title>
<title>会员套餐 - {{ site_config.site_name }}</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -11,7 +11,7 @@
<body>
<nav class="navbar navbar-expand-lg navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav">
<span class="navbar-toggler-icon"></span>
</button>
@@ -219,5 +219,12 @@
</main>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
<!-- 页脚 -->
<footer class="bg-light py-4 mt-5">
<div class="container text-center">
{{ site_config.site_footer | safe }}
</div>
</footer>
</body>
</html>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>个人中心 - PDF翻译助手</title>
<title>个人中心 - {{ site_config.site_name }}</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -15,7 +15,7 @@
<body>
<nav class="navbar navbar-expand-lg navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav">
<span class="navbar-toggler-icon"></span>
</button>
@@ -624,5 +624,12 @@
});
}
</script>
<!-- 页脚 -->
<footer class="bg-light py-4 mt-5">
<div class="container text-center">
{{ site_config.site_footer | safe }}
</div>
</footer>
</body>
</html>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>注册 - PDF翻译助手</title>
<title>注册 - {{ site_config.site_name }}</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -11,7 +11,7 @@
<body class="bg-light">
<nav class="navbar navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
</div>
</nav>
@@ -79,5 +79,12 @@
}
});
</script>
<!-- 页脚 -->
<footer class="bg-light py-4 mt-5">
<div class="container text-center">
{{ site_config.site_footer | safe }}
</div>
</footer>
</body>
</html>

View File

@@ -3,7 +3,7 @@
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>翻译详情 - PDF翻译助手</title>
<title>翻译详情 - {{ site_config.site_name }}</title>
<link rel="icon" href="/static/img/favicon.svg" type="image/svg+xml">
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
<link href="/static/css/style.css" rel="stylesheet">
@@ -11,7 +11,7 @@
<body>
<nav class="navbar navbar-expand-lg navbar-dark bg-primary">
<div class="container">
<a class="navbar-brand" href="/">📄 PDF翻译助手</a>
<a class="navbar-brand" href="/">📄 {{ site_config.site_name }}</a>
<div class="navbar-nav ms-auto">
{% if user %}
<span class="nav-link text-light">👋 {{ user.username }}</span>