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.
@@ -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-CfvT5wrm.css">
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
- <!DOCTYPE html>
2
- <html lang="en">
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-CfvT5wrm.css">
11
- </head>
12
- <body>
13
- <div id="app"></div>
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
- <!DOCTYPE html>
2
- <html lang="en">
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-CfvT5wrm.css">
11
- </head>
12
- <body>
13
- <div id="app"></div>
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 'fs';
7
- import { join, dirname } from 'path';
8
- import { fileURLToPath } from 'url';
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, '..', 'apps', 'realtime-dashboard.html');
12
+ const bundledPath = join(__dirname, "..", "apps", "realtime-dashboard.html");
13
13
  if (existsSync(bundledPath)) {
14
- return readFileSync(bundledPath, 'utf-8');
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: #1a1a2e;
35
- --bg-secondary: #16213e;
36
- --bg-tertiary: #0f3460;
37
- --text-primary: #eee;
38
- --text-secondary: #aaa;
39
- --accent: #e94560;
40
- --border: #333;
41
- --success: #4caf50;
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,7 @@
1
+ /**
2
+ * Kanban Board Resource Handler
3
+ *
4
+ * Serves the bundled Kanban Board HTML UI.
5
+ */
6
+ export declare function getKanbanResourceContent(): Promise<string>;
7
+ //# sourceMappingURL=kanban-board.d.ts.map
@@ -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: #1a1a2e;
35
- --bg-secondary: #16213e;
36
- --bg-tertiary: #0f3460;
37
- --text-primary: #eee;
38
- --text-secondary: #aaa;
39
- --accent: #e94560;
40
- --accent-hover: #ff6b6b;
41
- --border: #333;
42
- --success: #4caf50;
43
- --warning: #ff9800;
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>;
@@ -1 +1 @@
1
- {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AA+CH,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;AAED,wBAAsB,YAAY,IAAI,OAAO,CAAC,eAAe,CAAC,CAyK7D"}
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: "ui://schema-browser",
149
+ uri: RESOURCE_URIS.schemaBrowser,
104
150
  name: "Schema Browser",
105
151
  description: "Interactive UI for browsing Convex database schemas",
106
- mimeType: "text/html",
152
+ mimeType: MCP_APPS_MIME_TYPE,
107
153
  },
108
154
  {
109
- uri: "ui://dashboard",
155
+ uri: RESOURCE_URIS.dashboard,
110
156
  name: "Realtime Dashboard",
111
157
  description: "Live charts and metrics for Convex data",
112
- mimeType: "text/html",
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
- if (uri.startsWith("ui://schema-browser")) {
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: "text/html",
208
+ mimeType: MCP_APPS_MIME_TYPE,
126
209
  text: await getSchemaResourceContent(),
127
210
  },
128
211
  ],
129
212
  };
130
213
  }
131
- if (uri.startsWith("ui://dashboard")) {
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: "text/html",
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
- // Simple HTTP transport for MCP
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 through server
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
- else if (req.url === "/health") {
170
- res.writeHead(200, { "Content-Type": "application/json" });
171
- res.end(JSON.stringify({ status: "ok" }));
172
- }
173
- else {
174
- res.writeHead(404);
175
- res.end("Not found");
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());
@@ -1 +1 @@
1
- {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
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.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}}