feat: ToNav-go v1.0.0 - 内部服务导航系统

功能:
- 前台导航: 分类Tab切换、实时搜索、健康状态指示、响应式适配
- 后台管理: 服务/分类CRUD、系统设置、登录认证(bcrypt)
- 健康检查: 定时检测(5min)、独立检查URL、三态指示(在线/离线/未检测)
- 云端备份: WebDAV上传/下载/恢复/删除、定时自动备份、本地备份管理

技术栈: Go + Gin + GORM + SQLite
This commit is contained in:
2026-02-14 05:09:23 +08:00
commit efaf787981
23 changed files with 2735 additions and 0 deletions

View File

@@ -0,0 +1,289 @@
<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>服务管理 - ToNav</title>
<style>
:root { --main-red: #ff4d4f; --primary: #667eea; }
* { margin: 0; padding: 0; box-sizing: border-box; }
body { font-family: "PingFang SC", -apple-system, sans-serif; background: #f0f2f5; padding: 20px; }
a { text-decoration: none; color: inherit; }
.container { max-width: 1000px; margin: 0 auto; }
.header { display: flex; justify-content: space-between; align-items: center; background: #fff; padding: 20px 25px; border-radius: 15px; margin-bottom: 20px; box-shadow: 0 2px 8px rgba(0,0,0,0.06); }
.btn { padding: 10px 20px; border-radius: 8px; border: none; cursor: pointer; font-weight: bold; display: inline-block; font-size: 14px; transition: all .2s; }
.btn:hover { transform: translateY(-1px); }
.btn-primary { background: var(--main-red); color: #fff; }
.btn-back { background: #eee; color: #333; }
.header-actions { display: flex; gap: 10px; }
.table-card { background: #fff; border-radius: 15px; overflow: hidden; box-shadow: 0 2px 8px rgba(0,0,0,0.06); }
table { width: 100%; border-collapse: collapse; }
th { background: #fafafa; font-size: 13px; color: #999; text-align: left; padding: 12px 15px; font-weight: 500; }
td { padding: 12px 15px; border-bottom: 1px solid #f5f5f5; font-size: 14px; }
tr:hover td { background: #fafafa; }
.status-badge { padding: 3px 10px; border-radius: 10px; font-size: 12px; font-weight: 500; }
.status-online { background: #f6ffed; color: #52c41a; }
.status-offline { background: #fff2f0; color: #ff4d4f; }
.btn-edit { color: var(--primary); background: none; border: none; cursor: pointer; padding: 4px 8px; border-radius: 4px; }
.btn-edit:hover { background: rgba(102,126,234,0.1); }
.btn-del { color: var(--main-red); background: none; border: none; cursor: pointer; padding: 4px 8px; border-radius: 4px; }
.btn-del:hover { background: rgba(255,77,79,0.1); }
.empty { text-align: center; padding: 40px; color: #999; }
.icon-cell { font-size: 24px; }
/* 弹窗 */
.modal-overlay { display: none; position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0,0,0,0.5); align-items: center; justify-content: center; z-index: 100; }
.modal-content { background: #fff; padding: 30px; border-radius: 20px; width: 90%; max-width: 550px; max-height: 85vh; overflow-y: auto; }
.input { width: 100%; padding: 10px 12px; border: 1px solid #d9d9d9; border-radius: 8px; box-sizing: border-box; margin-bottom: 12px; font-size: 14px; }
.input:focus { border-color: var(--primary); outline: none; box-shadow: 0 0 0 2px rgba(102,126,234,0.2); }
select.input { appearance: none; -webkit-appearance: none; background: #fff url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23343a40' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M2 5l6 6 6-6'/%3e%3c/svg%3e") no-repeat right 12px center/12px; }
.form-row { display: flex; gap: 12px; }
.form-row > div { flex: 1; }
label { font-size: 13px; color: #666; margin-bottom: 4px; display: block; }
.checkbox-row { display: flex; align-items: center; gap: 8px; margin-bottom: 12px; }
.checkbox-row input[type="checkbox"] { width: 16px; height: 16px; }
.modal-actions { display: flex; gap: 10px; margin-top: 15px; }
.modal-actions .btn { flex: 1; }
@media (max-width: 640px) {
body { padding: 12px; }
.header { flex-direction: column; gap: 12px; text-align: center; padding: 15px; }
.header h1 { font-size: 20px; }
.header-actions { width: 100%; }
.header-actions .btn { flex: 1; text-align: center; }
.form-row { flex-direction: column; gap: 0; }
/* 手机端隐藏状态和排序列 */
th:nth-child(3), td:nth-child(3),
th:nth-child(4), td:nth-child(4),
th:nth-child(5), td:nth-child(5) { display: none; }
th, td { padding: 10px 8px; font-size: 13px; }
.icon-cell { font-size: 20px; }
.modal-content { padding: 20px; border-radius: 16px; }
}
</style>
</head>
<body>
<div class="container">
<div class="header">
<h1>📡 服务管理</h1>
<div class="header-actions">
<button class="btn btn-primary" onclick="openModal()">+ 新增服务</button>
<a href="/admin/dashboard" class="btn btn-back">返回</a>
</div>
</div>
<div class="table-card">
<table>
<thead>
<tr>
<th>图标</th>
<th>名称</th>
<th>分类</th>
<th>状态</th>
<th>排序</th>
<th>操作</th>
</tr>
</thead>
<tbody id="list"></tbody>
</table>
<div id="emptyMsg" class="empty" style="display:none">暂无服务,点击上方"+ 新增服务"添加</div>
</div>
</div>
<!-- 新增/编辑弹窗 -->
<div id="modal" class="modal-overlay">
<div class="modal-content">
<h2 id="modalTitle" style="margin-bottom:20px">新增服务</h2>
<input type="hidden" id="editId">
<label>服务名称 *</label>
<input type="text" id="svcName" class="input" placeholder="如Gitea" required>
<label>服务地址 *</label>
<input type="url" id="svcUrl" class="input" placeholder="https://example.com">
<div class="form-row">
<div>
<label>图标emoji</label>
<input type="text" id="svcIcon" class="input" placeholder="🔧" maxlength="4">
</div>
<div>
<label>所属分类</label>
<select id="svcCategory" class="input">
<option value="0">-- 无分类 --</option>
{{ range .categories }}
<option value="{{ .ID }}">{{ .Name }}</option>
{{ end }}
</select>
</div>
</div>
<label>描述</label>
<input type="text" id="svcDesc" class="input" placeholder="简短描述">
<label>标签(逗号分隔)</label>
<input type="text" id="svcTags" class="input" placeholder="开发,工具">
<div class="form-row">
<div>
<label>排序</label>
<input type="number" id="svcOrder" class="input" value="0">
</div>
<div>
<label>健康检查 URL</label>
<input type="url" id="svcHealthUrl" class="input" placeholder="可选">
</div>
</div>
<div class="checkbox-row">
<input type="checkbox" id="svcEnabled" checked>
<label for="svcEnabled" style="margin:0">启用</label>
</div>
<div class="checkbox-row">
<input type="checkbox" id="svcHealthEnabled">
<label for="svcHealthEnabled" style="margin:0">启用健康检查</label>
</div>
<div class="modal-actions">
<button class="btn" style="background:#eee" onclick="closeModal()">取消</button>
<button class="btn btn-primary" onclick="save()">保存</button>
</div>
</div>
</div>
<script>
// 分类映射
const categoryMap = {};
{{ range .categories }}
categoryMap[{{ .ID }}] = "{{ .Name }}";
{{ end }}
async function load() {
const res = await fetch('/admin/api/services');
const data = await res.json();
const list = document.getElementById('list');
const emptyMsg = document.getElementById('emptyMsg');
if (!data.data || data.data.length === 0) {
list.innerHTML = '';
emptyMsg.style.display = 'block';
return;
}
emptyMsg.style.display = 'none';
list.innerHTML = data.data.map(s => `
<tr>
<td class="icon-cell">${s.icon || '🔗'}</td>
<td><strong>${escapeHTML(s.name)}</strong></td>
<td>${categoryMap[s.category_id] || '-'}</td>
<td><span class="status-badge status-${s.status}">${s.status === 'online' ? '在线' : '离线'}</span></td>
<td>${s.sort_order}</td>
<td>
<button class="btn-edit" onclick='editSvc(${JSON.stringify(s)})'>编辑</button>
<button class="btn-del" onclick="del(${s.id}, '${escapeHTML(s.name)}')">删除</button>
</td>
</tr>
`).join('');
}
function openModal() {
document.getElementById('editId').value = '';
document.getElementById('svcName').value = '';
document.getElementById('svcUrl').value = '';
document.getElementById('svcIcon').value = '';
document.getElementById('svcCategory').value = '0';
document.getElementById('svcDesc').value = '';
document.getElementById('svcTags').value = '';
document.getElementById('svcOrder').value = '0';
document.getElementById('svcHealthUrl').value = '';
document.getElementById('svcEnabled').checked = true;
document.getElementById('svcHealthEnabled').checked = false;
document.getElementById('modalTitle').textContent = '新增服务';
document.getElementById('modal').style.display = 'flex';
document.getElementById('svcName').focus();
}
function editSvc(s) {
document.getElementById('editId').value = s.id;
document.getElementById('svcName').value = s.name;
document.getElementById('svcUrl').value = s.url;
document.getElementById('svcIcon').value = s.icon || '';
document.getElementById('svcCategory').value = s.category_id || '0';
document.getElementById('svcDesc').value = s.description || '';
document.getElementById('svcTags').value = s.tags || '';
document.getElementById('svcOrder').value = s.sort_order || 0;
document.getElementById('svcHealthUrl').value = s.health_check_url || '';
document.getElementById('svcEnabled').checked = s.is_enabled;
document.getElementById('svcHealthEnabled').checked = s.health_check_enabled;
document.getElementById('modalTitle').textContent = '编辑服务';
document.getElementById('modal').style.display = 'flex';
}
function closeModal() {
document.getElementById('modal').style.display = 'none';
}
async function save() {
const id = document.getElementById('editId').value;
const name = document.getElementById('svcName').value.trim();
const url = document.getElementById('svcUrl').value.trim();
if (!name) { alert('服务名称不能为空'); return; }
if (!url) { alert('服务地址不能为空'); return; }
const body = {
name,
url,
icon: document.getElementById('svcIcon').value.trim(),
category_id: parseInt(document.getElementById('svcCategory').value) || 0,
description: document.getElementById('svcDesc').value.trim(),
tags: document.getElementById('svcTags').value.trim(),
sort_order: parseInt(document.getElementById('svcOrder').value) || 0,
health_check_url: document.getElementById('svcHealthUrl').value.trim(),
is_enabled: document.getElementById('svcEnabled').checked,
health_check_enabled: document.getElementById('svcHealthEnabled').checked,
};
if (id) body.id = parseInt(id);
const apiUrl = id ? `/admin/api/services/${id}` : '/admin/api/services';
const method = id ? 'PUT' : 'POST';
const resp = await fetch(apiUrl, {
method,
headers: {'Content-Type': 'application/json'},
body: JSON.stringify(body)
});
const res = await resp.json();
if (res.success) {
closeModal();
load();
} else {
alert(res.message || '保存失败');
}
}
async function del(id, name) {
if (!confirm(`确定删除服务「${name}」?`)) return;
const resp = await fetch('/admin/api/services/' + id, { method: 'DELETE' });
const res = await resp.json();
if (res.success) {
load();
} else {
alert(res.message || '删除失败');
}
}
function escapeHTML(str) {
const div = document.createElement('div');
div.textContent = str || '';
return div.innerHTML;
}
document.addEventListener('keydown', e => {
if (e.key === 'Escape') closeModal();
});
load();
</script>
</body>
</html>