Add files via upload

This commit is contained in:
公明
2026-04-21 21:44:12 +08:00
committed by GitHub
parent 5b3f4e3556
commit 58d2e20274
3 changed files with 117 additions and 9 deletions
+94
View File
@@ -3590,6 +3590,83 @@ header {
margin-bottom: 32px;
}
.mcp-management-layout {
display: grid;
grid-template-columns: minmax(0, 2fr) minmax(0, 3fr);
gap: 16px;
align-items: stretch;
height: calc(100vh - 210px);
min-height: 520px;
max-height: calc(100vh - 210px);
}
.mcp-management-panel {
margin-bottom: 0 !important;
padding: 14px 16px 16px;
border: 1px solid var(--border-color);
border-radius: 12px;
background: var(--bg-primary);
display: flex;
flex-direction: column;
min-height: 0;
}
.mcp-tools-panel {
min-width: 0;
order: 2;
}
.mcp-external-panel {
min-width: 0;
order: 1;
}
.mcp-panel-body {
display: flex;
flex-direction: column;
flex: 1;
min-height: 0;
}
.mcp-tools-panel .tools-controls,
.mcp-external-panel .external-mcp-controls {
flex: 1;
min-height: 0;
}
.mcp-tools-panel .tools-list {
flex: 1;
min-height: 0;
max-height: 100%;
overflow-y: auto;
}
.mcp-tools-panel .tools-list-items {
max-height: none;
overflow: visible;
}
.mcp-external-panel .external-mcp-list {
flex: 1;
min-height: 0;
overflow-y: auto;
padding-right: 4px;
}
.mcp-external-panel .external-mcp-controls {
gap: 12px;
}
/* MCP 双栏内工具操作条允许换行,避免面板内溢出 */
.mcp-tools-panel .tools-actions {
flex-wrap: wrap;
row-gap: 8px;
}
.mcp-tools-panel .search-box {
min-width: min(280px, 100%);
}
.settings-section:last-child {
margin-bottom: 0;
}
@@ -5396,6 +5473,23 @@ header {
/* 响应式优化 */
@media (max-width: 768px) {
.mcp-management-layout {
grid-template-columns: 1fr;
height: auto;
min-height: auto;
max-height: none;
}
.mcp-management-panel {
min-height: 0;
}
.mcp-tools-panel .tools-list,
.mcp-external-panel .external-mcp-list {
min-height: 200px;
max-height: 52vh;
}
.tools-actions {
gap: 6px;
}
+13 -3
View File
@@ -519,7 +519,7 @@ function renderToolsList() {
toolItem.innerHTML = `
<input type="checkbox" id="${checkboxId}" ${toolState.enabled ? 'checked' : ''} ${toolState.is_external || tool.is_external ? 'data-external="true"' : ''} onchange="handleToolCheckboxChange('${escapeHtml(toolKey)}', this.checked)" />
<div class="tool-item-info" onclick="toggleToolDetail(this, '${escapeHtml(toolKey)}', ${tool.is_external ? 'true' : 'false'}, '${escapeHtml(tool.external_mcp || '')}', event)">
<div class="tool-item-info">
<div class="tool-item-name">
${escapeHtml(tool.name)}
${externalBadge}
@@ -529,6 +529,11 @@ function renderToolsList() {
<div class="tool-item-detail" style="display:none"></div>
</div>
`;
toolItem.addEventListener('click', function (event) {
const infoEl = toolItem.querySelector('.tool-item-info');
if (!infoEl) return;
toggleToolDetail(infoEl, toolKey, !!tool.is_external, tool.external_mcp || '', event);
});
listContainer.appendChild(toolItem);
});
@@ -543,14 +548,16 @@ function renderToolsList() {
// 展开/折叠工具详情面板(按需从后端加载 schema)
function toggleToolDetail(infoEl, toolKey, isExternal, externalMcp, event) {
// 点击 checkbox 或外部工具徽章时不展开
if (event.target.tagName === 'INPUT' || event.target.closest('.external-tool-badge')) return;
if (event && (event.target.tagName === 'INPUT' || event.target.closest('.external-tool-badge'))) return;
const detail = infoEl.querySelector('.tool-item-detail');
const icon = infoEl.querySelector('.tool-expand-icon');
if (!detail) return;
const isOpen = detail.style.display !== 'none';
// 使用 data-open 作为主状态,避免仅依赖 style.display 带来的首击偶发判定不一致
const isOpen = detail.dataset.open === '1';
detail.style.display = isOpen ? 'none' : 'block';
detail.dataset.open = isOpen ? '0' : '1';
if (icon) icon.textContent = isOpen ? '▶' : '▼';
// 首次展开时从后端按需加载
@@ -1466,12 +1473,15 @@ async function pollExternalMCPToolCount(name, maxAttempts = 10) {
function renderExternalMCPList(servers) {
const list = document.getElementById('external-mcp-list');
if (!list) return;
const layout = document.querySelector('.mcp-management-layout');
if (Object.keys(servers).length === 0) {
if (layout) layout.classList.add('external-empty');
const emptyT = typeof window.t === 'function' ? window.t : (k) => k;
list.innerHTML = '<div class="empty">📋 ' + emptyT('mcp.noExternalMCP') + '<br><span style="font-size: 0.875rem; margin-top: 8px; display: block;">' + emptyT('mcp.clickToAddExternal') + '</span></div>';
return;
}
if (layout) layout.classList.remove('external-empty');
let html = '<div class="external-mcp-items">';
for (const [name, server] of Object.entries(servers)) {
+10 -6
View File
@@ -735,17 +735,17 @@
<h2 data-i18n="mcp.managementTitle">MCP 管理</h2>
<div class="page-header-actions">
<button class="btn-secondary" onclick="loadExternalMCPs()"><span data-i18n="common.refresh">刷新</span></button>
<button class="btn-primary" onclick="showAddExternalMCPModal()" data-i18n="mcp.addExternalMCP">添加外部MCP</button>
</div>
</div>
<div class="page-content">
<div class="mcp-management-layout">
<!-- MCP工具配置 -->
<div class="settings-section" style="margin-bottom: 32px;">
<div class="settings-section mcp-management-panel mcp-tools-panel" style="margin-bottom: 32px;">
<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 16px;">
<h3 style="margin: 0;" data-i18n="mcp.toolConfig">MCP 工具配置</h3>
<button class="btn-primary" onclick="saveToolsConfig()" data-i18n="mcp.saveToolConfig">保存工具配置</button>
</div>
<div class="tools-controls">
<div class="tools-controls mcp-panel-body">
<div class="tools-actions">
<button class="btn-secondary" onclick="selectAllTools()" data-i18n="mcp.selectAll">全选</button>
<button class="btn-secondary" onclick="deselectAllTools()" data-i18n="mcp.deselectAll">全不选</button>
@@ -765,15 +765,19 @@
</div>
<!-- 外部MCP配置 -->
<div class="settings-section">
<h3 data-i18n="mcp.externalConfig">外部 MCP 配置</h3>
<div class="external-mcp-controls">
<div class="settings-section mcp-management-panel mcp-external-panel">
<div style="display: flex; justify-content: space-between; align-items: center; margin-bottom: 16px;">
<h3 style="margin: 0;" data-i18n="mcp.externalConfig">外部 MCP 配置</h3>
<button class="btn-primary" onclick="showAddExternalMCPModal()" data-i18n="mcp.addExternalMCP">添加外部MCP</button>
</div>
<div class="external-mcp-controls mcp-panel-body">
<div class="external-mcp-actions">
<div class="external-mcp-stats" id="external-mcp-stats"></div>
</div>
<div id="external-mcp-list" class="external-mcp-list"></div>
</div>
</div>
</div>
</div>
</div>