convex-mcp-visual 1.4.4 → 1.4.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/apps/apps/kanban-board/index.html +2 -2
- package/dist/apps/apps/realtime-dashboard/index.html +14 -14
- package/dist/apps/apps/schema-browser/index.html +14 -14
- package/dist/apps/assets/style-C_-VyIe_.css +1 -0
- package/dist/resources/dashboard.js +13 -13
- package/dist/resources/kanban-board.d.ts +7 -0
- package/dist/resources/kanban-board.d.ts.map +1 -0
- package/dist/resources/kanban-board.js +202 -0
- package/dist/resources/kanban-board.js.map +1 -0
- package/dist/resources/schema-browser.js +10 -10
- package/dist/server.d.ts +11 -0
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +243 -29
- package/dist/server.js.map +1 -1
- package/package.json +1 -1
- package/dist/apps/assets/style-CfvT5wrm.css +0 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
<!doctype html>
|
|
2
|
-
<html lang="en">
|
|
2
|
+
<html lang="en" data-app="kanban-board">
|
|
3
3
|
<head>
|
|
4
4
|
<meta charset="UTF-8" />
|
|
5
5
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
@@ -7,7 +7,7 @@
|
|
|
7
7
|
<title>Kanban Board - Convex MCP Apps</title>
|
|
8
8
|
<script type="module" crossorigin src="../../assets/kanban-board-B3LBzk3P.js"></script>
|
|
9
9
|
<link rel="modulepreload" crossorigin href="../../assets/modulepreload-polyfill-B5Qt9EMX.js">
|
|
10
|
-
<link rel="stylesheet" crossorigin href="../../assets/style-
|
|
10
|
+
<link rel="stylesheet" crossorigin href="../../assets/style-C_-VyIe_.css">
|
|
11
11
|
</head>
|
|
12
12
|
<body>
|
|
13
13
|
<div id="app"></div>
|
|
@@ -1,15 +1,15 @@
|
|
|
1
|
-
<!
|
|
2
|
-
<html lang="en">
|
|
3
|
-
<head>
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
</head>
|
|
12
|
-
<body>
|
|
13
|
-
|
|
14
|
-
</body>
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="en" data-app="realtime-dashboard">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
+
<meta name="theme-color" content="#faf8f5" />
|
|
7
|
+
<title>Realtime Dashboard - Convex MCP Apps</title>
|
|
8
|
+
<script type="module" crossorigin src="../../assets/realtime-dashboard-D9nem8DB.js"></script>
|
|
9
|
+
<link rel="modulepreload" crossorigin href="../../assets/modulepreload-polyfill-B5Qt9EMX.js">
|
|
10
|
+
<link rel="stylesheet" crossorigin href="../../assets/style-C_-VyIe_.css">
|
|
11
|
+
</head>
|
|
12
|
+
<body>
|
|
13
|
+
<div id="app"></div>
|
|
14
|
+
</body>
|
|
15
15
|
</html>
|
|
@@ -1,15 +1,15 @@
|
|
|
1
|
-
<!
|
|
2
|
-
<html lang="en">
|
|
3
|
-
<head>
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
</head>
|
|
12
|
-
<body>
|
|
13
|
-
|
|
14
|
-
</body>
|
|
1
|
+
<!doctype html>
|
|
2
|
+
<html lang="en" data-app="schema-browser">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
+
<meta name="theme-color" content="#faf8f5" />
|
|
7
|
+
<title>Schema Browser - Convex MCP Apps</title>
|
|
8
|
+
<script type="module" crossorigin src="../../assets/schema-browser-S5aw9ecT.js"></script>
|
|
9
|
+
<link rel="modulepreload" crossorigin href="../../assets/modulepreload-polyfill-B5Qt9EMX.js">
|
|
10
|
+
<link rel="stylesheet" crossorigin href="../../assets/style-C_-VyIe_.css">
|
|
11
|
+
</head>
|
|
12
|
+
<body>
|
|
13
|
+
<div id="app"></div>
|
|
14
|
+
</body>
|
|
15
15
|
</html>
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
html,body{width:100%;height:100%;overflow:hidden}:root[data-app=schema-browser]{--bg-primary: #faf8f5;--bg-secondary: #f5f3f0;--bg-tertiary: #ebe9e6;--bg-hover: #ebe9e6;--bg-canvas: #faf8f5;--text-primary: #1a1a1a;--text-secondary: #6b6b6b;--text-muted: #999999;--border: #e6e4e1;--border-strong: #d4d2cf;--accent: #8b7355;--accent-interactive: #eb5601;--accent-hover: #d14a01;--success: #4a8c5c;--warning: #c4842d;--warning-bg: #fef3e2;--warning-text: #8a5a00;--error: #dc3545;--info: #4a7c9b;--font-mono: "SF Mono", Monaco, "Cascadia Code", "Courier New", monospace;--code-bg: #1e1e1e;--code-text: #d4d4d4;--code-key: #9cdcfe;--code-string: #ce9178;--code-number: #b5cea8;--code-boolean: #569cd6;--shadow-sm: 0 1px 2px rgba(0, 0, 0, .05);--shadow-md: 0 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0 8px 24px rgba(0, 0, 0, .12);--node-bg: #ffffff;--node-header: #f8f7f5;--node-header-selected: #eb5601;--node-border: #e6e4e1;--grid-line: #e6e4e1}:root[data-app=schema-browser][data-theme=dark]{--bg-primary: #1e1e1e;--bg-secondary: #252526;--bg-tertiary: #2d2d2d;--bg-hover: #37373d;--bg-canvas: #1e1e1e;--text-primary: #cccccc;--text-secondary: #8b8b8b;--text-muted: #6b6b6b;--border: #3c3c3c;--border-strong: #4a4a4a;--accent: #c9a87c;--accent-interactive: #ff6b35;--accent-hover: #ff8555;--success: #4ec9b0;--warning: #dcdcaa;--warning-bg: #3d3520;--warning-text: #dcdcaa;--error: #f14c4c;--info: #4fc1ff;--code-bg: #1e1e1e;--code-text: #d4d4d4;--code-key: #9cdcfe;--code-string: #ce9178;--code-number: #b5cea8;--code-boolean: #569cd6;--shadow-sm: 0 1px 2px rgba(0, 0, 0, .3);--shadow-md: 0 4px 12px rgba(0, 0, 0, .4);--shadow-lg: 0 8px 24px rgba(0, 0, 0, .5);--node-bg: #2d2d2d;--node-header: #37373d;--node-header-selected: #ff6b35;--node-border: #4a4a4a;--grid-line: #2d2d2d}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,sans-serif;background:var(--bg-primary);color:var(--text-primary);min-height:100vh;display:flex;flex-direction:column;margin:0;padding:0}.header{background:var(--bg-secondary);padding:10px 16px;border-bottom:1px solid var(--border);display:flex;justify-content:space-between;align-items:center}.header h1{font-size:18px;font-weight:600;display:flex;align-items:center;gap:8px;color:var(--text-primary)}.header-info{flex:1;margin:0 16px}.deployment-url{font-size:12px;color:var(--text-secondary);font-family:var(--font-mono)}.header-actions{display:flex;gap:8px;align-items:center}.theme-toggle{width:36px;height:36px;background:var(--bg-hover);border:1px solid var(--border);border-radius:8px;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);transition:all .2s ease}.theme-toggle:hover{background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.theme-toggle svg{width:18px;height:18px}.theme-toggle .sun-icon{display:none}.theme-toggle .moon-icon,[data-theme=dark] .theme-toggle .sun-icon{display:block}[data-theme=dark] .theme-toggle .moon-icon{display:none}.btn{background:var(--bg-primary);color:var(--text-primary);border:1px solid var(--border);padding:6px 12px;border-radius:6px;cursor:pointer;font-size:13px;transition:all .2s}.btn:hover{background:var(--bg-hover);border-color:var(--accent)}.btn-primary{background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.btn-primary:hover{background:var(--accent-hover);border-color:var(--accent-hover)}.main{display:flex;flex:1;overflow:hidden}.sidebar{width:260px;background:var(--bg-secondary);border-right:1px solid var(--border);overflow-y:auto;flex-shrink:0;display:flex;flex-direction:column;margin:0;padding:0}.sidebar-header{padding:12px 16px;font-size:11px;text-transform:uppercase;color:var(--text-secondary);border-bottom:1px solid var(--border);display:flex;align-items:center;font-weight:600;letter-spacing:.5px;gap:8px;position:relative}.sidebar-header #tableCount{background:var(--bg-hover);padding:2px 8px;border-radius:10px;font-size:11px}.sidebar-header .sidebar-collapse-btn{margin-left:auto;width:24px;height:24px;background:var(--bg-hover);border:1px solid transparent;border-radius:4px;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);transition:all .15s;flex-shrink:0}.sidebar-header .sidebar-collapse-btn:hover{background:var(--accent-interactive);color:#fff}.sidebar-header .sidebar-collapse-btn svg{transition:transform .2s}.sidebar-search{padding:8px 12px;border-bottom:1px solid var(--border)}.sidebar-search input{width:100%;background:var(--bg-primary);border:1px solid var(--border);border-radius:6px;padding:8px 10px;color:var(--text-primary);font-size:13px}.sidebar-search input:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.table-list{list-style:none;flex:1;overflow-y:auto}.table-item{padding:10px 16px;cursor:pointer;display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid var(--border);transition:background .2s}.table-item:hover{background:var(--bg-hover)}.table-item.active{background:var(--accent-interactive);color:#fff}.table-item.active .table-count{background:#fff3;color:#fff}.table-item.active .table-icon{opacity:1}.table-name{font-size:14px;font-weight:500;display:flex;align-items:center;gap:6px}.table-icon{opacity:.5}.table-count{font-size:12px;color:var(--text-secondary);background:var(--bg-hover);padding:2px 8px;border-radius:10px}.content{flex:1;display:flex;flex-direction:column;overflow:hidden;background:var(--bg-primary)}.schema-panel{padding:20px;flex:1;overflow-y:auto}.schema-title{font-size:22px;margin-bottom:8px;display:flex;align-items:center;gap:12px;color:var(--text-primary)}.schema-subtitle{color:var(--text-secondary);font-size:13px;margin-bottom:20px}.schema-grid{display:grid;grid-template-columns:1fr 1fr;gap:20px;margin-bottom:20px}@media(max-width:900px){.schema-grid{grid-template-columns:1fr}}.schema-card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:8px;overflow:hidden}.schema-card-header{background:var(--bg-hover);padding:10px 16px;font-weight:600;font-size:14px;border-bottom:1px solid var(--border);display:flex;justify-content:space-between;align-items:center;color:var(--text-primary)}.schema-card-header .count{color:var(--text-secondary);font-weight:400;font-size:12px}.schema-card-body{padding:8px;max-height:400px;overflow-y:auto}.field-row{display:flex;justify-content:space-between;padding:8px 10px;border-radius:4px;font-size:13px;font-family:var(--font-mono)}.field-row:hover{background:var(--bg-hover)}.field-name{color:var(--accent-interactive)}.field-type{color:var(--text-secondary)}.field-optional{color:var(--warning);font-size:11px;margin-left:2px}.field-system{opacity:.6}.indexes-section{margin-top:20px}.indexes-title{font-size:14px;font-weight:600;margin-bottom:12px;color:var(--text-secondary)}.index-item{background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;padding:12px 16px;margin-bottom:8px}.index-name{font-family:var(--font-mono);font-size:13px;color:var(--accent-interactive);margin-bottom:4px}.index-fields{font-size:12px;color:var(--text-secondary)}.documents-panel{background:var(--bg-secondary);border-top:1px solid var(--border);max-height:320px;overflow:hidden;display:flex;flex-direction:column}.documents-header{padding:10px 16px;display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid var(--border);background:var(--bg-hover);flex-shrink:0}.documents-title{font-weight:600;font-size:14px;color:var(--text-primary)}.pagination{display:flex;align-items:center;gap:8px}.pagination button{background:var(--bg-primary);color:var(--text-primary);border:1px solid var(--border);width:28px;height:28px;border-radius:4px;cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:12px}.pagination button:hover:not(:disabled){background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.pagination button:disabled{opacity:.5;cursor:not-allowed}.pagination span{font-size:13px;color:var(--text-secondary)}.documents-table{overflow:auto;flex:1}table{width:100%;border-collapse:collapse;font-size:13px}th,td{padding:10px 14px;text-align:left;border-bottom:1px solid var(--border);white-space:nowrap}th{background:var(--bg-hover);font-weight:600;position:sticky;top:0;z-index:1;color:var(--text-primary)}td{font-family:var(--font-mono);font-size:12px}tr:hover td{background:var(--bg-hover)}td.id-cell{color:var(--accent-interactive);cursor:pointer}td.id-cell:hover{text-decoration:underline}td.null-value{color:var(--text-secondary);font-style:italic}td.truncated{max-width:200px;overflow:hidden;text-overflow:ellipsis}.empty-state h2{margin-bottom:8px;color:var(--text-primary);font-size:18px}.empty-state p{font-size:14px}.warning-badge{background:#c4842d1a;border:1px solid var(--warning);color:var(--warning);padding:10px 14px;border-radius:6px;font-size:13px;margin-top:16px;display:flex;align-items:center;gap:8px}.warning-badge:before{content:"⚠"}.loading{display:flex;align-items:center;justify-content:center;padding:40px;gap:12px;color:var(--text-secondary)}.spinner{width:24px;height:24px;border:2px solid var(--border);border-top-color:var(--accent-interactive);border-radius:50%;animation:spin .8s linear infinite}.status-dot{width:8px;height:8px;border-radius:50%;background:var(--success)}.status-dot.error{background:var(--accent-interactive)}.status-dot.warning{background:var(--warning)}.modal-overlay{position:fixed;top:0;left:0;right:0;bottom:0;background:#1a1a1a80;display:flex;align-items:center;justify-content:center;z-index:100}.modal{background:var(--bg-primary);border:1px solid var(--border);border-radius:12px;width:90%;max-width:600px;max-height:80vh;overflow:hidden;display:flex;flex-direction:column;box-shadow:0 20px 40px #00000026}.modal-header{padding:16px 20px;border-bottom:1px solid var(--border);display:flex;justify-content:space-between;align-items:center;background:var(--bg-secondary)}.modal-header h2{font-size:16px;color:var(--text-primary)}.modal-close{background:none;border:none;color:var(--text-secondary);font-size:20px;cursor:pointer;padding:4px}.modal-close:hover{color:var(--text-primary)}.modal-body{padding:20px;flex:1;overflow-y:auto}.query-editor{width:100%;min-height:150px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;padding:12px;color:var(--text-primary);font-family:var(--font-mono);font-size:13px;resize:vertical}.query-editor:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.modal-footer{padding:16px 20px;border-top:1px solid var(--border);display:flex;justify-content:flex-end;gap:8px;background:var(--bg-secondary)}kbd{background:var(--bg-secondary);border:1px solid var(--border);border-radius:3px;padding:2px 6px;font-size:11px;font-family:var(--font-mono)}.app-container{display:flex;flex-direction:column;height:100vh;width:100vw;overflow:hidden;margin:0;padding:0;position:absolute;top:0;left:0}.view-toggle{display:flex;gap:4px;background:var(--bg-hover);padding:4px;border-radius:8px;margin-right:12px}.view-btn{display:flex;align-items:center;justify-content:center;width:32px;height:28px;border:none;background:transparent;color:var(--text-secondary);border-radius:6px;cursor:pointer;transition:all .2s}.view-btn:hover{color:var(--text-primary);background:var(--bg-secondary)}.view-btn.active{background:var(--accent-interactive);color:#fff}.graph-view{display:flex;flex-direction:column;flex:1;overflow:hidden}.graph-view .toolbar{order:-1}.code-panel{width:360px;background:#1e1e1e;border-right:1px solid #333;display:flex;flex-direction:column;flex-shrink:0}.code-header{padding:12px 16px;background:#252526;border-bottom:1px solid #333;display:flex;justify-content:space-between;align-items:center;color:#ccc;font-size:12px}.code-filename{color:#888;font-family:var(--font-mono)}.code-content{flex:1;overflow:auto;padding:0}.code-content pre{margin:0;padding:16px;font-family:var(--font-mono);font-size:12px;line-height:1.6;color:#d4d4d4}.code-content code{font-family:inherit}.json-key{color:#9cdcfe}.json-string{color:#ce9178}.json-number{color:#b5cea8}.json-boolean,.json-null{color:#569cd6}.graph-panel{flex:1;position:relative;background:var(--bg-primary);overflow:hidden;min-width:0}#graphCanvas{width:100%;height:100%;display:block}.graph-controls{position:absolute;top:16px;right:16px;display:flex;flex-direction:column;gap:8px}.graph-btn{width:36px;height:36px;border:1px solid var(--border);background:#fff;color:var(--text-primary);border-radius:8px;cursor:pointer;font-size:18px;display:flex;align-items:center;justify-content:center;transition:all .2s;box-shadow:0 2px 8px #00000014}.graph-btn:hover{background:var(--bg-hover);border-color:var(--accent)}.graph-legend{position:absolute;bottom:16px;right:16px;background:#fff;border:1px solid var(--border);border-radius:8px;padding:12px 16px;display:flex;flex-direction:column;gap:8px;font-size:12px;box-shadow:0 2px 8px #00000014}.legend-item{display:flex;align-items:center;gap:8px;color:var(--text-secondary)}.legend-dot{width:12px;height:12px;border-radius:4px;background:var(--accent-interactive)}.legend-dot.table{background:#fff;border:2px solid var(--border)}.legend-line{width:20px;height:2px;background:var(--accent)}.list-view{display:flex;flex:1;overflow:hidden}.list-view .content{display:flex;flex-direction:column;flex:1;overflow:hidden}.list-view .content #listViewContent{display:flex;flex-direction:column;flex:1;min-height:0;overflow:hidden}.list-view .table-header{padding:16px 24px;border-bottom:1px solid var(--border);background:var(--bg-primary);flex-shrink:0}.list-view .table-header-title{font-size:24px;font-weight:600;color:var(--text-primary);margin-bottom:4px}.list-view .table-header-meta{font-size:13px;color:var(--text-secondary);display:flex;align-items:center;gap:12px}.list-view .table-header-meta .badge{background:var(--bg-hover);padding:2px 8px;border-radius:4px;font-size:12px}.list-view .content-split{display:flex;flex:1;min-height:0;overflow:hidden}.list-view .schema-sidebar{width:280px;min-width:240px;max-width:360px;background:var(--bg-secondary);border-right:1px solid var(--border);display:flex;flex-direction:column;overflow:hidden;flex-shrink:0}.list-view .schema-sidebar-header{padding:12px 16px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:.5px;color:var(--text-secondary);border-bottom:1px solid var(--border);background:var(--bg-hover);display:flex;justify-content:space-between;align-items:center}.list-view .schema-sidebar-header .field-count-badge{font-size:11px;color:var(--text-secondary);background:var(--bg-primary);padding:2px 8px;border-radius:10px}.list-view .schema-fields-list{flex:1;overflow-y:auto;padding:8px 0}.list-view .schema-field-item{display:flex;justify-content:space-between;align-items:center;padding:10px 16px;border-bottom:1px solid var(--border);transition:background .15s}.list-view .schema-field-item:hover{background:var(--bg-hover)}.list-view .schema-field-item:last-child{border-bottom:none}.list-view .schema-field-name{font-family:var(--font-mono);font-size:13px;color:var(--accent-interactive);display:flex;align-items:center;gap:4px}.list-view .schema-field-name.system-field{color:var(--text-secondary)}.list-view .schema-field-optional{font-size:10px;color:var(--warning);font-weight:600}.list-view .schema-field-type{font-family:var(--font-mono);font-size:12px;color:var(--text-secondary);background:var(--bg-primary);padding:2px 8px;border-radius:4px}.list-view .schema-indexes{border-top:1px solid var(--border);padding:12px 16px;background:var(--bg-hover)}.list-view .schema-indexes-title{font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:.5px;color:var(--text-secondary);margin-bottom:8px}.list-view .schema-index-item{font-family:var(--font-mono);font-size:12px;color:var(--accent);padding:4px 0}.list-view .documents-main{flex:1;display:flex;flex-direction:column;overflow:hidden;background:var(--bg-primary);min-width:0;min-height:0}.list-view .documents-toolbar{padding:12px 20px;display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid var(--border);background:var(--bg-secondary);flex-shrink:0}.list-view .documents-toolbar-title{font-size:13px;font-weight:600;color:var(--text-primary);display:flex;align-items:center;gap:8px}.list-view .documents-toolbar-title .doc-count{font-weight:400;color:var(--text-secondary)}.list-view .documents-table-wrapper{flex:1;min-height:0;overflow:auto;padding:0}.list-view .documents-table-wrapper table{width:100%;border-collapse:collapse;font-size:13px}.list-view .documents-table-wrapper th{position:sticky;top:0;background:var(--bg-hover);font-weight:600;text-align:left;padding:12px 16px;border-bottom:2px solid var(--border);white-space:nowrap;z-index:1}.list-view .documents-table-wrapper td{padding:10px 16px;border-bottom:1px solid var(--border);font-family:var(--font-mono);font-size:12px;vertical-align:top;max-width:300px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.list-view .documents-table-wrapper tr:hover td{background:var(--bg-secondary)}.list-view .documents-table-wrapper td.id-cell{color:var(--accent-interactive);cursor:pointer}.list-view .documents-table-wrapper td.id-cell:hover{text-decoration:underline}.list-view .pagination-bar{padding:12px 20px;display:flex;justify-content:space-between;align-items:center;border-top:1px solid var(--border);background:var(--bg-secondary);flex-shrink:0}.list-view .pagination-info{font-size:13px;color:var(--text-secondary)}.list-view .pagination-controls{display:flex;align-items:center;gap:8px}.list-view .pagination-controls button{width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:var(--bg-primary);border:1px solid var(--border);border-radius:6px;cursor:pointer;color:var(--text-primary);font-size:14px;transition:all .15s}.list-view .pagination-controls button:hover:not(:disabled){background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.list-view .pagination-controls button:disabled{opacity:.4;cursor:not-allowed}.list-view .pagination-controls .page-indicator{font-size:13px;color:var(--text-secondary);min-width:80px;text-align:center}.list-view .documents-empty{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:60px 20px;color:var(--text-secondary);text-align:center}.list-view .documents-empty-icon{font-size:48px;margin-bottom:16px;opacity:.5}.list-view .documents-empty h3{font-size:16px;color:var(--text-primary);margin-bottom:4px}.list-view .documents-empty p{font-size:13px}.list-view .schema-panel,.list-view .documents-panel{display:none!important}.sidebar-container{display:flex;flex-shrink:0;position:relative}.sidebar-container.collapsed .sidebar{width:48px!important;min-width:48px!important;overflow:hidden}.sidebar-container.collapsed .sidebar .sidebar-header{padding:12px;justify-content:center}.sidebar-container.collapsed .sidebar .sidebar-header>span{display:none}.sidebar-container.collapsed .sidebar .sidebar-header .sidebar-collapse-btn{margin-left:0}.sidebar-container.collapsed .sidebar .sidebar-search,.sidebar-container.collapsed .sidebar .table-list{display:none}.sidebar-container.collapsed .code-panel{width:0!important;min-width:0!important;padding:0;overflow:hidden;border:none}.sidebar-container.collapsed .resize-handle{display:none}.resize-handle{width:4px;cursor:col-resize;background:transparent;transition:background .2s;position:relative;z-index:10}.resize-handle:hover,.resize-handle.dragging{background:var(--accent-interactive)}.resize-handle:after{content:"";position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);width:4px;height:40px;border-radius:2px;background:var(--border);opacity:0;transition:opacity .2s}.resize-handle:hover:after{opacity:1;background:#fff}.sidebar-toggle{display:none}.list-view .sidebar-header{position:relative}.list-view .sidebar-header .sidebar-collapse-btn{position:absolute;right:8px;top:50%;transform:translateY(-50%);width:24px;height:24px;background:var(--bg-hover);border:1px solid var(--border);border-radius:4px;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);transition:all .2s}.list-view .sidebar-header .sidebar-collapse-btn:hover{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.list-view .sidebar-header .sidebar-collapse-btn svg{width:12px;height:12px;transition:transform .2s}.sidebar-container.collapsed .sidebar-collapse-btn svg{transform:rotate(180deg)}.list-view .sidebar{min-width:180px;max-width:400px}.list-view .sidebar-container{height:100%}.toolbar{display:flex;align-items:center;padding:8px 16px;background:var(--bg-secondary);border-bottom:1px solid var(--border);gap:4px;flex-shrink:0;width:100%;z-index:10}.toolbar-group{display:flex;align-items:center;gap:2px}.toolbar-separator{width:1px;height:24px;background:var(--border);margin:0 8px}.toolbar-spacer{flex:1}.toolbar-btn{display:flex;align-items:center;gap:6px;padding:6px 10px;background:transparent;border:1px solid transparent;border-radius:6px;color:var(--text-secondary);font-size:13px;cursor:pointer;transition:all .15s ease}.toolbar-btn:hover:not(:disabled){background:var(--bg-hover);color:var(--text-primary);border-color:var(--border)}.toolbar-btn.active{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.toolbar-btn.active:hover{background:var(--accent-hover);border-color:var(--accent-hover)}.toolbar-btn:disabled{opacity:.4;cursor:not-allowed}.toolbar-btn.icon-only{padding:6px}.toolbar-btn svg{flex-shrink:0}.toolbar-btn .dropdown-arrow{margin-left:2px;opacity:.6}.zoom-group{display:flex;align-items:center;gap:4px;background:var(--bg-primary);border:1px solid var(--border);border-radius:6px;padding:2px}.zoom-group .toolbar-btn{border-radius:4px}.zoom-display{min-width:48px;text-align:center;font-size:12px;font-weight:500;color:var(--text-secondary);font-family:var(--font-mono)}.graph-content{display:flex;flex:1;overflow:hidden;flex-direction:row}.graph-content .enhanced-sidebar{order:0}.graph-content .graph-panel{order:1;flex:1}.graph-content .sidebar-container{order:2}.sidebar-container.hidden .code-panel{width:0!important;min-width:0;padding:0;overflow:hidden;border:none}.sidebar-container.hidden .resize-handle{display:none}.sidebar-container.hidden .sidebar-toggle{left:0;border-radius:0 6px 6px 0}.zoom-panel{position:absolute;bottom:20px;left:20px;display:flex;flex-direction:column;gap:4px;background:#fff;border:1px solid var(--border);border-radius:10px;padding:8px;box-shadow:0 4px 12px #0000001a;z-index:10}.zoom-btn{width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:transparent;border:none;border-radius:6px;color:var(--text-secondary);cursor:pointer;transition:all .15s ease}.zoom-btn:hover{background:var(--bg-hover);color:var(--text-primary)}.zoom-btn:active{background:var(--accent-interactive);color:#fff}.zoom-level{text-align:center;font-size:11px;font-weight:500;color:var(--text-secondary);font-family:var(--font-mono);padding:4px 0}.zoom-separator{height:1px;background:var(--border);margin:4px 0}.dropdown-menu{position:absolute;background:#fff;border:1px solid var(--border);border-radius:10px;box-shadow:0 8px 24px #0000001f;z-index:100;overflow:hidden;min-width:180px}.export-menu{top:52px;left:140px}.dropdown-item{display:flex;align-items:center;gap:10px;width:100%;padding:10px 14px;background:transparent;border:none;color:var(--text-primary);font-size:13px;cursor:pointer;transition:background .15s ease;text-align:left}.dropdown-item:hover{background:var(--bg-hover)}.dropdown-item svg{color:var(--text-secondary)}.filter-menu{position:absolute;top:52px;right:200px;width:280px}.filter-header{display:flex;justify-content:space-between;align-items:center;padding:12px 16px;border-bottom:1px solid var(--border);font-weight:600;font-size:14px}.filter-close{background:none;border:none;font-size:18px;color:var(--text-secondary);cursor:pointer;padding:0;line-height:1}.filter-close:hover{color:var(--text-primary)}.filter-body{padding:16px}.filter-group{margin-bottom:14px}.filter-group:last-child{margin-bottom:0}.filter-group label{display:block;font-size:12px;font-weight:500;color:var(--text-secondary);margin-bottom:6px}.filter-group input[type=text],.filter-group select{width:100%;padding:8px 10px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;font-size:13px;color:var(--text-primary)}.filter-group input[type=text]:focus,.filter-group select:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.filter-group.checkbox label{display:flex;align-items:center;gap:8px;cursor:pointer}.filter-group.checkbox input[type=checkbox]{width:16px;height:16px;accent-color:var(--accent-interactive)}.filter-footer{display:flex;justify-content:flex-end;gap:8px;padding:12px 16px;border-top:1px solid var(--border);background:var(--bg-secondary)}.filter-footer .btn{padding:6px 14px;font-size:13px}.btn-secondary{background:var(--bg-primary);border:1px solid var(--border);color:var(--text-primary)}.btn-secondary:hover{background:var(--bg-hover)}.enhanced-sidebar{width:260px;background:var(--bg-secondary);border-right:1px solid var(--border);display:flex;flex-direction:column;flex-shrink:0;overflow:hidden;height:100%;margin:0;padding:0;position:relative;transition:width .2s ease,min-width .2s ease}.graph-sidebar-toggle{position:absolute;top:50%;right:-14px;transform:translateY(-50%);width:28px;height:48px;background:var(--bg-secondary);border:1px solid var(--border);border-left:none;border-radius:0 8px 8px 0;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);z-index:20;transition:all .15s ease;box-shadow:2px 0 8px #0000000d}.graph-sidebar-toggle:hover{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.graph-sidebar-toggle svg{transition:transform .2s ease}.enhanced-sidebar.collapsed{width:0!important;min-width:0!important;border-right:none;overflow:visible}.enhanced-sidebar.collapsed .sidebar-deployment,.enhanced-sidebar.collapsed .sidebar-section{display:none}.enhanced-sidebar.collapsed .graph-sidebar-toggle{right:-28px;border-left:1px solid var(--border);border-radius:0 8px 8px 0}.sidebar-deployment{padding:14px 16px;border-bottom:1px solid var(--border);background:linear-gradient(to bottom,var(--bg-hover),var(--bg-secondary))}.deployment-status{display:flex;align-items:center;gap:8px;margin-bottom:4px}.status-indicator{width:8px;height:8px;border-radius:50%;background:var(--success)}.status-indicator.error{background:var(--accent-interactive)}.deployment-label{font-size:13px;font-weight:600;color:var(--text-primary)}.deployment-url-small{font-size:11px;font-family:var(--font-mono);color:var(--text-secondary);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.sidebar-section{border-bottom:1px solid var(--border)}.section-header{display:flex;align-items:center;gap:8px;padding:12px 14px;cursor:pointer;-webkit-user-select:none;user-select:none;transition:background .15s ease}.section-header:hover{background:var(--bg-hover)}.section-chevron{transition:transform .2s ease;color:var(--text-secondary)}.sidebar-section:not(.collapsed) .section-chevron{transform:rotate(90deg)}.section-title{font-size:11px;font-weight:600;letter-spacing:.5px;color:var(--text-secondary);text-transform:uppercase}.section-count{margin-left:auto;font-size:11px;color:var(--text-secondary);background:var(--bg-hover);padding:2px 8px;border-radius:10px}.section-content{max-height:500px;overflow:hidden;transition:max-height .3s ease,opacity .2s ease}.sidebar-section.collapsed .section-content{max-height:0;opacity:0}.sidebar-toolbar{display:flex;gap:6px;padding:8px 12px;border-bottom:1px solid var(--border);background:var(--bg-primary)}.sidebar-filter{flex:1;padding:6px 10px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;font-size:12px;color:var(--text-primary)}.sidebar-filter:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.sort-btn{width:28px;height:28px;display:flex;align-items:center;justify-content:center;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;color:var(--text-secondary);cursor:pointer;transition:all .15s ease}.sort-btn:hover{background:var(--bg-hover);border-color:var(--accent);color:var(--text-primary)}.sidebar-table-list{list-style:none;max-height:400px;overflow-y:auto;padding:4px 0}.sidebar-table-item{display:flex;align-items:center;justify-content:space-between;padding:8px 14px;cursor:pointer;transition:background .15s ease}.sidebar-table-item:hover{background:var(--bg-hover)}.sidebar-table-item.active{background:var(--accent-interactive)}.sidebar-table-item.active .table-name,.sidebar-table-item.active .table-icon{color:#fff}.sidebar-table-item.active .table-item-meta span{background:#fff3;color:#fff}.sidebar-table-item.filtered-out{opacity:.4}.table-item-main{display:flex;align-items:center;gap:8px;min-width:0}.table-icon{flex-shrink:0;color:var(--text-secondary)}.sidebar-table-item .table-name{font-size:13px;font-weight:500;color:var(--text-primary);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.table-item-meta{display:flex;align-items:center;gap:6px}.field-count,.doc-count{font-size:10px;padding:2px 6px;border-radius:8px;background:var(--bg-hover);color:var(--text-secondary)}.field-count{background:#eb56011a;color:var(--accent-interactive)}.convex-info-list{padding:8px 14px}.convex-info-item{display:flex;justify-content:space-between;padding:8px 0;border-bottom:1px solid var(--border)}.convex-info-item:last-child{border-bottom:none}.info-label{font-size:12px;color:var(--text-secondary)}.info-value{font-size:12px;font-weight:600;color:var(--text-primary)}.legend-line.dashed{background:repeating-linear-gradient(to right,var(--accent) 0px,var(--accent) 4px,transparent 4px,transparent 8px)}.graph-view .sidebar-container{display:flex;flex-direction:row;flex-shrink:0;position:relative;border-left:1px solid var(--border);height:100%}.graph-view .sidebar-container .code-panel{border-right:none;border-left:none;min-width:200px;max-width:600px}.resize-handle-left{cursor:col-resize;width:4px;background:transparent;transition:background .2s}.resize-handle-left:hover,.resize-handle-left.dragging{background:var(--accent-interactive)}.sidebar-toggle-right{position:absolute;top:12px;left:-32px;right:auto}.sidebar-toggle-right svg{transform:rotate(180deg)}.sidebar-container.collapsed .sidebar-toggle-right{left:-24px}.sidebar-container.collapsed .sidebar-toggle-right svg{transform:rotate(0)}.sidebar-container.hidden{display:none}.tooltip{position:fixed;z-index:1000;max-width:280px;padding:10px 14px;background:#fff;border:1px solid var(--border);border-radius:8px;box-shadow:0 8px 24px #0000001f;font-size:12px;line-height:1.5;animation:tooltipFadeIn .2s ease;pointer-events:none}@keyframes tooltipFadeIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}.tooltip-title{font-weight:600;color:var(--text-primary);margin-bottom:4px}.tooltip-content{color:var(--text-secondary)}.tooltip-content code{background:var(--bg-secondary);padding:1px 5px;border-radius:4px;font-family:var(--font-mono);font-size:11px;color:var(--accent-interactive)}.tooltip-content strong{color:var(--text-primary)}.tooltip-content em{color:var(--warning);font-style:normal}.tooltip-system{border-left:3px solid var(--text-secondary)}.tooltip-field{border-left:3px solid var(--accent-interactive)}.tooltip-relationship{border-left:3px solid var(--accent)}.loading-overlay{position:absolute;top:0;left:0;right:0;bottom:0;background:#faf8f5e6;display:flex;align-items:center;justify-content:center;z-index:50}.loading-content{text-align:center}.loading-spinner{width:40px;height:40px;border:3px solid var(--border);border-top-color:var(--accent-interactive);border-radius:50%;animation:spin .8s linear infinite;margin:0 auto 16px}.loading-text{color:var(--text-secondary);font-size:14px}.skeleton-sidebar{padding:16px}.skeleton-item{display:flex;align-items:center;gap:10px;padding:10px 0;border-bottom:1px solid var(--border)}.skeleton-icon{width:20px;height:20px;border-radius:4px;background:linear-gradient(90deg,#f0efed 25%,#e6e4e1,#f0efed 75%);background-size:200px 100%;animation:shimmer 1.5s infinite}.skeleton-text{flex:1;height:14px;border-radius:4px;background:linear-gradient(90deg,#f0efed 25%,#e6e4e1,#f0efed 75%);background-size:200px 100%;animation:shimmer 1.5s infinite}.skeleton-badge{width:32px;height:18px;border-radius:9px;background:linear-gradient(90deg,#f0efed 25%,#e6e4e1,#f0efed 75%);background-size:200px 100%;animation:shimmer 1.5s infinite}@keyframes shimmer{0%{background-position:-200px 0}to{background-position:200px 0}}.empty-state-enhanced{text-align:center;padding:60px 30px;color:var(--text-secondary)}.empty-icon{font-size:48px;margin-bottom:16px;opacity:.6}.empty-title{font-size:18px;font-weight:600;color:var(--text-primary);margin-bottom:8px}.empty-description{font-size:14px;margin-bottom:20px;max-width:320px;margin-left:auto;margin-right:auto}.empty-action .btn{display:inline-flex;align-items:center;gap:8px}.table-card,.sidebar-table-item{transition:transform .15s ease,box-shadow .15s ease}.graph-view,.list-view{animation:viewFadeIn .3s ease}@keyframes viewFadeIn{0%{opacity:0}to{opacity:1}}.toolbar-btn:active,.zoom-btn:active,.graph-btn:active{transform:scale(.95)}.toolbar-btn:focus-visible,.zoom-btn:focus-visible,.btn:focus-visible{outline:2px solid var(--accent-interactive);outline-offset:2px}@media(min-width:1600px){.enhanced-sidebar{width:300px}.graph-view .sidebar-container .code-panel{width:420px!important}.sidebar-table-list{max-height:600px}}@media(min-width:1920px){.enhanced-sidebar{width:320px}.graph-view .sidebar-container .code-panel{width:480px!important}}@media(max-width:1200px){.enhanced-sidebar{width:220px}.code-panel{width:300px!important}}@media(max-width:900px){.toolbar-btn span{display:none}.toolbar-btn,.toolbar-btn.icon-only{padding:6px}.zoom-display{min-width:40px;font-size:11px}.enhanced-sidebar{width:180px}.sidebar-deployment{padding:10px 12px}.deployment-label{font-size:12px}}@media print{.toolbar,.enhanced-sidebar,.sidebar-container,.zoom-panel,.graph-legend{display:none!important}.graph-panel{width:100%!important}}.shortcuts-modal{position:fixed;top:0;left:0;right:0;bottom:0;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:1000}.shortcuts-modal-content{background:var(--bg-primary);border-radius:12px;box-shadow:var(--shadow-lg);max-width:480px;width:90%;max-height:80vh;overflow:hidden}.shortcuts-modal-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid var(--border)}.shortcuts-modal-header h3{margin:0;font-size:16px;font-weight:600;color:var(--text-primary)}.shortcuts-close{background:none;border:none;font-size:24px;cursor:pointer;color:var(--text-secondary);padding:0;line-height:1}.shortcuts-close:hover{color:var(--text-primary)}.shortcuts-modal-body{padding:20px;overflow-y:auto;max-height:calc(80vh - 60px)}.shortcuts-section{margin-bottom:20px}.shortcuts-section:last-child{margin-bottom:0}.shortcuts-section h4{font-size:12px;font-weight:600;color:var(--text-secondary);text-transform:uppercase;letter-spacing:.5px;margin:0 0 12px}.shortcut-item{display:flex;align-items:center;gap:8px;padding:8px 0;font-size:13px;color:var(--text-primary)}.shortcut-item kbd{display:inline-flex;align-items:center;justify-content:center;min-width:24px;height:24px;padding:0 8px;background:var(--bg-tertiary);border:1px solid var(--border);border-radius:4px;font-family:var(--font-mono);font-size:11px;color:var(--text-primary)}*{box-sizing:border-box;margin:0;padding:0}html{width:100%;height:100%}body{width:100%;min-height:100%;overflow-x:hidden;overflow-y:auto}:root[data-app=realtime-dashboard]{--bg-primary: #faf8f5;--bg-secondary: #f5f3f0;--bg-hover: #ebe9e6;--text-primary: #1a1a1a;--text-secondary: #6b6b6b;--border: #e6e4e1;--accent: #8b7355;--accent-interactive: #eb5601;--accent-hover: #d14a01;--success: #4a8c5c;--warning: #c4842d;--info: #4a7c9b;--font-mono: "SF Mono", Monaco, "Cascadia Code", "Courier New", monospace;--chart-gradient-start: #eb5601;--chart-gradient-end: #d14a01;--chart-area-opacity: .1}:root[data-app=realtime-dashboard][data-theme=dark]{--bg-primary: #1e1e1e;--bg-secondary: #252526;--bg-hover: #37373d;--text-primary: #cccccc;--text-secondary: #8b8b8b;--border: #3c3c3c;--accent: #c9a87c;--accent-interactive: #ff6b35;--accent-hover: #ff8555;--success: #4ec9b0;--warning: #dcdcaa;--info: #4fc1ff;--chart-gradient-start: #ff6b35;--chart-gradient-end: #ff8555;--chart-area-opacity: .15}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,sans-serif;background:var(--bg-primary);color:var(--text-primary);min-height:100vh;padding:0;margin:0}#app{width:100%;max-width:none;margin:0;padding:16px 24px 40px}.header{display:flex;justify-content:space-between;align-items:center;margin-bottom:20px}.header h1{font-size:20px;display:flex;align-items:center;gap:10px;color:var(--text-primary)}.header-right{display:flex;align-items:center;gap:16px}.deployment-url{font-size:12px;color:var(--text-secondary);font-family:var(--font-mono);margin-right:16px}.last-update{color:var(--text-secondary);font-size:12px}.theme-toggle-btn{background:var(--bg-hover);border:1px solid var(--border);color:var(--text-primary);width:32px;height:32px;border-radius:6px;cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:16px;transition:all .2s ease}.theme-toggle-btn:hover{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.chart-subtitle{font-size:12px;color:var(--text-secondary)}.pie-legend{padding:20px}.pie-item{display:flex;align-items:center;gap:8px;padding:8px 0;border-bottom:1px solid var(--border)}.pie-item:last-child{border-bottom:none}.pie-color{width:12px;height:12px;border-radius:2px}.pie-label{flex:1;font-size:13px}.pie-value{font-family:var(--font-mono);font-size:12px;color:var(--text-secondary)}.status-dot{width:8px;height:8px;border-radius:50%;background:var(--success);animation:pulse 2s infinite}.status-dot.disconnected{background:var(--accent-interactive);animation:none}@keyframes pulse{0%,to{opacity:1}50%{opacity:.5}}.metrics-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(220px,1fr));gap:16px;margin-bottom:24px}.metric-card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:12px;padding:20px;transition:transform .2s,box-shadow .2s}.metric-card:hover{transform:translateY(-2px);box-shadow:0 4px 12px #00000014}.metric-header{display:flex;justify-content:space-between;align-items:flex-start;margin-bottom:12px}.metric-label{font-size:12px;color:var(--text-secondary);text-transform:uppercase;letter-spacing:.5px;font-weight:600}.metric-icon{width:32px;height:32px;background:var(--bg-hover);border-radius:8px;display:flex;align-items:center;justify-content:center;font-size:16px;color:var(--accent)}.metric-value{font-size:36px;font-weight:700;margin-bottom:8px;font-family:var(--font-mono);color:var(--text-primary)}.metric-change{font-size:13px;display:flex;align-items:center;gap:4px}.metric-change.positive{color:var(--success)}.metric-change.negative{color:var(--accent-interactive)}.metric-change.neutral{color:var(--text-secondary)}.metric-source{font-size:11px;color:var(--text-secondary);margin-top:8px;font-family:var(--font-mono)}.charts-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(400px,1fr));gap:16px}@media(max-width:900px){.charts-grid{grid-template-columns:1fr}}.chart-card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:12px;padding:20px}.chart-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.chart-title{font-size:14px;font-weight:600;color:var(--text-primary)}.chart-actions{display:flex;gap:8px}.chart-btn{background:var(--bg-hover);border:1px solid var(--border);color:var(--text-secondary);padding:4px 8px;border-radius:4px;font-size:11px;cursor:pointer;transition:all .2s}.chart-btn:hover{color:var(--text-primary);border-color:var(--accent)}.chart-btn.active{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.chart-container{height:220px;background:var(--bg-hover);border-radius:8px;display:flex;align-items:center;justify-content:center;position:relative;overflow:hidden}.bar-chart{display:flex;align-items:flex-end;justify-content:space-around;height:100%;width:100%;padding:20px;gap:8px}.bar{flex:1;max-width:40px;background:linear-gradient(to top,var(--accent-interactive),var(--accent-hover));border-radius:4px 4px 0 0;transition:height .3s ease;position:relative;min-height:4px}.bar:after{content:attr(data-value);position:absolute;top:-20px;left:50%;transform:translate(-50%);font-size:11px;color:var(--text-secondary);font-family:var(--font-mono)}.bar-label{position:absolute;bottom:-20px;left:50%;transform:translate(-50%);font-size:10px;color:var(--text-secondary);white-space:nowrap}.tables-overview-card{grid-column:1 / -1}.tables-overview-container{max-height:400px;overflow-y:auto;overflow-x:hidden;padding:0 4px}.horizontal-bar-chart{display:flex;flex-direction:column;gap:8px;padding:12px 0}.h-bar-row{display:grid;grid-template-columns:140px 1fr 60px;align-items:center;gap:12px;padding:6px 8px;border-radius:6px;transition:background .15s ease}.h-bar-row:hover{background:var(--bg-hover)}.h-bar-label{font-size:12px;font-weight:500;color:var(--text-primary);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;font-family:var(--font-mono)}.h-bar-track{height:20px;background:var(--bg-hover);border-radius:4px;overflow:hidden;position:relative}.h-bar-fill{height:100%;background:linear-gradient(to right,var(--accent-interactive),var(--accent-hover));border-radius:4px;transition:width .4s cubic-bezier(.4,0,.2,1);min-width:2px}.h-bar-value{font-size:12px;font-weight:600;color:var(--text-secondary);font-family:var(--font-mono);text-align:right}.tables-overview-container::-webkit-scrollbar{width:6px}.tables-overview-container::-webkit-scrollbar-track{background:var(--bg-hover);border-radius:3px}.tables-overview-container::-webkit-scrollbar-thumb{background:var(--border);border-radius:3px}.tables-overview-container::-webkit-scrollbar-thumb:hover{background:var(--text-secondary)}@media(max-width:600px){.h-bar-row{grid-template-columns:100px 1fr 50px;gap:8px}.h-bar-label,.h-bar-value{font-size:11px}}.recent-activity-card{grid-column:1 / -1}.recent-activity-container{max-height:400px;overflow-y:auto;overflow-x:auto}.activity-table{width:100%;border-collapse:collapse;font-size:13px}.activity-table thead{position:sticky;top:0;z-index:1}.activity-table th{background:var(--bg-secondary);font-weight:600;text-align:left;padding:12px 16px;border-bottom:2px solid var(--border);color:var(--text-primary);font-size:11px;text-transform:uppercase;letter-spacing:.5px}.activity-table td{padding:12px 16px;border-bottom:1px solid var(--border);vertical-align:middle}.activity-table tbody tr{transition:background .15s ease}.activity-table tbody tr:hover{background:var(--bg-hover)}.table-badge{display:inline-block;padding:4px 10px;background:var(--bg-hover);border-radius:4px;font-family:var(--font-mono);font-size:12px;font-weight:500;color:var(--text-primary)}.id-cell{font-family:var(--font-mono);color:var(--accent);font-size:12px}.time-cell{color:var(--text-secondary);font-size:12px;white-space:nowrap}.recent-activity-container::-webkit-scrollbar{width:6px;height:6px}.recent-activity-container::-webkit-scrollbar-track{background:var(--bg-hover);border-radius:3px}.recent-activity-container::-webkit-scrollbar-thumb{background:var(--border);border-radius:3px}.recent-activity-container::-webkit-scrollbar-thumb:hover{background:var(--text-secondary)}@media(max-width:600px){.activity-table th,.activity-table td{padding:10px 12px}.table-badge{font-size:11px;padding:3px 8px}.id-cell,.time-cell{font-size:11px}}.line-chart{width:100%;height:100%;padding:20px}.line-chart svg{width:100%;height:100%}.line-chart path{fill:none;stroke:var(--accent-interactive);stroke-width:2}.line-chart .area{fill:var(--accent-interactive);opacity:.1}.table-chart{width:100%;height:100%;overflow:auto;padding:0}.table-chart table{width:100%;border-collapse:collapse;font-size:12px}.table-chart th,.table-chart td{padding:10px 12px;text-align:left;border-bottom:1px solid var(--border)}.table-chart th{background:var(--bg-secondary);font-weight:600;position:sticky;top:0;color:var(--text-primary)}.table-chart tr:hover td{background:#eb56010d}.empty-state{text-align:center;color:var(--text-secondary);padding:40px}.empty-state h3{color:var(--text-primary);margin-bottom:8px;font-size:16px}.empty-state p{font-size:13px}.loading{display:flex;align-items:center;justify-content:center;gap:12px;color:var(--text-secondary)}.spinner{width:20px;height:20px;border:2px solid var(--border);border-top-color:var(--accent-interactive);border-radius:50%;animation:spin .8s linear infinite}@media(max-width:600px){#app{padding:12px 16px 32px}.header{flex-direction:column;align-items:flex-start;gap:12px}.metrics-grid{grid-template-columns:1fr}.metric-value{font-size:28px}}:root[data-app=kanban-board]{--bg-primary: #0d1117;--bg-secondary: #161b22;--bg-tertiary: #21262d;--text-primary: #f0f6fc;--text-secondary: #8b949e;--text-muted: #484f58;--border-color: #30363d;--accent-color: #58a6ff;--success-color: #3fb950;--warning-color: #d29922;--error-color: #f85149;--processing-color: #58a6ff}:root[data-app=kanban-board][data-theme=github-light]{--bg-primary: #ffffff;--bg-secondary: #f6f8fa;--bg-tertiary: #eaeef2;--text-primary: #1f2328;--text-secondary: #656d76;--text-muted: #8c959f;--border-color: #d0d7de;--accent-color: #0969da;--success-color: #1a7f37;--warning-color: #9a6700;--error-color: #cf222e;--processing-color: #0969da}*{margin:0;padding:0;box-sizing:border-box}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Noto Sans,Helvetica,Arial,sans-serif;background:var(--bg-primary);color:var(--text-primary);line-height:1.5;min-height:100vh}#app{padding:24px;max-width:1600px;margin:0 auto}.header{display:flex;justify-content:space-between;align-items:center;margin-bottom:24px;padding-bottom:16px;border-bottom:1px solid var(--border-color)}.header h1{font-size:24px;font-weight:600}.header-meta{display:flex;align-items:center;gap:16px}.deployment{font-size:13px;color:var(--text-secondary);padding:4px 8px;background:var(--bg-secondary);border-radius:4px}.theme-toggle{background:var(--bg-secondary);border:1px solid var(--border-color);border-radius:6px;width:36px;height:36px;padding:0;cursor:pointer;color:var(--text-secondary);display:flex;align-items:center;justify-content:center;transition:background .15s,color .15s}.theme-toggle:hover{background:var(--bg-tertiary);color:var(--text-primary)}.theme-toggle svg{display:block}.theme-toggle .sun-icon{display:block}.theme-toggle .moon-icon,[data-theme=github-light] .theme-toggle .sun-icon{display:none}[data-theme=github-light] .theme-toggle .moon-icon{display:block}.mode-tabs{display:flex;gap:8px;margin-bottom:24px}.mode-tab{padding:10px 20px;border-radius:6px;border:1px solid var(--border-color);background:var(--bg-secondary);color:var(--text-secondary);cursor:pointer;font-size:14px;font-weight:500;transition:all .15s}.mode-tab:hover{background:var(--bg-tertiary);color:var(--text-primary)}.mode-tab.active{background:var(--accent-color);color:#fff;border-color:var(--accent-color)}.section{margin-bottom:32px}.section-header{margin-bottom:16px}.section-title{font-size:18px;font-weight:600;margin-bottom:4px}.section-subtitle{font-size:13px;color:var(--text-secondary)}.kanban-container{display:flex;gap:16px;overflow-x:auto;padding:4px 0 16px;min-height:400px}.kanban-column{flex:0 0 300px;background:var(--bg-secondary);border-radius:8px;border:1px solid var(--border-color);display:flex;flex-direction:column;max-height:calc(100vh - 240px)}.column-header{display:flex;align-items:center;gap:10px;padding:14px 16px;border-bottom:1px solid var(--border-color)}.column-dot{width:12px;height:12px;border-radius:50%;flex-shrink:0}.column-title{font-weight:600;font-size:14px;flex:1}.column-count{background:var(--bg-tertiary);padding:2px 10px;border-radius:12px;font-size:12px;color:var(--text-secondary);font-weight:500}.column-items{flex:1;overflow-y:auto;padding:12px;display:flex;flex-direction:column;gap:10px}.kanban-card{background:var(--bg-primary);border:1px solid var(--border-color);border-radius:8px;padding:14px;cursor:pointer;transition:transform .1s,box-shadow .1s,border-color .15s}.kanban-card:hover{transform:translateY(-2px);box-shadow:0 4px 12px #00000026;border-color:var(--accent-color)}.card-title{font-weight:500;font-size:14px;margin-bottom:6px;word-break:break-word;color:var(--text-primary)}.card-meta{font-size:12px;color:var(--text-secondary);margin-bottom:8px}.card-footer{display:flex;align-items:center;justify-content:space-between;margin-top:8px}.card-badge{display:inline-block;padding:3px 8px;border-radius:4px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:.3px}.badge-cron{background:#8b5cf633;color:#a78bfa}.badge-scheduled{background:#3b82f633;color:#60a5fa}.badge-agent{background:#22c55e33;color:#4ade80}.badge-error{background:#f8514933;color:#f87171}.status-pending{color:var(--warning-color)}.status-running{color:var(--processing-color)}.status-completed{color:var(--success-color)}.status-failed{color:var(--error-color)}.status-idle{color:var(--text-secondary)}.status-processing{color:var(--processing-color)}.status-waiting{color:var(--warning-color)}.empty-state{text-align:center;padding:60px 20px;color:var(--text-secondary)}.empty-state p{margin-bottom:12px}.empty-state a{color:var(--accent-color);text-decoration:none}.empty-state a:hover{text-decoration:underline}.loading{display:flex;align-items:center;justify-content:center;padding:60px;color:var(--text-secondary)}.loading-spinner{width:24px;height:24px;border:2px solid var(--border-color);border-top-color:var(--accent-color);border-radius:50%;animation:spin .8s linear infinite;margin-right:12px}@keyframes spin{to{transform:rotate(360deg)}}.column-items::-webkit-scrollbar{width:6px}.column-items::-webkit-scrollbar-track{background:transparent}.column-items::-webkit-scrollbar-thumb{background:var(--border-color);border-radius:3px}.column-items::-webkit-scrollbar-thumb:hover{background:var(--text-muted)}@media(max-width:768px){#app{padding:16px}.kanban-column{flex:0 0 280px}.header{flex-direction:column;gap:12px;align-items:flex-start}}
|
|
@@ -3,15 +3,15 @@
|
|
|
3
3
|
*
|
|
4
4
|
* Serves the bundled Realtime Dashboard HTML UI.
|
|
5
5
|
*/
|
|
6
|
-
import { readFileSync, existsSync } from
|
|
7
|
-
import { join, dirname } from
|
|
8
|
-
import { fileURLToPath } from
|
|
6
|
+
import { readFileSync, existsSync } from "fs";
|
|
7
|
+
import { join, dirname } from "path";
|
|
8
|
+
import { fileURLToPath } from "url";
|
|
9
9
|
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
10
10
|
export async function getDashboardResourceContent() {
|
|
11
11
|
// Try to load the bundled HTML file
|
|
12
|
-
const bundledPath = join(__dirname,
|
|
12
|
+
const bundledPath = join(__dirname, "..", "apps", "realtime-dashboard.html");
|
|
13
13
|
if (existsSync(bundledPath)) {
|
|
14
|
-
return readFileSync(bundledPath,
|
|
14
|
+
return readFileSync(bundledPath, "utf-8");
|
|
15
15
|
}
|
|
16
16
|
// Fallback: return inline HTML for development/testing
|
|
17
17
|
return getInlineDashboardUI();
|
|
@@ -31,14 +31,14 @@ function getInlineDashboardUI() {
|
|
|
31
31
|
}
|
|
32
32
|
|
|
33
33
|
:root {
|
|
34
|
-
--bg-primary: #
|
|
35
|
-
--bg-secondary: #
|
|
36
|
-
--bg-tertiary: #
|
|
37
|
-
--text-primary: #
|
|
38
|
-
--text-secondary: #
|
|
39
|
-
--accent: #
|
|
40
|
-
--border: #
|
|
41
|
-
--success: #
|
|
34
|
+
--bg-primary: #faf8f5;
|
|
35
|
+
--bg-secondary: #f5f3f0;
|
|
36
|
+
--bg-tertiary: #ebe9e6;
|
|
37
|
+
--text-primary: #1a1a1a;
|
|
38
|
+
--text-secondary: #6b6b6b;
|
|
39
|
+
--accent: #eb5601;
|
|
40
|
+
--border: #e6e4e1;
|
|
41
|
+
--success: #4a8c5c;
|
|
42
42
|
}
|
|
43
43
|
|
|
44
44
|
body {
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"kanban-board.d.ts","sourceRoot":"","sources":["../../src/resources/kanban-board.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAQH,wBAAsB,wBAAwB,IAAI,OAAO,CAAC,MAAM,CAAC,CAUhE"}
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Kanban Board Resource Handler
|
|
3
|
+
*
|
|
4
|
+
* Serves the bundled Kanban Board HTML UI.
|
|
5
|
+
*/
|
|
6
|
+
import { readFileSync, existsSync } from "fs";
|
|
7
|
+
import { join, dirname } from "path";
|
|
8
|
+
import { fileURLToPath } from "url";
|
|
9
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
10
|
+
export async function getKanbanResourceContent() {
|
|
11
|
+
// Try to load the bundled HTML file
|
|
12
|
+
const bundledPath = join(__dirname, "..", "apps", "kanban-board.html");
|
|
13
|
+
if (existsSync(bundledPath)) {
|
|
14
|
+
return readFileSync(bundledPath, "utf-8");
|
|
15
|
+
}
|
|
16
|
+
// Fallback: return inline HTML for development/testing
|
|
17
|
+
return getInlineKanbanUI();
|
|
18
|
+
}
|
|
19
|
+
function getInlineKanbanUI() {
|
|
20
|
+
return `<!DOCTYPE html>
|
|
21
|
+
<html lang="en">
|
|
22
|
+
<head>
|
|
23
|
+
<meta charset="UTF-8">
|
|
24
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
25
|
+
<title>Kanban Board - Convex MCP Apps</title>
|
|
26
|
+
<style>
|
|
27
|
+
* {
|
|
28
|
+
box-sizing: border-box;
|
|
29
|
+
margin: 0;
|
|
30
|
+
padding: 0;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
:root {
|
|
34
|
+
--bg-primary: #faf8f5;
|
|
35
|
+
--bg-secondary: #f5f3f0;
|
|
36
|
+
--bg-tertiary: #ebe9e6;
|
|
37
|
+
--text-primary: #1a1a1a;
|
|
38
|
+
--text-secondary: #6b6b6b;
|
|
39
|
+
--accent: #eb5601;
|
|
40
|
+
--border: #e6e4e1;
|
|
41
|
+
--success: #4a8c5c;
|
|
42
|
+
--warning: #c4842d;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
body {
|
|
46
|
+
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
|
47
|
+
background: var(--bg-primary);
|
|
48
|
+
color: var(--text-primary);
|
|
49
|
+
min-height: 100vh;
|
|
50
|
+
padding: 20px;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
.header {
|
|
54
|
+
display: flex;
|
|
55
|
+
justify-content: space-between;
|
|
56
|
+
align-items: center;
|
|
57
|
+
margin-bottom: 24px;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
.header h1 {
|
|
61
|
+
font-size: 20px;
|
|
62
|
+
display: flex;
|
|
63
|
+
align-items: center;
|
|
64
|
+
gap: 8px;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
.status-dot {
|
|
68
|
+
width: 8px;
|
|
69
|
+
height: 8px;
|
|
70
|
+
border-radius: 50%;
|
|
71
|
+
background: var(--success);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
.board {
|
|
75
|
+
display: grid;
|
|
76
|
+
grid-template-columns: repeat(4, 1fr);
|
|
77
|
+
gap: 16px;
|
|
78
|
+
min-height: 400px;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
.column {
|
|
82
|
+
background: var(--bg-secondary);
|
|
83
|
+
border: 1px solid var(--border);
|
|
84
|
+
border-radius: 8px;
|
|
85
|
+
padding: 12px;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
.column-header {
|
|
89
|
+
font-weight: 600;
|
|
90
|
+
font-size: 14px;
|
|
91
|
+
margin-bottom: 12px;
|
|
92
|
+
padding-bottom: 8px;
|
|
93
|
+
border-bottom: 1px solid var(--border);
|
|
94
|
+
display: flex;
|
|
95
|
+
justify-content: space-between;
|
|
96
|
+
align-items: center;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
.column-count {
|
|
100
|
+
background: var(--bg-tertiary);
|
|
101
|
+
padding: 2px 8px;
|
|
102
|
+
border-radius: 10px;
|
|
103
|
+
font-size: 12px;
|
|
104
|
+
color: var(--text-secondary);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
.card {
|
|
108
|
+
background: var(--bg-primary);
|
|
109
|
+
border: 1px solid var(--border);
|
|
110
|
+
border-radius: 6px;
|
|
111
|
+
padding: 12px;
|
|
112
|
+
margin-bottom: 8px;
|
|
113
|
+
font-size: 13px;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
.card-title {
|
|
117
|
+
font-weight: 500;
|
|
118
|
+
margin-bottom: 4px;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
.card-meta {
|
|
122
|
+
font-size: 11px;
|
|
123
|
+
color: var(--text-secondary);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
.empty-column {
|
|
127
|
+
color: var(--text-secondary);
|
|
128
|
+
text-align: center;
|
|
129
|
+
padding: 20px;
|
|
130
|
+
font-size: 13px;
|
|
131
|
+
}
|
|
132
|
+
</style>
|
|
133
|
+
</head>
|
|
134
|
+
<body>
|
|
135
|
+
<div class="header">
|
|
136
|
+
<h1>
|
|
137
|
+
<span class="status-dot"></span>
|
|
138
|
+
Kanban Board
|
|
139
|
+
</h1>
|
|
140
|
+
<span style="color: var(--text-secondary); font-size: 12px;">Scheduled Functions</span>
|
|
141
|
+
</div>
|
|
142
|
+
|
|
143
|
+
<div class="board" id="board">
|
|
144
|
+
<div class="column">
|
|
145
|
+
<div class="column-header">
|
|
146
|
+
<span>Pending</span>
|
|
147
|
+
<span class="column-count">0</span>
|
|
148
|
+
</div>
|
|
149
|
+
<div class="empty-column">No pending jobs</div>
|
|
150
|
+
</div>
|
|
151
|
+
<div class="column">
|
|
152
|
+
<div class="column-header">
|
|
153
|
+
<span>Running</span>
|
|
154
|
+
<span class="column-count">0</span>
|
|
155
|
+
</div>
|
|
156
|
+
<div class="empty-column">No running jobs</div>
|
|
157
|
+
</div>
|
|
158
|
+
<div class="column">
|
|
159
|
+
<div class="column-header">
|
|
160
|
+
<span>Completed</span>
|
|
161
|
+
<span class="column-count">0</span>
|
|
162
|
+
</div>
|
|
163
|
+
<div class="empty-column">No completed jobs</div>
|
|
164
|
+
</div>
|
|
165
|
+
<div class="column">
|
|
166
|
+
<div class="column-header">
|
|
167
|
+
<span>Failed</span>
|
|
168
|
+
<span class="column-count">0</span>
|
|
169
|
+
</div>
|
|
170
|
+
<div class="empty-column">No failed jobs</div>
|
|
171
|
+
</div>
|
|
172
|
+
</div>
|
|
173
|
+
|
|
174
|
+
<script>
|
|
175
|
+
const app = {
|
|
176
|
+
config: null,
|
|
177
|
+
|
|
178
|
+
init() {
|
|
179
|
+
const params = new URLSearchParams(window.location.search);
|
|
180
|
+
const configParam = params.get('config');
|
|
181
|
+
if (configParam) {
|
|
182
|
+
try {
|
|
183
|
+
this.config = JSON.parse(decodeURIComponent(configParam));
|
|
184
|
+
this.render();
|
|
185
|
+
} catch (e) {
|
|
186
|
+
console.error('Failed to parse config:', e);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
},
|
|
190
|
+
|
|
191
|
+
render() {
|
|
192
|
+
// Render jobs/agents based on config
|
|
193
|
+
console.log('Config:', this.config);
|
|
194
|
+
}
|
|
195
|
+
};
|
|
196
|
+
|
|
197
|
+
document.addEventListener('DOMContentLoaded', () => app.init());
|
|
198
|
+
</script>
|
|
199
|
+
</body>
|
|
200
|
+
</html>`;
|
|
201
|
+
}
|
|
202
|
+
//# sourceMappingURL=kanban-board.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"kanban-board.js","sourceRoot":"","sources":["../../src/resources/kanban-board.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,IAAI,CAAC;AAC9C,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AACrC,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AAEpC,MAAM,SAAS,GAAG,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;AAE1D,MAAM,CAAC,KAAK,UAAU,wBAAwB;IAC5C,oCAAoC;IACpC,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,mBAAmB,CAAC,CAAC;IAEvE,IAAI,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;QAC5B,OAAO,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;IAC5C,CAAC;IAED,uDAAuD;IACvD,OAAO,iBAAiB,EAAE,CAAC;AAC7B,CAAC;AAED,SAAS,iBAAiB;IACxB,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;QAoLD,CAAC;AACT,CAAC"}
|
|
@@ -31,16 +31,16 @@ function getInlineSchemaUI() {
|
|
|
31
31
|
}
|
|
32
32
|
|
|
33
33
|
:root {
|
|
34
|
-
--bg-primary: #
|
|
35
|
-
--bg-secondary: #
|
|
36
|
-
--bg-tertiary: #
|
|
37
|
-
--text-primary: #
|
|
38
|
-
--text-secondary: #
|
|
39
|
-
--accent: #
|
|
40
|
-
--accent-hover: #
|
|
41
|
-
--border: #
|
|
42
|
-
--success: #
|
|
43
|
-
--warning: #
|
|
34
|
+
--bg-primary: #faf8f5;
|
|
35
|
+
--bg-secondary: #f5f3f0;
|
|
36
|
+
--bg-tertiary: #ebe9e6;
|
|
37
|
+
--text-primary: #1a1a1a;
|
|
38
|
+
--text-secondary: #6b6b6b;
|
|
39
|
+
--accent: #eb5601;
|
|
40
|
+
--accent-hover: #d14a01;
|
|
41
|
+
--border: #e6e4e1;
|
|
42
|
+
--success: #4a8c5c;
|
|
43
|
+
--warning: #c4842d;
|
|
44
44
|
}
|
|
45
45
|
|
|
46
46
|
body {
|
package/dist/server.d.ts
CHANGED
|
@@ -3,7 +3,18 @@
|
|
|
3
3
|
*
|
|
4
4
|
* Creates and configures the MCP server with tools and resources
|
|
5
5
|
* for Convex database exploration.
|
|
6
|
+
*
|
|
7
|
+
* Supports MCP Apps (SEP-1865) for embedded UI rendering in ChatGPT,
|
|
8
|
+
* Claude web, VS Code, and other MCP Apps compatible hosts.
|
|
9
|
+
*/
|
|
10
|
+
/**
|
|
11
|
+
* Store dynamically generated HTML for MCP Apps resource serving
|
|
12
|
+
*/
|
|
13
|
+
export declare function storeDynamicHtml(resourceUri: string, html: string): void;
|
|
14
|
+
/**
|
|
15
|
+
* Get stored dynamic HTML
|
|
6
16
|
*/
|
|
17
|
+
export declare function getDynamicHtml(resourceUri: string): string | undefined;
|
|
7
18
|
export interface ConvexMcpServer {
|
|
8
19
|
startStdio: () => Promise<void>;
|
|
9
20
|
startHttp: (port: number) => Promise<void>;
|
package/dist/server.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAuDH;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI,CAExE;AAED;;GAEG;AACH,wBAAgB,cAAc,CAAC,WAAW,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAEtE;AAcD,MAAM,WAAW,eAAe;IAC9B,UAAU,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAChC,SAAS,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CAC5C;AAiBD,wBAAsB,YAAY,IAAI,OAAO,CAAC,eAAe,CAAC,CA4V7D"}
|
package/dist/server.js
CHANGED
|
@@ -3,6 +3,9 @@
|
|
|
3
3
|
*
|
|
4
4
|
* Creates and configures the MCP server with tools and resources
|
|
5
5
|
* for Convex database exploration.
|
|
6
|
+
*
|
|
7
|
+
* Supports MCP Apps (SEP-1865) for embedded UI rendering in ChatGPT,
|
|
8
|
+
* Claude web, VS Code, and other MCP Apps compatible hosts.
|
|
6
9
|
*/
|
|
7
10
|
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
8
11
|
import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js";
|
|
@@ -18,6 +21,7 @@ import { writeConflictReportTool, handleWriteConflictReport, } from "./tools/wri
|
|
|
18
21
|
import { kanbanBoardTool, handleKanbanBoard } from "./tools/kanban-board.js";
|
|
19
22
|
import { getSchemaResourceContent } from "./resources/schema-browser.js";
|
|
20
23
|
import { getDashboardResourceContent } from "./resources/dashboard.js";
|
|
24
|
+
import { getKanbanResourceContent } from "./resources/kanban-board.js";
|
|
21
25
|
import { ConvexClient } from "./convex-client.js";
|
|
22
26
|
import { readFileSync } from "fs";
|
|
23
27
|
import { join, dirname } from "path";
|
|
@@ -28,6 +32,48 @@ const __dirname = dirname(__filename);
|
|
|
28
32
|
const packageJsonPath = join(__dirname, "..", "package.json");
|
|
29
33
|
const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
|
|
30
34
|
const VERSION = packageJson.version;
|
|
35
|
+
// MCP Apps MIME type for embedded UI rendering (SEP-1865)
|
|
36
|
+
const MCP_APPS_MIME_TYPE = "text/html;profile=mcp-app";
|
|
37
|
+
// In-memory store for dynamically generated HTML (diagram, heatmap, etc.)
|
|
38
|
+
// This allows MCP Apps hosts to fetch the generated HTML as a resource
|
|
39
|
+
const dynamicHtmlStore = new Map();
|
|
40
|
+
/**
|
|
41
|
+
* Store dynamically generated HTML for MCP Apps resource serving
|
|
42
|
+
*/
|
|
43
|
+
export function storeDynamicHtml(resourceUri, html) {
|
|
44
|
+
dynamicHtmlStore.set(resourceUri, html);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Get stored dynamic HTML
|
|
48
|
+
*/
|
|
49
|
+
export function getDynamicHtml(resourceUri) {
|
|
50
|
+
return dynamicHtmlStore.get(resourceUri);
|
|
51
|
+
}
|
|
52
|
+
// Resource URIs for MCP Apps
|
|
53
|
+
const RESOURCE_URIS = {
|
|
54
|
+
schemaBrowser: "ui://convex-visual/schema-browser.html",
|
|
55
|
+
dashboard: "ui://convex-visual/dashboard.html",
|
|
56
|
+
schemaDiagram: "ui://convex-visual/schema-diagram.html",
|
|
57
|
+
subwayMap: "ui://convex-visual/subway-map.html",
|
|
58
|
+
tableHeatmap: "ui://convex-visual/table-heatmap.html",
|
|
59
|
+
schemaDrift: "ui://convex-visual/schema-drift.html",
|
|
60
|
+
writeConflicts: "ui://convex-visual/write-conflicts.html",
|
|
61
|
+
kanbanBoard: "ui://convex-visual/kanban-board.html",
|
|
62
|
+
};
|
|
63
|
+
/**
|
|
64
|
+
* Helper to add MCP Apps metadata to tool definitions
|
|
65
|
+
*/
|
|
66
|
+
function withMcpAppsMetadata(tool, resourceUri) {
|
|
67
|
+
return {
|
|
68
|
+
...tool,
|
|
69
|
+
_meta: {
|
|
70
|
+
...(tool._meta || {}),
|
|
71
|
+
ui: {
|
|
72
|
+
resourceUri,
|
|
73
|
+
},
|
|
74
|
+
},
|
|
75
|
+
};
|
|
76
|
+
}
|
|
31
77
|
export async function createServer() {
|
|
32
78
|
const convexClient = new ConvexClient();
|
|
33
79
|
const server = new Server({
|
|
@@ -39,18 +85,18 @@ export async function createServer() {
|
|
|
39
85
|
resources: {},
|
|
40
86
|
},
|
|
41
87
|
});
|
|
42
|
-
// List available tools
|
|
88
|
+
// List available tools with MCP Apps metadata
|
|
43
89
|
server.setRequestHandler(ListToolsRequestSchema, async () => {
|
|
44
90
|
return {
|
|
45
91
|
tools: [
|
|
46
|
-
schemaBrowserTool,
|
|
47
|
-
dashboardTool,
|
|
48
|
-
schemaDiagramTool,
|
|
49
|
-
codebaseSubwayMapTool,
|
|
50
|
-
tableHeatmapTool,
|
|
51
|
-
schemaDriftTool,
|
|
52
|
-
writeConflictReportTool,
|
|
53
|
-
kanbanBoardTool,
|
|
92
|
+
withMcpAppsMetadata(schemaBrowserTool, RESOURCE_URIS.schemaBrowser),
|
|
93
|
+
withMcpAppsMetadata(dashboardTool, RESOURCE_URIS.dashboard),
|
|
94
|
+
withMcpAppsMetadata(schemaDiagramTool, RESOURCE_URIS.schemaDiagram),
|
|
95
|
+
withMcpAppsMetadata(codebaseSubwayMapTool, RESOURCE_URIS.subwayMap),
|
|
96
|
+
withMcpAppsMetadata(tableHeatmapTool, RESOURCE_URIS.tableHeatmap),
|
|
97
|
+
withMcpAppsMetadata(schemaDriftTool, RESOURCE_URIS.schemaDrift),
|
|
98
|
+
withMcpAppsMetadata(writeConflictReportTool, RESOURCE_URIS.writeConflicts),
|
|
99
|
+
withMcpAppsMetadata(kanbanBoardTool, RESOURCE_URIS.kanbanBoard),
|
|
54
100
|
],
|
|
55
101
|
};
|
|
56
102
|
});
|
|
@@ -95,68 +141,213 @@ export async function createServer() {
|
|
|
95
141
|
isError: result.isError,
|
|
96
142
|
};
|
|
97
143
|
});
|
|
98
|
-
// List available resources
|
|
144
|
+
// List available resources (MCP Apps compatible with profile MIME type)
|
|
99
145
|
server.setRequestHandler(ListResourcesRequestSchema, async () => {
|
|
100
146
|
return {
|
|
101
147
|
resources: [
|
|
102
148
|
{
|
|
103
|
-
uri:
|
|
149
|
+
uri: RESOURCE_URIS.schemaBrowser,
|
|
104
150
|
name: "Schema Browser",
|
|
105
151
|
description: "Interactive UI for browsing Convex database schemas",
|
|
106
|
-
mimeType:
|
|
152
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
107
153
|
},
|
|
108
154
|
{
|
|
109
|
-
uri:
|
|
155
|
+
uri: RESOURCE_URIS.dashboard,
|
|
110
156
|
name: "Realtime Dashboard",
|
|
111
157
|
description: "Live charts and metrics for Convex data",
|
|
112
|
-
mimeType:
|
|
158
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
159
|
+
},
|
|
160
|
+
{
|
|
161
|
+
uri: RESOURCE_URIS.schemaDiagram,
|
|
162
|
+
name: "Schema Diagram",
|
|
163
|
+
description: "Mermaid ER diagram of database relationships",
|
|
164
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
165
|
+
},
|
|
166
|
+
{
|
|
167
|
+
uri: RESOURCE_URIS.subwayMap,
|
|
168
|
+
name: "Codebase Subway Map",
|
|
169
|
+
description: "Visual map of codebase file dependencies",
|
|
170
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
171
|
+
},
|
|
172
|
+
{
|
|
173
|
+
uri: RESOURCE_URIS.tableHeatmap,
|
|
174
|
+
name: "Table Heatmap",
|
|
175
|
+
description: "Heatmap showing table write activity",
|
|
176
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
177
|
+
},
|
|
178
|
+
{
|
|
179
|
+
uri: RESOURCE_URIS.schemaDrift,
|
|
180
|
+
name: "Schema Drift",
|
|
181
|
+
description: "Compare declared vs inferred schema",
|
|
182
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
183
|
+
},
|
|
184
|
+
{
|
|
185
|
+
uri: RESOURCE_URIS.writeConflicts,
|
|
186
|
+
name: "Write Conflicts Report",
|
|
187
|
+
description: "Analysis of write conflicts from logs",
|
|
188
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
189
|
+
},
|
|
190
|
+
{
|
|
191
|
+
uri: RESOURCE_URIS.kanbanBoard,
|
|
192
|
+
name: "Kanban Board",
|
|
193
|
+
description: "Kanban view of scheduled functions and agents",
|
|
194
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
113
195
|
},
|
|
114
196
|
],
|
|
115
197
|
};
|
|
116
198
|
});
|
|
117
|
-
// Handle resource reads
|
|
199
|
+
// Handle resource reads (serves HTML for MCP Apps hosts)
|
|
118
200
|
server.setRequestHandler(ReadResourceRequestSchema, async (request) => {
|
|
119
201
|
const { uri } = request.params;
|
|
120
|
-
|
|
202
|
+
// Schema Browser
|
|
203
|
+
if (uri === RESOURCE_URIS.schemaBrowser || uri.startsWith("ui://schema-browser")) {
|
|
121
204
|
return {
|
|
122
205
|
contents: [
|
|
123
206
|
{
|
|
124
207
|
uri,
|
|
125
|
-
mimeType:
|
|
208
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
126
209
|
text: await getSchemaResourceContent(),
|
|
127
210
|
},
|
|
128
211
|
],
|
|
129
212
|
};
|
|
130
213
|
}
|
|
131
|
-
|
|
214
|
+
// Dashboard
|
|
215
|
+
if (uri === RESOURCE_URIS.dashboard || uri.startsWith("ui://dashboard")) {
|
|
132
216
|
return {
|
|
133
217
|
contents: [
|
|
134
218
|
{
|
|
135
219
|
uri,
|
|
136
|
-
mimeType:
|
|
220
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
137
221
|
text: await getDashboardResourceContent(),
|
|
138
222
|
},
|
|
139
223
|
],
|
|
140
224
|
};
|
|
141
225
|
}
|
|
226
|
+
// Kanban Board
|
|
227
|
+
if (uri === RESOURCE_URIS.kanbanBoard || uri.startsWith("ui://kanban")) {
|
|
228
|
+
return {
|
|
229
|
+
contents: [
|
|
230
|
+
{
|
|
231
|
+
uri,
|
|
232
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
233
|
+
text: await getKanbanResourceContent(),
|
|
234
|
+
},
|
|
235
|
+
],
|
|
236
|
+
};
|
|
237
|
+
}
|
|
238
|
+
// Dynamic HTML resources (diagram, heatmap, drift, subway map, write conflicts)
|
|
239
|
+
// These are generated when the tool is called and stored in memory
|
|
240
|
+
const dynamicHtml = getDynamicHtml(uri);
|
|
241
|
+
if (dynamicHtml) {
|
|
242
|
+
return {
|
|
243
|
+
contents: [
|
|
244
|
+
{
|
|
245
|
+
uri,
|
|
246
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
247
|
+
text: dynamicHtml,
|
|
248
|
+
},
|
|
249
|
+
],
|
|
250
|
+
};
|
|
251
|
+
}
|
|
252
|
+
// Fallback for dynamic resources: return a placeholder that fetches data
|
|
253
|
+
if (uri === RESOURCE_URIS.schemaDiagram ||
|
|
254
|
+
uri === RESOURCE_URIS.subwayMap ||
|
|
255
|
+
uri === RESOURCE_URIS.tableHeatmap ||
|
|
256
|
+
uri === RESOURCE_URIS.schemaDrift ||
|
|
257
|
+
uri === RESOURCE_URIS.writeConflicts) {
|
|
258
|
+
return {
|
|
259
|
+
contents: [
|
|
260
|
+
{
|
|
261
|
+
uri,
|
|
262
|
+
mimeType: MCP_APPS_MIME_TYPE,
|
|
263
|
+
text: getPlaceholderHtml(uri),
|
|
264
|
+
},
|
|
265
|
+
],
|
|
266
|
+
};
|
|
267
|
+
}
|
|
142
268
|
throw new Error(`Unknown resource: ${uri}`);
|
|
143
269
|
});
|
|
270
|
+
/**
|
|
271
|
+
* Returns placeholder HTML for dynamic resources
|
|
272
|
+
* The actual content is generated when the tool is called
|
|
273
|
+
*/
|
|
274
|
+
function getPlaceholderHtml(uri) {
|
|
275
|
+
const toolName = uri.split("/").pop()?.replace(".html", "") || "tool";
|
|
276
|
+
return `<!DOCTYPE html>
|
|
277
|
+
<html lang="en">
|
|
278
|
+
<head>
|
|
279
|
+
<meta charset="UTF-8">
|
|
280
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
281
|
+
<title>${toolName} - Convex MCP Apps</title>
|
|
282
|
+
<style>
|
|
283
|
+
body {
|
|
284
|
+
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
|
285
|
+
background: #faf8f5;
|
|
286
|
+
color: #1a1a1a;
|
|
287
|
+
min-height: 100vh;
|
|
288
|
+
display: flex;
|
|
289
|
+
align-items: center;
|
|
290
|
+
justify-content: center;
|
|
291
|
+
}
|
|
292
|
+
.message {
|
|
293
|
+
text-align: center;
|
|
294
|
+
padding: 40px;
|
|
295
|
+
}
|
|
296
|
+
.message h2 {
|
|
297
|
+
margin-bottom: 8px;
|
|
298
|
+
}
|
|
299
|
+
.message p {
|
|
300
|
+
color: #6b6b6b;
|
|
301
|
+
}
|
|
302
|
+
</style>
|
|
303
|
+
</head>
|
|
304
|
+
<body>
|
|
305
|
+
<div class="message">
|
|
306
|
+
<h2>${toolName.replace(/-/g, " ").replace(/\b\w/g, c => c.toUpperCase())}</h2>
|
|
307
|
+
<p>Run the tool to generate this visualization.</p>
|
|
308
|
+
</div>
|
|
309
|
+
</body>
|
|
310
|
+
</html>`;
|
|
311
|
+
}
|
|
144
312
|
return {
|
|
145
313
|
async startStdio() {
|
|
146
314
|
const transport = new StdioServerTransport();
|
|
147
315
|
await server.connect(transport);
|
|
148
316
|
},
|
|
149
317
|
async startHttp(port) {
|
|
150
|
-
//
|
|
318
|
+
// HTTP transport for MCP with CORS support for MCP Apps
|
|
319
|
+
const corsHeaders = {
|
|
320
|
+
"Access-Control-Allow-Origin": "*",
|
|
321
|
+
"Access-Control-Allow-Methods": "GET, POST, OPTIONS",
|
|
322
|
+
"Access-Control-Allow-Headers": "Content-Type, Authorization",
|
|
323
|
+
"Access-Control-Max-Age": "86400",
|
|
324
|
+
};
|
|
151
325
|
const httpServer = createHttpServer(async (req, res) => {
|
|
326
|
+
// Handle CORS preflight requests
|
|
327
|
+
if (req.method === "OPTIONS") {
|
|
328
|
+
res.writeHead(204, corsHeaders);
|
|
329
|
+
res.end();
|
|
330
|
+
return;
|
|
331
|
+
}
|
|
332
|
+
// Add CORS headers to all responses
|
|
333
|
+
Object.entries(corsHeaders).forEach(([key, value]) => {
|
|
334
|
+
res.setHeader(key, value);
|
|
335
|
+
});
|
|
336
|
+
// Health check endpoint
|
|
337
|
+
if (req.url === "/health") {
|
|
338
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
339
|
+
res.end(JSON.stringify({ status: "ok", version: VERSION }));
|
|
340
|
+
return;
|
|
341
|
+
}
|
|
342
|
+
// MCP endpoint
|
|
152
343
|
if (req.url === "/mcp" && req.method === "POST") {
|
|
153
344
|
let body = "";
|
|
154
345
|
req.on("data", (chunk) => (body += chunk));
|
|
155
346
|
req.on("end", async () => {
|
|
156
347
|
try {
|
|
157
348
|
const request = JSON.parse(body);
|
|
158
|
-
// Process MCP request
|
|
159
|
-
// This is a simplified implementation
|
|
349
|
+
// Process MCP request
|
|
350
|
+
// This is a simplified implementation for basic MCP Apps compatibility
|
|
160
351
|
res.writeHead(200, { "Content-Type": "application/json" });
|
|
161
352
|
res.end(JSON.stringify({ jsonrpc: "2.0", id: request.id, result: {} }));
|
|
162
353
|
}
|
|
@@ -165,15 +356,38 @@ export async function createServer() {
|
|
|
165
356
|
res.end(JSON.stringify({ error: "Invalid request" }));
|
|
166
357
|
}
|
|
167
358
|
});
|
|
359
|
+
return;
|
|
168
360
|
}
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
361
|
+
// Serve UI resources directly via HTTP for MCP Apps hosts
|
|
362
|
+
if (req.url?.startsWith("/ui/")) {
|
|
363
|
+
const resourceName = req.url.replace("/ui/", "").replace(".html", "");
|
|
364
|
+
const resourceUri = `ui://convex-visual/${resourceName}.html`;
|
|
365
|
+
try {
|
|
366
|
+
let html;
|
|
367
|
+
if (resourceName === "schema-browser") {
|
|
368
|
+
html = await getSchemaResourceContent();
|
|
369
|
+
}
|
|
370
|
+
else if (resourceName === "dashboard" || resourceName === "realtime-dashboard") {
|
|
371
|
+
html = await getDashboardResourceContent();
|
|
372
|
+
}
|
|
373
|
+
else if (resourceName === "kanban-board") {
|
|
374
|
+
html = await getKanbanResourceContent();
|
|
375
|
+
}
|
|
376
|
+
else {
|
|
377
|
+
html = getDynamicHtml(resourceUri);
|
|
378
|
+
}
|
|
379
|
+
if (html) {
|
|
380
|
+
res.writeHead(200, { "Content-Type": "text/html; charset=utf-8" });
|
|
381
|
+
res.end(html);
|
|
382
|
+
return;
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
catch (error) {
|
|
386
|
+
console.error("Error serving UI resource:", error);
|
|
387
|
+
}
|
|
176
388
|
}
|
|
389
|
+
res.writeHead(404, { "Content-Type": "application/json" });
|
|
390
|
+
res.end(JSON.stringify({ error: "Not found" }));
|
|
177
391
|
});
|
|
178
392
|
return new Promise((resolve) => {
|
|
179
393
|
httpServer.listen(port, () => resolve());
|
package/dist/server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "convex-mcp-visual",
|
|
3
|
-
"version": "1.4.
|
|
3
|
+
"version": "1.4.7",
|
|
4
4
|
"description": "Visual MCP tools for exploring Convex databases - schema browser, dashboard, ER diagrams. Works as CLI, MCP server, or Claude Code plugin.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
html,body{width:100%;height:100%;overflow:hidden}:root{--bg-primary: #faf8f5;--bg-secondary: #f5f3f0;--bg-tertiary: #ebe9e6;--bg-hover: #ebe9e6;--bg-canvas: #faf8f5;--text-primary: #1a1a1a;--text-secondary: #6b6b6b;--text-muted: #999999;--border: #e6e4e1;--border-strong: #d4d2cf;--accent: #8b7355;--accent-interactive: #eb5601;--accent-hover: #d14a01;--success: #4a8c5c;--warning: #c4842d;--warning-bg: #fef3e2;--warning-text: #8a5a00;--error: #dc3545;--info: #4a7c9b;--font-mono: "SF Mono", Monaco, "Cascadia Code", "Courier New", monospace;--code-bg: #1e1e1e;--code-text: #d4d4d4;--code-key: #9cdcfe;--code-string: #ce9178;--code-number: #b5cea8;--code-boolean: #569cd6;--shadow-sm: 0 1px 2px rgba(0, 0, 0, .05);--shadow-md: 0 4px 12px rgba(0, 0, 0, .08);--shadow-lg: 0 8px 24px rgba(0, 0, 0, .12);--node-bg: #ffffff;--node-header: #f8f7f5;--node-header-selected: #eb5601;--node-border: #e6e4e1;--grid-line: #e6e4e1}[data-theme=dark]{--bg-primary: #1e1e1e;--bg-secondary: #252526;--bg-tertiary: #2d2d2d;--bg-hover: #37373d;--bg-canvas: #1e1e1e;--text-primary: #cccccc;--text-secondary: #8b8b8b;--text-muted: #6b6b6b;--border: #3c3c3c;--border-strong: #4a4a4a;--accent: #c9a87c;--accent-interactive: #ff6b35;--accent-hover: #ff8555;--success: #4ec9b0;--warning: #dcdcaa;--warning-bg: #3d3520;--warning-text: #dcdcaa;--error: #f14c4c;--info: #4fc1ff;--code-bg: #1e1e1e;--code-text: #d4d4d4;--code-key: #9cdcfe;--code-string: #ce9178;--code-number: #b5cea8;--code-boolean: #569cd6;--shadow-sm: 0 1px 2px rgba(0, 0, 0, .3);--shadow-md: 0 4px 12px rgba(0, 0, 0, .4);--shadow-lg: 0 8px 24px rgba(0, 0, 0, .5);--node-bg: #2d2d2d;--node-header: #37373d;--node-header-selected: #ff6b35;--node-border: #4a4a4a;--grid-line: #2d2d2d}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,sans-serif;background:var(--bg-primary);color:var(--text-primary);min-height:100vh;display:flex;flex-direction:column;margin:0;padding:0}.header{background:var(--bg-secondary);padding:10px 16px;border-bottom:1px solid var(--border);display:flex;justify-content:space-between;align-items:center}.header h1{font-size:18px;font-weight:600;display:flex;align-items:center;gap:8px;color:var(--text-primary)}.header-info{flex:1;margin:0 16px}.deployment-url{font-size:12px;color:var(--text-secondary);font-family:var(--font-mono)}.header-actions{display:flex;gap:8px;align-items:center}.theme-toggle{width:36px;height:36px;background:var(--bg-hover);border:1px solid var(--border);border-radius:8px;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);transition:all .2s ease}.theme-toggle:hover{background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.theme-toggle svg{width:18px;height:18px}.theme-toggle .sun-icon{display:none}.theme-toggle .moon-icon,[data-theme=dark] .theme-toggle .sun-icon{display:block}[data-theme=dark] .theme-toggle .moon-icon{display:none}.btn{background:var(--bg-primary);color:var(--text-primary);border:1px solid var(--border);padding:6px 12px;border-radius:6px;cursor:pointer;font-size:13px;transition:all .2s}.btn:hover{background:var(--bg-hover);border-color:var(--accent)}.btn-primary{background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.btn-primary:hover{background:var(--accent-hover);border-color:var(--accent-hover)}.main{display:flex;flex:1;overflow:hidden}.sidebar{width:260px;background:var(--bg-secondary);border-right:1px solid var(--border);overflow-y:auto;flex-shrink:0;display:flex;flex-direction:column;margin:0;padding:0}.sidebar-header{padding:12px 16px;font-size:11px;text-transform:uppercase;color:var(--text-secondary);border-bottom:1px solid var(--border);display:flex;align-items:center;font-weight:600;letter-spacing:.5px;gap:8px;position:relative}.sidebar-header #tableCount{background:var(--bg-hover);padding:2px 8px;border-radius:10px;font-size:11px}.sidebar-header .sidebar-collapse-btn{margin-left:auto;width:24px;height:24px;background:var(--bg-hover);border:1px solid transparent;border-radius:4px;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);transition:all .15s;flex-shrink:0}.sidebar-header .sidebar-collapse-btn:hover{background:var(--accent-interactive);color:#fff}.sidebar-header .sidebar-collapse-btn svg{transition:transform .2s}.sidebar-search{padding:8px 12px;border-bottom:1px solid var(--border)}.sidebar-search input{width:100%;background:var(--bg-primary);border:1px solid var(--border);border-radius:6px;padding:8px 10px;color:var(--text-primary);font-size:13px}.sidebar-search input:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.table-list{list-style:none;flex:1;overflow-y:auto}.table-item{padding:10px 16px;cursor:pointer;display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid var(--border);transition:background .2s}.table-item:hover{background:var(--bg-hover)}.table-item.active{background:var(--accent-interactive);color:#fff}.table-item.active .table-count{background:#fff3;color:#fff}.table-item.active .table-icon{opacity:1}.table-name{font-size:14px;font-weight:500;display:flex;align-items:center;gap:6px}.table-icon{opacity:.5}.table-count{font-size:12px;color:var(--text-secondary);background:var(--bg-hover);padding:2px 8px;border-radius:10px}.content{flex:1;display:flex;flex-direction:column;overflow:hidden;background:var(--bg-primary)}.schema-panel{padding:20px;flex:1;overflow-y:auto}.schema-title{font-size:22px;margin-bottom:8px;display:flex;align-items:center;gap:12px;color:var(--text-primary)}.schema-subtitle{color:var(--text-secondary);font-size:13px;margin-bottom:20px}.schema-grid{display:grid;grid-template-columns:1fr 1fr;gap:20px;margin-bottom:20px}@media(max-width:900px){.schema-grid{grid-template-columns:1fr}}.schema-card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:8px;overflow:hidden}.schema-card-header{background:var(--bg-hover);padding:10px 16px;font-weight:600;font-size:14px;border-bottom:1px solid var(--border);display:flex;justify-content:space-between;align-items:center;color:var(--text-primary)}.schema-card-header .count{color:var(--text-secondary);font-weight:400;font-size:12px}.schema-card-body{padding:8px;max-height:400px;overflow-y:auto}.field-row{display:flex;justify-content:space-between;padding:8px 10px;border-radius:4px;font-size:13px;font-family:var(--font-mono)}.field-row:hover{background:var(--bg-hover)}.field-name{color:var(--accent-interactive)}.field-type{color:var(--text-secondary)}.field-optional{color:var(--warning);font-size:11px;margin-left:2px}.field-system{opacity:.6}.indexes-section{margin-top:20px}.indexes-title{font-size:14px;font-weight:600;margin-bottom:12px;color:var(--text-secondary)}.index-item{background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;padding:12px 16px;margin-bottom:8px}.index-name{font-family:var(--font-mono);font-size:13px;color:var(--accent-interactive);margin-bottom:4px}.index-fields{font-size:12px;color:var(--text-secondary)}.documents-panel{background:var(--bg-secondary);border-top:1px solid var(--border);max-height:320px;overflow:hidden;display:flex;flex-direction:column}.documents-header{padding:10px 16px;display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid var(--border);background:var(--bg-hover);flex-shrink:0}.documents-title{font-weight:600;font-size:14px;color:var(--text-primary)}.pagination{display:flex;align-items:center;gap:8px}.pagination button{background:var(--bg-primary);color:var(--text-primary);border:1px solid var(--border);width:28px;height:28px;border-radius:4px;cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:12px}.pagination button:hover:not(:disabled){background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.pagination button:disabled{opacity:.5;cursor:not-allowed}.pagination span{font-size:13px;color:var(--text-secondary)}.documents-table{overflow:auto;flex:1}table{width:100%;border-collapse:collapse;font-size:13px}th,td{padding:10px 14px;text-align:left;border-bottom:1px solid var(--border);white-space:nowrap}th{background:var(--bg-hover);font-weight:600;position:sticky;top:0;z-index:1;color:var(--text-primary)}td{font-family:var(--font-mono);font-size:12px}tr:hover td{background:var(--bg-hover)}td.id-cell{color:var(--accent-interactive);cursor:pointer}td.id-cell:hover{text-decoration:underline}td.null-value{color:var(--text-secondary);font-style:italic}td.truncated{max-width:200px;overflow:hidden;text-overflow:ellipsis}.empty-state h2{margin-bottom:8px;color:var(--text-primary);font-size:18px}.empty-state p{font-size:14px}.warning-badge{background:#c4842d1a;border:1px solid var(--warning);color:var(--warning);padding:10px 14px;border-radius:6px;font-size:13px;margin-top:16px;display:flex;align-items:center;gap:8px}.warning-badge:before{content:"⚠"}.loading{display:flex;align-items:center;justify-content:center;padding:40px;gap:12px;color:var(--text-secondary)}.spinner{width:24px;height:24px;border:2px solid var(--border);border-top-color:var(--accent-interactive);border-radius:50%;animation:spin .8s linear infinite}.status-dot{width:8px;height:8px;border-radius:50%;background:var(--success)}.status-dot.error{background:var(--accent-interactive)}.status-dot.warning{background:var(--warning)}.modal-overlay{position:fixed;top:0;left:0;right:0;bottom:0;background:#1a1a1a80;display:flex;align-items:center;justify-content:center;z-index:100}.modal{background:var(--bg-primary);border:1px solid var(--border);border-radius:12px;width:90%;max-width:600px;max-height:80vh;overflow:hidden;display:flex;flex-direction:column;box-shadow:0 20px 40px #00000026}.modal-header{padding:16px 20px;border-bottom:1px solid var(--border);display:flex;justify-content:space-between;align-items:center;background:var(--bg-secondary)}.modal-header h2{font-size:16px;color:var(--text-primary)}.modal-close{background:none;border:none;color:var(--text-secondary);font-size:20px;cursor:pointer;padding:4px}.modal-close:hover{color:var(--text-primary)}.modal-body{padding:20px;flex:1;overflow-y:auto}.query-editor{width:100%;min-height:150px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;padding:12px;color:var(--text-primary);font-family:var(--font-mono);font-size:13px;resize:vertical}.query-editor:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.modal-footer{padding:16px 20px;border-top:1px solid var(--border);display:flex;justify-content:flex-end;gap:8px;background:var(--bg-secondary)}kbd{background:var(--bg-secondary);border:1px solid var(--border);border-radius:3px;padding:2px 6px;font-size:11px;font-family:var(--font-mono)}.app-container{display:flex;flex-direction:column;height:100vh;width:100vw;overflow:hidden;margin:0;padding:0;position:absolute;top:0;left:0}.view-toggle{display:flex;gap:4px;background:var(--bg-hover);padding:4px;border-radius:8px;margin-right:12px}.view-btn{display:flex;align-items:center;justify-content:center;width:32px;height:28px;border:none;background:transparent;color:var(--text-secondary);border-radius:6px;cursor:pointer;transition:all .2s}.view-btn:hover{color:var(--text-primary);background:var(--bg-secondary)}.view-btn.active{background:var(--accent-interactive);color:#fff}.graph-view{display:flex;flex-direction:column;flex:1;overflow:hidden}.graph-view .toolbar{order:-1}.code-panel{width:360px;background:#1e1e1e;border-right:1px solid #333;display:flex;flex-direction:column;flex-shrink:0}.code-header{padding:12px 16px;background:#252526;border-bottom:1px solid #333;display:flex;justify-content:space-between;align-items:center;color:#ccc;font-size:12px}.code-filename{color:#888;font-family:var(--font-mono)}.code-content{flex:1;overflow:auto;padding:0}.code-content pre{margin:0;padding:16px;font-family:var(--font-mono);font-size:12px;line-height:1.6;color:#d4d4d4}.code-content code{font-family:inherit}.json-key{color:#9cdcfe}.json-string{color:#ce9178}.json-number{color:#b5cea8}.json-boolean,.json-null{color:#569cd6}.graph-panel{flex:1;position:relative;background:var(--bg-primary);overflow:hidden;min-width:0}#graphCanvas{width:100%;height:100%;display:block}.graph-controls{position:absolute;top:16px;right:16px;display:flex;flex-direction:column;gap:8px}.graph-btn{width:36px;height:36px;border:1px solid var(--border);background:#fff;color:var(--text-primary);border-radius:8px;cursor:pointer;font-size:18px;display:flex;align-items:center;justify-content:center;transition:all .2s;box-shadow:0 2px 8px #00000014}.graph-btn:hover{background:var(--bg-hover);border-color:var(--accent)}.graph-legend{position:absolute;bottom:16px;right:16px;background:#fff;border:1px solid var(--border);border-radius:8px;padding:12px 16px;display:flex;flex-direction:column;gap:8px;font-size:12px;box-shadow:0 2px 8px #00000014}.legend-item{display:flex;align-items:center;gap:8px;color:var(--text-secondary)}.legend-dot{width:12px;height:12px;border-radius:4px;background:var(--accent-interactive)}.legend-dot.table{background:#fff;border:2px solid var(--border)}.legend-line{width:20px;height:2px;background:var(--accent)}.list-view{display:flex;flex:1;overflow:hidden}.list-view .content{display:flex;flex-direction:column;flex:1;overflow:hidden}.list-view .content #listViewContent{display:flex;flex-direction:column;flex:1;min-height:0;overflow:hidden}.list-view .table-header{padding:16px 24px;border-bottom:1px solid var(--border);background:var(--bg-primary);flex-shrink:0}.list-view .table-header-title{font-size:24px;font-weight:600;color:var(--text-primary);margin-bottom:4px}.list-view .table-header-meta{font-size:13px;color:var(--text-secondary);display:flex;align-items:center;gap:12px}.list-view .table-header-meta .badge{background:var(--bg-hover);padding:2px 8px;border-radius:4px;font-size:12px}.list-view .content-split{display:flex;flex:1;min-height:0;overflow:hidden}.list-view .schema-sidebar{width:280px;min-width:240px;max-width:360px;background:var(--bg-secondary);border-right:1px solid var(--border);display:flex;flex-direction:column;overflow:hidden;flex-shrink:0}.list-view .schema-sidebar-header{padding:12px 16px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:.5px;color:var(--text-secondary);border-bottom:1px solid var(--border);background:var(--bg-hover);display:flex;justify-content:space-between;align-items:center}.list-view .schema-sidebar-header .field-count-badge{font-size:11px;color:var(--text-secondary);background:var(--bg-primary);padding:2px 8px;border-radius:10px}.list-view .schema-fields-list{flex:1;overflow-y:auto;padding:8px 0}.list-view .schema-field-item{display:flex;justify-content:space-between;align-items:center;padding:10px 16px;border-bottom:1px solid var(--border);transition:background .15s}.list-view .schema-field-item:hover{background:var(--bg-hover)}.list-view .schema-field-item:last-child{border-bottom:none}.list-view .schema-field-name{font-family:var(--font-mono);font-size:13px;color:var(--accent-interactive);display:flex;align-items:center;gap:4px}.list-view .schema-field-name.system-field{color:var(--text-secondary)}.list-view .schema-field-optional{font-size:10px;color:var(--warning);font-weight:600}.list-view .schema-field-type{font-family:var(--font-mono);font-size:12px;color:var(--text-secondary);background:var(--bg-primary);padding:2px 8px;border-radius:4px}.list-view .schema-indexes{border-top:1px solid var(--border);padding:12px 16px;background:var(--bg-hover)}.list-view .schema-indexes-title{font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:.5px;color:var(--text-secondary);margin-bottom:8px}.list-view .schema-index-item{font-family:var(--font-mono);font-size:12px;color:var(--accent);padding:4px 0}.list-view .documents-main{flex:1;display:flex;flex-direction:column;overflow:hidden;background:var(--bg-primary);min-width:0;min-height:0}.list-view .documents-toolbar{padding:12px 20px;display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid var(--border);background:var(--bg-secondary);flex-shrink:0}.list-view .documents-toolbar-title{font-size:13px;font-weight:600;color:var(--text-primary);display:flex;align-items:center;gap:8px}.list-view .documents-toolbar-title .doc-count{font-weight:400;color:var(--text-secondary)}.list-view .documents-table-wrapper{flex:1;min-height:0;overflow:auto;padding:0}.list-view .documents-table-wrapper table{width:100%;border-collapse:collapse;font-size:13px}.list-view .documents-table-wrapper th{position:sticky;top:0;background:var(--bg-hover);font-weight:600;text-align:left;padding:12px 16px;border-bottom:2px solid var(--border);white-space:nowrap;z-index:1}.list-view .documents-table-wrapper td{padding:10px 16px;border-bottom:1px solid var(--border);font-family:var(--font-mono);font-size:12px;vertical-align:top;max-width:300px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.list-view .documents-table-wrapper tr:hover td{background:var(--bg-secondary)}.list-view .documents-table-wrapper td.id-cell{color:var(--accent-interactive);cursor:pointer}.list-view .documents-table-wrapper td.id-cell:hover{text-decoration:underline}.list-view .pagination-bar{padding:12px 20px;display:flex;justify-content:space-between;align-items:center;border-top:1px solid var(--border);background:var(--bg-secondary);flex-shrink:0}.list-view .pagination-info{font-size:13px;color:var(--text-secondary)}.list-view .pagination-controls{display:flex;align-items:center;gap:8px}.list-view .pagination-controls button{width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:var(--bg-primary);border:1px solid var(--border);border-radius:6px;cursor:pointer;color:var(--text-primary);font-size:14px;transition:all .15s}.list-view .pagination-controls button:hover:not(:disabled){background:var(--accent-interactive);border-color:var(--accent-interactive);color:#fff}.list-view .pagination-controls button:disabled{opacity:.4;cursor:not-allowed}.list-view .pagination-controls .page-indicator{font-size:13px;color:var(--text-secondary);min-width:80px;text-align:center}.list-view .documents-empty{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:60px 20px;color:var(--text-secondary);text-align:center}.list-view .documents-empty-icon{font-size:48px;margin-bottom:16px;opacity:.5}.list-view .documents-empty h3{font-size:16px;color:var(--text-primary);margin-bottom:4px}.list-view .documents-empty p{font-size:13px}.list-view .schema-panel,.list-view .documents-panel{display:none!important}.sidebar-container{display:flex;flex-shrink:0;position:relative}.sidebar-container.collapsed .sidebar{width:48px!important;min-width:48px!important;overflow:hidden}.sidebar-container.collapsed .sidebar .sidebar-header{padding:12px;justify-content:center}.sidebar-container.collapsed .sidebar .sidebar-header>span{display:none}.sidebar-container.collapsed .sidebar .sidebar-header .sidebar-collapse-btn{margin-left:0}.sidebar-container.collapsed .sidebar .sidebar-search,.sidebar-container.collapsed .sidebar .table-list{display:none}.sidebar-container.collapsed .code-panel{width:0!important;min-width:0!important;padding:0;overflow:hidden;border:none}.sidebar-container.collapsed .resize-handle{display:none}.resize-handle{width:4px;cursor:col-resize;background:transparent;transition:background .2s;position:relative;z-index:10}.resize-handle:hover,.resize-handle.dragging{background:var(--accent-interactive)}.resize-handle:after{content:"";position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);width:4px;height:40px;border-radius:2px;background:var(--border);opacity:0;transition:opacity .2s}.resize-handle:hover:after{opacity:1;background:#fff}.sidebar-toggle{display:none}.list-view .sidebar-header{position:relative}.list-view .sidebar-header .sidebar-collapse-btn{position:absolute;right:8px;top:50%;transform:translateY(-50%);width:24px;height:24px;background:var(--bg-hover);border:1px solid var(--border);border-radius:4px;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);transition:all .2s}.list-view .sidebar-header .sidebar-collapse-btn:hover{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.list-view .sidebar-header .sidebar-collapse-btn svg{width:12px;height:12px;transition:transform .2s}.sidebar-container.collapsed .sidebar-collapse-btn svg{transform:rotate(180deg)}.list-view .sidebar{min-width:180px;max-width:400px}.list-view .sidebar-container{height:100%}.toolbar{display:flex;align-items:center;padding:8px 16px;background:var(--bg-secondary);border-bottom:1px solid var(--border);gap:4px;flex-shrink:0;width:100%;z-index:10}.toolbar-group{display:flex;align-items:center;gap:2px}.toolbar-separator{width:1px;height:24px;background:var(--border);margin:0 8px}.toolbar-spacer{flex:1}.toolbar-btn{display:flex;align-items:center;gap:6px;padding:6px 10px;background:transparent;border:1px solid transparent;border-radius:6px;color:var(--text-secondary);font-size:13px;cursor:pointer;transition:all .15s ease}.toolbar-btn:hover:not(:disabled){background:var(--bg-hover);color:var(--text-primary);border-color:var(--border)}.toolbar-btn.active{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.toolbar-btn.active:hover{background:var(--accent-hover);border-color:var(--accent-hover)}.toolbar-btn:disabled{opacity:.4;cursor:not-allowed}.toolbar-btn.icon-only{padding:6px}.toolbar-btn svg{flex-shrink:0}.toolbar-btn .dropdown-arrow{margin-left:2px;opacity:.6}.zoom-group{display:flex;align-items:center;gap:4px;background:var(--bg-primary);border:1px solid var(--border);border-radius:6px;padding:2px}.zoom-group .toolbar-btn{border-radius:4px}.zoom-display{min-width:48px;text-align:center;font-size:12px;font-weight:500;color:var(--text-secondary);font-family:var(--font-mono)}.graph-content{display:flex;flex:1;overflow:hidden;flex-direction:row}.graph-content .enhanced-sidebar{order:0}.graph-content .graph-panel{order:1;flex:1}.graph-content .sidebar-container{order:2}.sidebar-container.hidden .code-panel{width:0!important;min-width:0;padding:0;overflow:hidden;border:none}.sidebar-container.hidden .resize-handle{display:none}.sidebar-container.hidden .sidebar-toggle{left:0;border-radius:0 6px 6px 0}.zoom-panel{position:absolute;bottom:20px;left:20px;display:flex;flex-direction:column;gap:4px;background:#fff;border:1px solid var(--border);border-radius:10px;padding:8px;box-shadow:0 4px 12px #0000001a;z-index:10}.zoom-btn{width:32px;height:32px;display:flex;align-items:center;justify-content:center;background:transparent;border:none;border-radius:6px;color:var(--text-secondary);cursor:pointer;transition:all .15s ease}.zoom-btn:hover{background:var(--bg-hover);color:var(--text-primary)}.zoom-btn:active{background:var(--accent-interactive);color:#fff}.zoom-level{text-align:center;font-size:11px;font-weight:500;color:var(--text-secondary);font-family:var(--font-mono);padding:4px 0}.zoom-separator{height:1px;background:var(--border);margin:4px 0}.dropdown-menu{position:absolute;background:#fff;border:1px solid var(--border);border-radius:10px;box-shadow:0 8px 24px #0000001f;z-index:100;overflow:hidden;min-width:180px}.export-menu{top:52px;left:140px}.dropdown-item{display:flex;align-items:center;gap:10px;width:100%;padding:10px 14px;background:transparent;border:none;color:var(--text-primary);font-size:13px;cursor:pointer;transition:background .15s ease;text-align:left}.dropdown-item:hover{background:var(--bg-hover)}.dropdown-item svg{color:var(--text-secondary)}.filter-menu{position:absolute;top:52px;right:200px;width:280px}.filter-header{display:flex;justify-content:space-between;align-items:center;padding:12px 16px;border-bottom:1px solid var(--border);font-weight:600;font-size:14px}.filter-close{background:none;border:none;font-size:18px;color:var(--text-secondary);cursor:pointer;padding:0;line-height:1}.filter-close:hover{color:var(--text-primary)}.filter-body{padding:16px}.filter-group{margin-bottom:14px}.filter-group:last-child{margin-bottom:0}.filter-group label{display:block;font-size:12px;font-weight:500;color:var(--text-secondary);margin-bottom:6px}.filter-group input[type=text],.filter-group select{width:100%;padding:8px 10px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;font-size:13px;color:var(--text-primary)}.filter-group input[type=text]:focus,.filter-group select:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.filter-group.checkbox label{display:flex;align-items:center;gap:8px;cursor:pointer}.filter-group.checkbox input[type=checkbox]{width:16px;height:16px;accent-color:var(--accent-interactive)}.filter-footer{display:flex;justify-content:flex-end;gap:8px;padding:12px 16px;border-top:1px solid var(--border);background:var(--bg-secondary)}.filter-footer .btn{padding:6px 14px;font-size:13px}.btn-secondary{background:var(--bg-primary);border:1px solid var(--border);color:var(--text-primary)}.btn-secondary:hover{background:var(--bg-hover)}.enhanced-sidebar{width:260px;background:var(--bg-secondary);border-right:1px solid var(--border);display:flex;flex-direction:column;flex-shrink:0;overflow:hidden;height:100%;margin:0;padding:0;position:relative;transition:width .2s ease,min-width .2s ease}.graph-sidebar-toggle{position:absolute;top:50%;right:-14px;transform:translateY(-50%);width:28px;height:48px;background:var(--bg-secondary);border:1px solid var(--border);border-left:none;border-radius:0 8px 8px 0;cursor:pointer;display:flex;align-items:center;justify-content:center;color:var(--text-secondary);z-index:20;transition:all .15s ease;box-shadow:2px 0 8px #0000000d}.graph-sidebar-toggle:hover{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.graph-sidebar-toggle svg{transition:transform .2s ease}.enhanced-sidebar.collapsed{width:0!important;min-width:0!important;border-right:none;overflow:visible}.enhanced-sidebar.collapsed .sidebar-deployment,.enhanced-sidebar.collapsed .sidebar-section{display:none}.enhanced-sidebar.collapsed .graph-sidebar-toggle{right:-28px;border-left:1px solid var(--border);border-radius:0 8px 8px 0}.sidebar-deployment{padding:14px 16px;border-bottom:1px solid var(--border);background:linear-gradient(to bottom,var(--bg-hover),var(--bg-secondary))}.deployment-status{display:flex;align-items:center;gap:8px;margin-bottom:4px}.status-indicator{width:8px;height:8px;border-radius:50%;background:var(--success)}.status-indicator.error{background:var(--accent-interactive)}.deployment-label{font-size:13px;font-weight:600;color:var(--text-primary)}.deployment-url-small{font-size:11px;font-family:var(--font-mono);color:var(--text-secondary);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.sidebar-section{border-bottom:1px solid var(--border)}.section-header{display:flex;align-items:center;gap:8px;padding:12px 14px;cursor:pointer;-webkit-user-select:none;user-select:none;transition:background .15s ease}.section-header:hover{background:var(--bg-hover)}.section-chevron{transition:transform .2s ease;color:var(--text-secondary)}.sidebar-section:not(.collapsed) .section-chevron{transform:rotate(90deg)}.section-title{font-size:11px;font-weight:600;letter-spacing:.5px;color:var(--text-secondary);text-transform:uppercase}.section-count{margin-left:auto;font-size:11px;color:var(--text-secondary);background:var(--bg-hover);padding:2px 8px;border-radius:10px}.section-content{max-height:500px;overflow:hidden;transition:max-height .3s ease,opacity .2s ease}.sidebar-section.collapsed .section-content{max-height:0;opacity:0}.sidebar-toolbar{display:flex;gap:6px;padding:8px 12px;border-bottom:1px solid var(--border);background:var(--bg-primary)}.sidebar-filter{flex:1;padding:6px 10px;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;font-size:12px;color:var(--text-primary)}.sidebar-filter:focus{outline:none;border-color:var(--accent-interactive);box-shadow:0 0 0 2px #eb56011a}.sort-btn{width:28px;height:28px;display:flex;align-items:center;justify-content:center;background:var(--bg-secondary);border:1px solid var(--border);border-radius:6px;color:var(--text-secondary);cursor:pointer;transition:all .15s ease}.sort-btn:hover{background:var(--bg-hover);border-color:var(--accent);color:var(--text-primary)}.sidebar-table-list{list-style:none;max-height:400px;overflow-y:auto;padding:4px 0}.sidebar-table-item{display:flex;align-items:center;justify-content:space-between;padding:8px 14px;cursor:pointer;transition:background .15s ease}.sidebar-table-item:hover{background:var(--bg-hover)}.sidebar-table-item.active{background:var(--accent-interactive)}.sidebar-table-item.active .table-name,.sidebar-table-item.active .table-icon{color:#fff}.sidebar-table-item.active .table-item-meta span{background:#fff3;color:#fff}.sidebar-table-item.filtered-out{opacity:.4}.table-item-main{display:flex;align-items:center;gap:8px;min-width:0}.table-icon{flex-shrink:0;color:var(--text-secondary)}.sidebar-table-item .table-name{font-size:13px;font-weight:500;color:var(--text-primary);overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.table-item-meta{display:flex;align-items:center;gap:6px}.field-count,.doc-count{font-size:10px;padding:2px 6px;border-radius:8px;background:var(--bg-hover);color:var(--text-secondary)}.field-count{background:#eb56011a;color:var(--accent-interactive)}.convex-info-list{padding:8px 14px}.convex-info-item{display:flex;justify-content:space-between;padding:8px 0;border-bottom:1px solid var(--border)}.convex-info-item:last-child{border-bottom:none}.info-label{font-size:12px;color:var(--text-secondary)}.info-value{font-size:12px;font-weight:600;color:var(--text-primary)}.legend-line.dashed{background:repeating-linear-gradient(to right,var(--accent) 0px,var(--accent) 4px,transparent 4px,transparent 8px)}.graph-view .sidebar-container{display:flex;flex-direction:row;flex-shrink:0;position:relative;border-left:1px solid var(--border);height:100%}.graph-view .sidebar-container .code-panel{border-right:none;border-left:none;min-width:200px;max-width:600px}.resize-handle-left{cursor:col-resize;width:4px;background:transparent;transition:background .2s}.resize-handle-left:hover,.resize-handle-left.dragging{background:var(--accent-interactive)}.sidebar-toggle-right{position:absolute;top:12px;left:-32px;right:auto}.sidebar-toggle-right svg{transform:rotate(180deg)}.sidebar-container.collapsed .sidebar-toggle-right{left:-24px}.sidebar-container.collapsed .sidebar-toggle-right svg{transform:rotate(0)}.sidebar-container.hidden{display:none}.tooltip{position:fixed;z-index:1000;max-width:280px;padding:10px 14px;background:#fff;border:1px solid var(--border);border-radius:8px;box-shadow:0 8px 24px #0000001f;font-size:12px;line-height:1.5;animation:tooltipFadeIn .2s ease;pointer-events:none}@keyframes tooltipFadeIn{0%{opacity:0;transform:translateY(4px)}to{opacity:1;transform:translateY(0)}}.tooltip-title{font-weight:600;color:var(--text-primary);margin-bottom:4px}.tooltip-content{color:var(--text-secondary)}.tooltip-content code{background:var(--bg-secondary);padding:1px 5px;border-radius:4px;font-family:var(--font-mono);font-size:11px;color:var(--accent-interactive)}.tooltip-content strong{color:var(--text-primary)}.tooltip-content em{color:var(--warning);font-style:normal}.tooltip-system{border-left:3px solid var(--text-secondary)}.tooltip-field{border-left:3px solid var(--accent-interactive)}.tooltip-relationship{border-left:3px solid var(--accent)}.loading-overlay{position:absolute;top:0;left:0;right:0;bottom:0;background:#faf8f5e6;display:flex;align-items:center;justify-content:center;z-index:50}.loading-content{text-align:center}.loading-spinner{width:40px;height:40px;border:3px solid var(--border);border-top-color:var(--accent-interactive);border-radius:50%;animation:spin .8s linear infinite;margin:0 auto 16px}.loading-text{color:var(--text-secondary);font-size:14px}.skeleton-sidebar{padding:16px}.skeleton-item{display:flex;align-items:center;gap:10px;padding:10px 0;border-bottom:1px solid var(--border)}.skeleton-icon{width:20px;height:20px;border-radius:4px;background:linear-gradient(90deg,#f0efed 25%,#e6e4e1,#f0efed 75%);background-size:200px 100%;animation:shimmer 1.5s infinite}.skeleton-text{flex:1;height:14px;border-radius:4px;background:linear-gradient(90deg,#f0efed 25%,#e6e4e1,#f0efed 75%);background-size:200px 100%;animation:shimmer 1.5s infinite}.skeleton-badge{width:32px;height:18px;border-radius:9px;background:linear-gradient(90deg,#f0efed 25%,#e6e4e1,#f0efed 75%);background-size:200px 100%;animation:shimmer 1.5s infinite}@keyframes shimmer{0%{background-position:-200px 0}to{background-position:200px 0}}.empty-state-enhanced{text-align:center;padding:60px 30px;color:var(--text-secondary)}.empty-icon{font-size:48px;margin-bottom:16px;opacity:.6}.empty-title{font-size:18px;font-weight:600;color:var(--text-primary);margin-bottom:8px}.empty-description{font-size:14px;margin-bottom:20px;max-width:320px;margin-left:auto;margin-right:auto}.empty-action .btn{display:inline-flex;align-items:center;gap:8px}.table-card,.sidebar-table-item{transition:transform .15s ease,box-shadow .15s ease}.graph-view,.list-view{animation:viewFadeIn .3s ease}@keyframes viewFadeIn{0%{opacity:0}to{opacity:1}}.toolbar-btn:active,.zoom-btn:active,.graph-btn:active{transform:scale(.95)}.toolbar-btn:focus-visible,.zoom-btn:focus-visible,.btn:focus-visible{outline:2px solid var(--accent-interactive);outline-offset:2px}@media(min-width:1600px){.enhanced-sidebar{width:300px}.graph-view .sidebar-container .code-panel{width:420px!important}.sidebar-table-list{max-height:600px}}@media(min-width:1920px){.enhanced-sidebar{width:320px}.graph-view .sidebar-container .code-panel{width:480px!important}}@media(max-width:1200px){.enhanced-sidebar{width:220px}.code-panel{width:300px!important}}@media(max-width:900px){.toolbar-btn span{display:none}.toolbar-btn,.toolbar-btn.icon-only{padding:6px}.zoom-display{min-width:40px;font-size:11px}.enhanced-sidebar{width:180px}.sidebar-deployment{padding:10px 12px}.deployment-label{font-size:12px}}@media print{.toolbar,.enhanced-sidebar,.sidebar-container,.zoom-panel,.graph-legend{display:none!important}.graph-panel{width:100%!important}}.shortcuts-modal{position:fixed;top:0;left:0;right:0;bottom:0;background:#00000080;display:flex;align-items:center;justify-content:center;z-index:1000}.shortcuts-modal-content{background:var(--bg-primary);border-radius:12px;box-shadow:var(--shadow-lg);max-width:480px;width:90%;max-height:80vh;overflow:hidden}.shortcuts-modal-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid var(--border)}.shortcuts-modal-header h3{margin:0;font-size:16px;font-weight:600;color:var(--text-primary)}.shortcuts-close{background:none;border:none;font-size:24px;cursor:pointer;color:var(--text-secondary);padding:0;line-height:1}.shortcuts-close:hover{color:var(--text-primary)}.shortcuts-modal-body{padding:20px;overflow-y:auto;max-height:calc(80vh - 60px)}.shortcuts-section{margin-bottom:20px}.shortcuts-section:last-child{margin-bottom:0}.shortcuts-section h4{font-size:12px;font-weight:600;color:var(--text-secondary);text-transform:uppercase;letter-spacing:.5px;margin:0 0 12px}.shortcut-item{display:flex;align-items:center;gap:8px;padding:8px 0;font-size:13px;color:var(--text-primary)}.shortcut-item kbd{display:inline-flex;align-items:center;justify-content:center;min-width:24px;height:24px;padding:0 8px;background:var(--bg-tertiary);border:1px solid var(--border);border-radius:4px;font-family:var(--font-mono);font-size:11px;color:var(--text-primary)}*{box-sizing:border-box;margin:0;padding:0}html{width:100%;height:100%}body{width:100%;min-height:100%;overflow-x:hidden;overflow-y:auto}:root{--bg-primary: #faf8f5;--bg-secondary: #f5f3f0;--bg-hover: #ebe9e6;--text-primary: #1a1a1a;--text-secondary: #6b6b6b;--border: #e6e4e1;--accent: #8b7355;--accent-interactive: #eb5601;--accent-hover: #d14a01;--success: #4a8c5c;--warning: #c4842d;--info: #4a7c9b;--font-mono: "SF Mono", Monaco, "Cascadia Code", "Courier New", monospace;--chart-gradient-start: #eb5601;--chart-gradient-end: #d14a01;--chart-area-opacity: .1}[data-theme=dark]{--bg-primary: #1e1e1e;--bg-secondary: #252526;--bg-hover: #37373d;--text-primary: #cccccc;--text-secondary: #8b8b8b;--border: #3c3c3c;--accent: #c9a87c;--accent-interactive: #ff6b35;--accent-hover: #ff8555;--success: #4ec9b0;--warning: #dcdcaa;--info: #4fc1ff;--chart-gradient-start: #ff6b35;--chart-gradient-end: #ff8555;--chart-area-opacity: .15}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,sans-serif;background:var(--bg-primary);color:var(--text-primary);min-height:100vh;padding:16px 20px 40px;margin:0}.header{display:flex;justify-content:space-between;align-items:center;margin-bottom:20px}.header h1{font-size:20px;display:flex;align-items:center;gap:10px;color:var(--text-primary)}.header-right{display:flex;align-items:center;gap:16px}.deployment-url{font-size:12px;color:var(--text-secondary);font-family:var(--font-mono);margin-right:16px}.last-update{color:var(--text-secondary);font-size:12px}.theme-toggle-btn{background:var(--bg-hover);border:1px solid var(--border);color:var(--text-primary);width:32px;height:32px;border-radius:6px;cursor:pointer;display:flex;align-items:center;justify-content:center;font-size:16px;transition:all .2s ease}.theme-toggle-btn:hover{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.chart-subtitle{font-size:12px;color:var(--text-secondary)}.pie-legend{padding:20px}.pie-item{display:flex;align-items:center;gap:8px;padding:8px 0;border-bottom:1px solid var(--border)}.pie-item:last-child{border-bottom:none}.pie-color{width:12px;height:12px;border-radius:2px}.pie-label{flex:1;font-size:13px}.pie-value{font-family:var(--font-mono);font-size:12px;color:var(--text-secondary)}.status-dot{width:8px;height:8px;border-radius:50%;background:var(--success);animation:pulse 2s infinite}.status-dot.disconnected{background:var(--accent-interactive);animation:none}@keyframes pulse{0%,to{opacity:1}50%{opacity:.5}}.metrics-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(220px,1fr));gap:16px;margin-bottom:24px}.metric-card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:12px;padding:20px;transition:transform .2s,box-shadow .2s}.metric-card:hover{transform:translateY(-2px);box-shadow:0 4px 12px #00000014}.metric-header{display:flex;justify-content:space-between;align-items:flex-start;margin-bottom:12px}.metric-label{font-size:12px;color:var(--text-secondary);text-transform:uppercase;letter-spacing:.5px;font-weight:600}.metric-icon{width:32px;height:32px;background:var(--bg-hover);border-radius:8px;display:flex;align-items:center;justify-content:center;font-size:16px;color:var(--accent)}.metric-value{font-size:36px;font-weight:700;margin-bottom:8px;font-family:var(--font-mono);color:var(--text-primary)}.metric-change{font-size:13px;display:flex;align-items:center;gap:4px}.metric-change.positive{color:var(--success)}.metric-change.negative{color:var(--accent-interactive)}.metric-change.neutral{color:var(--text-secondary)}.metric-source{font-size:11px;color:var(--text-secondary);margin-top:8px;font-family:var(--font-mono)}.charts-grid{display:grid;grid-template-columns:repeat(auto-fit,minmax(400px,1fr));gap:16px}@media(max-width:900px){.charts-grid{grid-template-columns:1fr}}.chart-card{background:var(--bg-secondary);border:1px solid var(--border);border-radius:12px;padding:20px}.chart-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.chart-title{font-size:14px;font-weight:600;color:var(--text-primary)}.chart-actions{display:flex;gap:8px}.chart-btn{background:var(--bg-hover);border:1px solid var(--border);color:var(--text-secondary);padding:4px 8px;border-radius:4px;font-size:11px;cursor:pointer;transition:all .2s}.chart-btn:hover{color:var(--text-primary);border-color:var(--accent)}.chart-btn.active{background:var(--accent-interactive);color:#fff;border-color:var(--accent-interactive)}.chart-container{height:220px;background:var(--bg-hover);border-radius:8px;display:flex;align-items:center;justify-content:center;position:relative;overflow:hidden}.bar-chart{display:flex;align-items:flex-end;justify-content:space-around;height:100%;width:100%;padding:20px;gap:8px}.bar{flex:1;max-width:40px;background:linear-gradient(to top,var(--accent-interactive),var(--accent-hover));border-radius:4px 4px 0 0;transition:height .3s ease;position:relative;min-height:4px}.bar:after{content:attr(data-value);position:absolute;top:-20px;left:50%;transform:translate(-50%);font-size:11px;color:var(--text-secondary);font-family:var(--font-mono)}.bar-label{position:absolute;bottom:-20px;left:50%;transform:translate(-50%);font-size:10px;color:var(--text-secondary);white-space:nowrap}.tables-overview-card{grid-column:1 / -1}.tables-overview-container{max-height:400px;overflow-y:auto;overflow-x:hidden;padding:0 4px}.horizontal-bar-chart{display:flex;flex-direction:column;gap:8px;padding:12px 0}.h-bar-row{display:grid;grid-template-columns:140px 1fr 60px;align-items:center;gap:12px;padding:6px 8px;border-radius:6px;transition:background .15s ease}.h-bar-row:hover{background:var(--bg-hover)}.h-bar-label{font-size:12px;font-weight:500;color:var(--text-primary);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;font-family:var(--font-mono)}.h-bar-track{height:20px;background:var(--bg-hover);border-radius:4px;overflow:hidden;position:relative}.h-bar-fill{height:100%;background:linear-gradient(to right,var(--accent-interactive),var(--accent-hover));border-radius:4px;transition:width .4s cubic-bezier(.4,0,.2,1);min-width:2px}.h-bar-value{font-size:12px;font-weight:600;color:var(--text-secondary);font-family:var(--font-mono);text-align:right}.tables-overview-container::-webkit-scrollbar{width:6px}.tables-overview-container::-webkit-scrollbar-track{background:var(--bg-hover);border-radius:3px}.tables-overview-container::-webkit-scrollbar-thumb{background:var(--border);border-radius:3px}.tables-overview-container::-webkit-scrollbar-thumb:hover{background:var(--text-secondary)}@media(max-width:600px){.h-bar-row{grid-template-columns:100px 1fr 50px;gap:8px}.h-bar-label,.h-bar-value{font-size:11px}}.recent-activity-card{grid-column:1 / -1}.recent-activity-container{max-height:400px;overflow-y:auto;overflow-x:auto}.activity-table{width:100%;border-collapse:collapse;font-size:13px}.activity-table thead{position:sticky;top:0;z-index:1}.activity-table th{background:var(--bg-secondary);font-weight:600;text-align:left;padding:12px 16px;border-bottom:2px solid var(--border);color:var(--text-primary);font-size:11px;text-transform:uppercase;letter-spacing:.5px}.activity-table td{padding:12px 16px;border-bottom:1px solid var(--border);vertical-align:middle}.activity-table tbody tr{transition:background .15s ease}.activity-table tbody tr:hover{background:var(--bg-hover)}.table-badge{display:inline-block;padding:4px 10px;background:var(--bg-hover);border-radius:4px;font-family:var(--font-mono);font-size:12px;font-weight:500;color:var(--text-primary)}.id-cell{font-family:var(--font-mono);color:var(--accent);font-size:12px}.time-cell{color:var(--text-secondary);font-size:12px;white-space:nowrap}.recent-activity-container::-webkit-scrollbar{width:6px;height:6px}.recent-activity-container::-webkit-scrollbar-track{background:var(--bg-hover);border-radius:3px}.recent-activity-container::-webkit-scrollbar-thumb{background:var(--border);border-radius:3px}.recent-activity-container::-webkit-scrollbar-thumb:hover{background:var(--text-secondary)}@media(max-width:600px){.activity-table th,.activity-table td{padding:10px 12px}.table-badge{font-size:11px;padding:3px 8px}.id-cell,.time-cell{font-size:11px}}.line-chart{width:100%;height:100%;padding:20px}.line-chart svg{width:100%;height:100%}.line-chart path{fill:none;stroke:var(--accent-interactive);stroke-width:2}.line-chart .area{fill:var(--accent-interactive);opacity:.1}.table-chart{width:100%;height:100%;overflow:auto;padding:0}.table-chart table{width:100%;border-collapse:collapse;font-size:12px}.table-chart th,.table-chart td{padding:10px 12px;text-align:left;border-bottom:1px solid var(--border)}.table-chart th{background:var(--bg-secondary);font-weight:600;position:sticky;top:0;color:var(--text-primary)}.table-chart tr:hover td{background:#eb56010d}.empty-state{text-align:center;color:var(--text-secondary);padding:40px}.empty-state h3{color:var(--text-primary);margin-bottom:8px;font-size:16px}.empty-state p{font-size:13px}.loading{display:flex;align-items:center;justify-content:center;gap:12px;color:var(--text-secondary)}.spinner{width:20px;height:20px;border:2px solid var(--border);border-top-color:var(--accent-interactive);border-radius:50%;animation:spin .8s linear infinite}@media(max-width:600px){body{padding:12px}.header{flex-direction:column;align-items:flex-start;gap:12px}.metrics-grid{grid-template-columns:1fr}.metric-value{font-size:28px}}:root{--bg-primary: #0d1117;--bg-secondary: #161b22;--bg-tertiary: #21262d;--text-primary: #f0f6fc;--text-secondary: #8b949e;--text-muted: #484f58;--border-color: #30363d;--accent-color: #58a6ff;--success-color: #3fb950;--warning-color: #d29922;--error-color: #f85149;--processing-color: #58a6ff}[data-theme=github-light]{--bg-primary: #ffffff;--bg-secondary: #f6f8fa;--bg-tertiary: #eaeef2;--text-primary: #1f2328;--text-secondary: #656d76;--text-muted: #8c959f;--border-color: #d0d7de;--accent-color: #0969da;--success-color: #1a7f37;--warning-color: #9a6700;--error-color: #cf222e;--processing-color: #0969da}*{margin:0;padding:0;box-sizing:border-box}body{font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Noto Sans,Helvetica,Arial,sans-serif;background:var(--bg-primary);color:var(--text-primary);line-height:1.5;min-height:100vh}#app{padding:24px;max-width:1600px;margin:0 auto}.header{display:flex;justify-content:space-between;align-items:center;margin-bottom:24px;padding-bottom:16px;border-bottom:1px solid var(--border-color)}.header h1{font-size:24px;font-weight:600}.header-meta{display:flex;align-items:center;gap:16px}.deployment{font-size:13px;color:var(--text-secondary);padding:4px 8px;background:var(--bg-secondary);border-radius:4px}.theme-toggle{background:var(--bg-secondary);border:1px solid var(--border-color);border-radius:6px;width:36px;height:36px;padding:0;cursor:pointer;color:var(--text-secondary);display:flex;align-items:center;justify-content:center;transition:background .15s,color .15s}.theme-toggle:hover{background:var(--bg-tertiary);color:var(--text-primary)}.theme-toggle svg{display:block}.theme-toggle .sun-icon{display:block}.theme-toggle .moon-icon,[data-theme=github-light] .theme-toggle .sun-icon{display:none}[data-theme=github-light] .theme-toggle .moon-icon{display:block}.mode-tabs{display:flex;gap:8px;margin-bottom:24px}.mode-tab{padding:10px 20px;border-radius:6px;border:1px solid var(--border-color);background:var(--bg-secondary);color:var(--text-secondary);cursor:pointer;font-size:14px;font-weight:500;transition:all .15s}.mode-tab:hover{background:var(--bg-tertiary);color:var(--text-primary)}.mode-tab.active{background:var(--accent-color);color:#fff;border-color:var(--accent-color)}.section{margin-bottom:32px}.section-header{margin-bottom:16px}.section-title{font-size:18px;font-weight:600;margin-bottom:4px}.section-subtitle{font-size:13px;color:var(--text-secondary)}.kanban-container{display:flex;gap:16px;overflow-x:auto;padding:4px 0 16px;min-height:400px}.kanban-column{flex:0 0 300px;background:var(--bg-secondary);border-radius:8px;border:1px solid var(--border-color);display:flex;flex-direction:column;max-height:calc(100vh - 240px)}.column-header{display:flex;align-items:center;gap:10px;padding:14px 16px;border-bottom:1px solid var(--border-color)}.column-dot{width:12px;height:12px;border-radius:50%;flex-shrink:0}.column-title{font-weight:600;font-size:14px;flex:1}.column-count{background:var(--bg-tertiary);padding:2px 10px;border-radius:12px;font-size:12px;color:var(--text-secondary);font-weight:500}.column-items{flex:1;overflow-y:auto;padding:12px;display:flex;flex-direction:column;gap:10px}.kanban-card{background:var(--bg-primary);border:1px solid var(--border-color);border-radius:8px;padding:14px;cursor:pointer;transition:transform .1s,box-shadow .1s,border-color .15s}.kanban-card:hover{transform:translateY(-2px);box-shadow:0 4px 12px #00000026;border-color:var(--accent-color)}.card-title{font-weight:500;font-size:14px;margin-bottom:6px;word-break:break-word;color:var(--text-primary)}.card-meta{font-size:12px;color:var(--text-secondary);margin-bottom:8px}.card-footer{display:flex;align-items:center;justify-content:space-between;margin-top:8px}.card-badge{display:inline-block;padding:3px 8px;border-radius:4px;font-size:11px;font-weight:600;text-transform:uppercase;letter-spacing:.3px}.badge-cron{background:#8b5cf633;color:#a78bfa}.badge-scheduled{background:#3b82f633;color:#60a5fa}.badge-agent{background:#22c55e33;color:#4ade80}.badge-error{background:#f8514933;color:#f87171}.status-pending{color:var(--warning-color)}.status-running{color:var(--processing-color)}.status-completed{color:var(--success-color)}.status-failed{color:var(--error-color)}.status-idle{color:var(--text-secondary)}.status-processing{color:var(--processing-color)}.status-waiting{color:var(--warning-color)}.empty-state{text-align:center;padding:60px 20px;color:var(--text-secondary)}.empty-state p{margin-bottom:12px}.empty-state a{color:var(--accent-color);text-decoration:none}.empty-state a:hover{text-decoration:underline}.loading{display:flex;align-items:center;justify-content:center;padding:60px;color:var(--text-secondary)}.loading-spinner{width:24px;height:24px;border:2px solid var(--border-color);border-top-color:var(--accent-color);border-radius:50%;animation:spin .8s linear infinite;margin-right:12px}@keyframes spin{to{transform:rotate(360deg)}}.column-items::-webkit-scrollbar{width:6px}.column-items::-webkit-scrollbar-track{background:transparent}.column-items::-webkit-scrollbar-thumb{background:var(--border-color);border-radius:3px}.column-items::-webkit-scrollbar-thumb:hover{background:var(--text-muted)}@media(max-width:768px){#app{padding:16px}.kanban-column{flex:0 0 280px}.header{flex-direction:column;gap:12px;align-items:flex-start}}
|