fix(front): 优化前端显示

This commit is contained in:
cnlimiter
2026-03-18 19:43:04 +08:00
parent 4fe5b177b9
commit 74d3b4c0de
4 changed files with 144 additions and 53 deletions

View File

@@ -85,13 +85,13 @@ const elements = {
// 注册后自动操作
autoUploadCpa: document.getElementById('auto-upload-cpa'),
cpaServiceSelectGroup: document.getElementById('cpa-service-select-group'),
cpaServiceCheckboxes: document.getElementById('cpa-service-checkboxes'),
cpaServiceSelect: document.getElementById('cpa-service-select'),
autoUploadSub2api: document.getElementById('auto-upload-sub2api'),
sub2apiServiceSelectGroup: document.getElementById('sub2api-service-select-group'),
sub2apiServiceCheckboxes: document.getElementById('sub2api-service-checkboxes'),
sub2apiServiceSelect: document.getElementById('sub2api-service-select'),
autoUploadTm: document.getElementById('auto-upload-tm'),
tmServiceSelectGroup: document.getElementById('tm-service-select-group'),
tmServiceCheckboxes: document.getElementById('tm-service-checkboxes'),
tmServiceSelect: document.getElementById('tm-service-select'),
};
// 初始化
@@ -108,14 +108,14 @@ document.addEventListener('DOMContentLoaded', () => {
// 初始化注册后自动操作选项CPA / Sub2API / TM
async function initAutoUploadOptions() {
await Promise.all([
loadServiceCheckboxes('cpa', '/cpa-services?enabled=true', elements.cpaServiceCheckboxes, elements.autoUploadCpa, elements.cpaServiceSelectGroup),
loadServiceCheckboxes('sub2api', '/sub2api-services?enabled=true', elements.sub2apiServiceCheckboxes, elements.autoUploadSub2api, elements.sub2apiServiceSelectGroup),
loadServiceCheckboxes('tm', '/tm-services?enabled=true', elements.tmServiceCheckboxes, elements.autoUploadTm, elements.tmServiceSelectGroup),
loadServiceSelect('/cpa-services?enabled=true', elements.cpaServiceSelect, elements.autoUploadCpa, elements.cpaServiceSelectGroup),
loadServiceSelect('/sub2api-services?enabled=true', elements.sub2apiServiceSelect, elements.autoUploadSub2api, elements.sub2apiServiceSelectGroup),
loadServiceSelect('/tm-services?enabled=true', elements.tmServiceSelect, elements.autoUploadTm, elements.tmServiceSelectGroup),
]);
}
// 通用:加载服务 checkbox 列表,并处理联动
async function loadServiceCheckboxes(type, apiPath, container, checkbox, selectGroup) {
// 通用:构建自定义多选下拉组件并处理联动
async function loadServiceSelect(apiPath, container, checkbox, selectGroup) {
if (!checkbox || !container) return;
let services = [];
try {
@@ -127,14 +127,31 @@ async function loadServiceCheckboxes(type, apiPath, container, checkbox, selectG
checkbox.title = '请先在设置中添加对应服务';
const label = checkbox.closest('label');
if (label) label.style.opacity = '0.5';
if (container) container.innerHTML = '<div style="color:var(--text-muted);font-size:0.8rem;">暂无可用服务</div>';
container.innerHTML = '<div class="msd-empty">暂无可用服务</div>';
} else {
container.innerHTML = services.map(s => `
<label style="display:flex;align-items:center;gap:6px;cursor:pointer;">
<input type="checkbox" class="upload-svc-cb upload-svc-${type}" value="${s.id}" checked>
<span style="font-size:0.85rem;">${escapeHtml(s.name)}</span>
</label>
`).join('');
const items = services.map(s =>
`<label class="msd-item">
<input type="checkbox" value="${s.id}" checked>
<span>${escapeHtml(s.name)}</span>
</label>`
).join('');
container.innerHTML = `
<div class="msd-dropdown" id="${container.id}-dd">
<div class="msd-trigger" onclick="toggleMsd('${container.id}-dd')">
<span class="msd-label">全部 (${services.length})</span>
<span class="msd-arrow">▼</span>
</div>
<div class="msd-list">${items}</div>
</div>`;
// 监听 checkbox 变化,更新触发器文字
container.querySelectorAll('.msd-item input').forEach(cb => {
cb.addEventListener('change', () => updateMsdLabel(container.id + '-dd'));
});
// 点击外部关闭
document.addEventListener('click', (e) => {
const dd = document.getElementById(container.id + '-dd');
if (dd && !dd.contains(e.target)) dd.classList.remove('open');
}, true);
}
// 联动显示/隐藏服务选择区
@@ -143,13 +160,27 @@ async function loadServiceCheckboxes(type, apiPath, container, checkbox, selectG
});
}
// 获取选中的服务 ID 列表
function getCheckedServiceIds(type) {
const ids = [];
document.querySelectorAll(`.upload-svc-${type}:checked`).forEach(cb => {
ids.push(parseInt(cb.value));
});
return ids;
function toggleMsd(ddId) {
const dd = document.getElementById(ddId);
if (dd) dd.classList.toggle('open');
}
function updateMsdLabel(ddId) {
const dd = document.getElementById(ddId);
if (!dd) return;
const all = dd.querySelectorAll('.msd-item input');
const checked = dd.querySelectorAll('.msd-item input:checked');
const label = dd.querySelector('.msd-label');
if (!label) return;
if (checked.length === 0) label.textContent = '未选择';
else if (checked.length === all.length) label.textContent = `全部 (${all.length})`;
else label.textContent = Array.from(checked).map(c => c.nextElementSibling.textContent).join(', ');
}
// 获取自定义多选下拉中选中的服务 ID 列表
function getSelectedServiceIds(container) {
if (!container) return [];
return Array.from(container.querySelectorAll('.msd-item input:checked')).map(cb => parseInt(cb.value));
}
// 事件监听
@@ -395,11 +426,11 @@ async function handleStartRegistration(e) {
const requestData = {
email_service_type: emailServiceType,
auto_upload_cpa: elements.autoUploadCpa ? elements.autoUploadCpa.checked : false,
cpa_service_ids: elements.autoUploadCpa && elements.autoUploadCpa.checked ? getCheckedServiceIds('cpa') : [],
cpa_service_ids: elements.autoUploadCpa && elements.autoUploadCpa.checked ? getSelectedServiceIds(elements.cpaServiceSelect) : [],
auto_upload_sub2api: elements.autoUploadSub2api ? elements.autoUploadSub2api.checked : false,
sub2api_service_ids: elements.autoUploadSub2api && elements.autoUploadSub2api.checked ? getCheckedServiceIds('sub2api') : [],
sub2api_service_ids: elements.autoUploadSub2api && elements.autoUploadSub2api.checked ? getSelectedServiceIds(elements.sub2apiServiceSelect) : [],
auto_upload_tm: elements.autoUploadTm ? elements.autoUploadTm.checked : false,
tm_service_ids: elements.autoUploadTm && elements.autoUploadTm.checked ? getCheckedServiceIds('tm') : [],
tm_service_ids: elements.autoUploadTm && elements.autoUploadTm.checked ? getSelectedServiceIds(elements.tmServiceSelect) : [],
};
// 如果选择了数据库中的服务,传递 service_id
@@ -1099,11 +1130,11 @@ async function handleOutlookBatchRegistration() {
concurrency: Math.min(50, Math.max(1, concurrency)),
mode: mode,
auto_upload_cpa: elements.autoUploadCpa ? elements.autoUploadCpa.checked : false,
cpa_service_ids: elements.autoUploadCpa && elements.autoUploadCpa.checked ? getCheckedServiceIds('cpa') : [],
cpa_service_ids: elements.autoUploadCpa && elements.autoUploadCpa.checked ? getSelectedServiceIds(elements.cpaServiceSelect) : [],
auto_upload_sub2api: elements.autoUploadSub2api ? elements.autoUploadSub2api.checked : false,
sub2api_service_ids: elements.autoUploadSub2api && elements.autoUploadSub2api.checked ? getCheckedServiceIds('sub2api') : [],
sub2api_service_ids: elements.autoUploadSub2api && elements.autoUploadSub2api.checked ? getSelectedServiceIds(elements.sub2apiServiceSelect) : [],
auto_upload_tm: elements.autoUploadTm ? elements.autoUploadTm.checked : false,
tm_service_ids: elements.autoUploadTm && elements.autoUploadTm.checked ? getCheckedServiceIds('tm') : [],
tm_service_ids: elements.autoUploadTm && elements.autoUploadTm.checked ? getSelectedServiceIds(elements.tmServiceSelect) : [],
};
addLog('info', `[系统] 正在启动 Outlook 批量注册 (${selectedIds.length} 个账户)...`);

View File

@@ -1076,7 +1076,7 @@ function renderTmServicesTable(services) {
</span>
</td>
<td style="text-align:center;">${s.priority}</td>
<td>
<td style="white-space:nowrap;">
<button class="btn btn-secondary btn-sm" onclick="editTmService(${s.id})">编辑</button>
<button class="btn btn-secondary btn-sm" onclick="testTmServiceById(${s.id})">测试</button>
<button class="btn btn-danger btn-sm" onclick="deleteTmService(${s.id}, '${escapeHtml(s.name)}')">删除</button>
@@ -1241,7 +1241,7 @@ function renderCpaServicesTable(services) {
</span>
</td>
<td style="text-align:center;">${s.priority}</td>
<td>
<td style="white-space:nowrap;">
<button class="btn btn-secondary btn-sm" onclick="editCpaService(${s.id})">编辑</button>
<button class="btn btn-secondary btn-sm" onclick="testCpaServiceById(${s.id})">测试</button>
<button class="btn btn-danger btn-sm" onclick="deleteCpaService(${s.id}, '${escapeHtml(s.name)}')">删除</button>
@@ -1395,20 +1395,23 @@ async function loadSub2ApiServices() {
function renderSub2ApiServices(services) {
if (!elements.sub2ApiServicesTable) return;
if (!services || services.length === 0) {
elements.sub2ApiServicesTable.innerHTML = '<tr><td colspan="5" style="text-align:center;color:var(--text-muted);padding:20px;">暂无服务,点击"添加服务"按钮添加</td></tr>';
elements.sub2ApiServicesTable.innerHTML = '<tr><td colspan="5" style="text-align:center;color:var(--text-muted);padding:20px;">暂无 Sub2API 服务,点击添加服务」新增</td></tr>';
return;
}
elements.sub2ApiServicesTable.innerHTML = services.map(s => `
<tr>
<td>${escapeHtml(s.name)}</td>
<td><code>${escapeHtml(s.api_url)}</code></td>
<td><span class="status-badge ${s.enabled ? 'active' : 'disabled'}">${s.enabled ? '已启用' : '已禁用'}</span></td>
<td>${s.priority}</td>
<td style="font-size:0.85rem;color:var(--text-muted);">${escapeHtml(s.api_url)}</td>
<td>
<div class="action-buttons">
<button class="btn btn-ghost btn-sm" onclick="editSub2ApiService(${s.id})" title="编辑">✏️</button>
<button class="btn btn-ghost btn-sm" onclick="deleteSub2ApiService(${s.id}, '${escapeHtml(s.name)}')" title="删除">🗑️</button>
</div>
<span class="badge" style="background:${s.enabled ? 'var(--success-color)' : 'var(--border)'};color:${s.enabled ? '#fff' : 'var(--text-muted)'};font-size:0.75rem;padding:2px 8px;border-radius:10px;">
${s.enabled ? '启用' : '禁用'}
</span>
</td>
<td style="text-align:center;">${s.priority}</td>
<td style="white-space:nowrap;">
<button class="btn btn-secondary btn-sm" onclick="editSub2ApiService(${s.id})">编辑</button>
<button class="btn btn-secondary btn-sm" onclick="testSub2ApiServiceById(${s.id})">测试</button>
<button class="btn btn-danger btn-sm" onclick="deleteSub2ApiService(${s.id}, '${escapeHtml(s.name)}')">删除</button>
</td>
</tr>
`).join('');
@@ -1486,6 +1489,19 @@ async function handleSaveSub2ApiService(e) {
}
}
async function testSub2ApiServiceById(id) {
try {
const result = await api.post(`/sub2api-services/${id}/test`);
if (result.success) {
toast.success(result.message);
} else {
toast.error(result.message);
}
} catch (e) {
toast.error('测试失败: ' + e.message);
}
}
async function handleTestSub2ApiService() {
const apiUrl = document.getElementById('sub2api-service-url').value.trim();
const apiKey = document.getElementById('sub2api-service-key').value.trim();