fdb2 1.0.8 → 1.0.10

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.
Files changed (235) hide show
  1. package/.dockerignore +21 -21
  2. package/.editorconfig +11 -11
  3. package/.eslintrc.cjs +14 -14
  4. package/.eslintrc.json +7 -7
  5. package/.prettierrc.js +3 -3
  6. package/.tpl.env +21 -21
  7. package/.vscodeignore +45 -45
  8. package/README.md +312 -312
  9. package/bin/build.sh +28 -28
  10. package/bin/deploy.sh +8 -8
  11. package/bin/dev.sh +10 -10
  12. package/bin/docker/dev-docker-compose.yml +43 -43
  13. package/bin/docker/dev.Dockerfile +24 -24
  14. package/bin/docker/prod-docker-compose.yml +17 -17
  15. package/bin/docker/prod.Dockerfile +29 -29
  16. package/bin/fdb2.js +220 -220
  17. package/dist/package.json +29 -29
  18. package/dist/pnpm-lock.yaml +1042 -354
  19. package/dist/public/explorer.css +1464 -1437
  20. package/dist/public/explorer.js +759 -223
  21. package/dist/public/index.css +1026 -1026
  22. package/dist/public/index.js +15 -9
  23. package/dist/public/layout.css +221 -221
  24. package/dist/public/layout.js +1 -1
  25. package/dist/public/vue.js +8 -2
  26. package/dist/scripts/preinstall.js +112 -112
  27. package/dist/server/index.d.ts.map +1 -1
  28. package/dist/server/index.js +8 -0
  29. package/dist/server/index.js.map +1 -1
  30. package/dist/server/index.ts +680 -671
  31. package/dist/server/model/connection.entity.ts +65 -65
  32. package/dist/server/model/database.entity.ts +245 -245
  33. package/dist/server/service/connection.service.d.ts +6 -1
  34. package/dist/server/service/connection.service.d.ts.map +1 -1
  35. package/dist/server/service/connection.service.js +18 -1
  36. package/dist/server/service/connection.service.js.map +1 -1
  37. package/dist/server/service/connection.service.ts +358 -341
  38. package/dist/server/service/database/base.service.d.ts +27 -0
  39. package/dist/server/service/database/base.service.d.ts.map +1 -1
  40. package/dist/server/service/database/base.service.js +17 -0
  41. package/dist/server/service/database/base.service.js.map +1 -1
  42. package/dist/server/service/database/base.service.ts +406 -367
  43. package/dist/server/service/database/cockroachdb.service.d.ts +16 -0
  44. package/dist/server/service/database/cockroachdb.service.d.ts.map +1 -1
  45. package/dist/server/service/database/cockroachdb.service.js +220 -154
  46. package/dist/server/service/database/cockroachdb.service.js.map +1 -1
  47. package/dist/server/service/database/cockroachdb.service.ts +871 -782
  48. package/dist/server/service/database/database.service.d.ts +4 -0
  49. package/dist/server/service/database/database.service.d.ts.map +1 -1
  50. package/dist/server/service/database/database.service.js +123 -0
  51. package/dist/server/service/database/database.service.js.map +1 -1
  52. package/dist/server/service/database/database.service.ts +775 -638
  53. package/dist/server/service/database/index.ts +6 -6
  54. package/dist/server/service/database/mongodb.service.d.ts +16 -0
  55. package/dist/server/service/database/mongodb.service.d.ts.map +1 -1
  56. package/dist/server/service/database/mongodb.service.js +35 -0
  57. package/dist/server/service/database/mongodb.service.js.map +1 -1
  58. package/dist/server/service/database/mongodb.service.ts +39 -1
  59. package/dist/server/service/database/mssql.service.d.ts +16 -0
  60. package/dist/server/service/database/mssql.service.d.ts.map +1 -1
  61. package/dist/server/service/database/mssql.service.js +168 -96
  62. package/dist/server/service/database/mssql.service.js.map +1 -1
  63. package/dist/server/service/database/mssql.service.ts +931 -840
  64. package/dist/server/service/database/mysql.service.d.ts +16 -0
  65. package/dist/server/service/database/mysql.service.d.ts.map +1 -1
  66. package/dist/server/service/database/mysql.service.js +189 -80
  67. package/dist/server/service/database/mysql.service.js.map +1 -1
  68. package/dist/server/service/database/mysql.service.ts +1025 -890
  69. package/dist/server/service/database/oracle.service.d.ts +16 -0
  70. package/dist/server/service/database/oracle.service.d.ts.map +1 -1
  71. package/dist/server/service/database/oracle.service.js +182 -120
  72. package/dist/server/service/database/oracle.service.js.map +1 -1
  73. package/dist/server/service/database/oracle.service.ts +1035 -959
  74. package/dist/server/service/database/postgres.service.d.ts +16 -0
  75. package/dist/server/service/database/postgres.service.d.ts.map +1 -1
  76. package/dist/server/service/database/postgres.service.js +154 -88
  77. package/dist/server/service/database/postgres.service.js.map +1 -1
  78. package/dist/server/service/database/postgres.service.ts +960 -871
  79. package/dist/server/service/database/sap.service.d.ts +16 -0
  80. package/dist/server/service/database/sap.service.d.ts.map +1 -1
  81. package/dist/server/service/database/sap.service.js +66 -0
  82. package/dist/server/service/database/sap.service.js.map +1 -1
  83. package/dist/server/service/database/sap.service.ts +89 -0
  84. package/dist/server/service/database/sqlite.service.d.ts +17 -1
  85. package/dist/server/service/database/sqlite.service.d.ts.map +1 -1
  86. package/dist/server/service/database/sqlite.service.js +78 -19
  87. package/dist/server/service/database/sqlite.service.js.map +1 -1
  88. package/dist/server/service/database/sqlite.service.ts +787 -708
  89. package/dist/server/service/session.service.ts +158 -158
  90. package/dist/view/index.html +38 -38
  91. package/env.d.ts +1 -1
  92. package/package.json +7 -2
  93. package/packages/vscode/.vscodeignore +44 -44
  94. package/packages/vscode/README.md +62 -62
  95. package/packages/vscode/out/database-services/cockroachdb.service.js +154 -154
  96. package/packages/vscode/out/database-services/mssql.service.js +96 -96
  97. package/packages/vscode/out/database-services/mysql.service.js +80 -80
  98. package/packages/vscode/out/database-services/oracle.service.js +120 -120
  99. package/packages/vscode/out/database-services/postgres.service.js +88 -88
  100. package/packages/vscode/out/database-services/sqlite.service.js +18 -18
  101. package/packages/vscode/out/provider/WebViewProvider.js +32 -32
  102. package/packages/vscode/package.json +142 -142
  103. package/packages/vscode/resources/icon.svg +5 -5
  104. package/packages/vscode/resources/webview/connection.css +41 -41
  105. package/packages/vscode/resources/webview/database.css +163 -163
  106. package/packages/vscode/resources/webview/index.html +9 -9
  107. package/packages/vscode/resources/webview/modules/header.tpl +13 -13
  108. package/packages/vscode/resources/webview/modules/initial_state.tpl +54 -54
  109. package/packages/vscode/resources/webview/query.css +104 -104
  110. package/packages/vscode/src/database-services/base.service.ts +362 -362
  111. package/packages/vscode/src/database-services/cockroachdb.service.ts +659 -659
  112. package/packages/vscode/src/database-services/connection.service.ts +340 -340
  113. package/packages/vscode/src/database-services/database.service.ts +629 -629
  114. package/packages/vscode/src/database-services/index.ts +6 -6
  115. package/packages/vscode/src/database-services/model/connection.entity.ts +65 -65
  116. package/packages/vscode/src/database-services/model/database.entity.ts +245 -245
  117. package/packages/vscode/src/database-services/mssql.service.ts +722 -722
  118. package/packages/vscode/src/database-services/mysql.service.ts +760 -760
  119. package/packages/vscode/src/database-services/oracle.service.ts +831 -831
  120. package/packages/vscode/src/database-services/postgres.service.ts +740 -740
  121. package/packages/vscode/src/database-services/sqlite.service.ts +558 -558
  122. package/packages/vscode/src/extension.ts +76 -76
  123. package/packages/vscode/src/provider/DatabaseTreeProvider.ts +167 -167
  124. package/packages/vscode/src/provider/WebViewProvider.ts +277 -277
  125. package/packages/vscode/src/service/DatabaseServiceBridge.ts +414 -414
  126. package/packages/vscode/src/typings/connection.ts +90 -90
  127. package/packages/vscode/tsconfig.json +21 -21
  128. package/public/index.html +9 -9
  129. package/public/modules/header.tpl +13 -13
  130. package/public/modules/initial_state.tpl +54 -54
  131. package/scripts/preinstall.js +112 -112
  132. package/server/index.ts +680 -671
  133. package/server/model/connection.entity.ts +65 -65
  134. package/server/model/database.entity.ts +245 -245
  135. package/server/service/connection.service.ts +358 -341
  136. package/server/service/database/base.service.ts +406 -367
  137. package/server/service/database/cockroachdb.service.ts +871 -782
  138. package/server/service/database/database.service.ts +775 -638
  139. package/server/service/database/index.ts +6 -6
  140. package/server/service/database/mongodb.service.ts +39 -1
  141. package/server/service/database/mssql.service.ts +931 -840
  142. package/server/service/database/mysql.service.ts +1025 -890
  143. package/server/service/database/oracle.service.ts +1035 -959
  144. package/server/service/database/postgres.service.ts +960 -871
  145. package/server/service/database/sap.service.ts +89 -0
  146. package/server/service/database/sqlite.service.ts +787 -708
  147. package/server/service/session.service.ts +158 -158
  148. package/server/tsconfig.json +20 -20
  149. package/server.js +149 -149
  150. package/server.pid +1 -0
  151. package/src/adapter/ajax.ts +135 -135
  152. package/src/assets/base.css +1 -1
  153. package/src/assets/database.css +949 -949
  154. package/src/assets/images/svg/illustrations/illustration-1.svg +1 -1
  155. package/src/assets/images/svg/illustrations/illustration-2.svg +2 -2
  156. package/src/assets/images/svg/illustrations/illustration-3.svg +50 -50
  157. package/src/assets/images/svg/illustrations/illustration-4.svg +1 -1
  158. package/src/assets/images/svg/illustrations/illustration-5.svg +73 -73
  159. package/src/assets/images/svg/illustrations/illustration-6.svg +89 -89
  160. package/src/assets/images/svg/illustrations/illustration-7.svg +39 -39
  161. package/src/assets/images/svg/separators/curve-2.svg +3 -3
  162. package/src/assets/images/svg/separators/curve.svg +3 -3
  163. package/src/assets/images/svg/separators/line.svg +3 -3
  164. package/src/assets/logo.svg +73 -73
  165. package/src/assets/main.css +1 -1
  166. package/src/base/config.ts +20 -20
  167. package/src/base/detect.ts +134 -134
  168. package/src/base/entity.ts +92 -92
  169. package/src/base/eventBus.ts +36 -36
  170. package/src/components/connection-editor/index.vue +588 -588
  171. package/src/components/dataGrid/index.vue +104 -104
  172. package/src/components/dataGrid/pagination.vue +105 -105
  173. package/src/components/loading/index.vue +42 -42
  174. package/src/components/modal/index.ts +180 -180
  175. package/src/components/modal/index.vue +560 -560
  176. package/src/components/toast/index.ts +43 -43
  177. package/src/components/toast/toast.vue +57 -57
  178. package/src/components/user/name.vue +103 -103
  179. package/src/components/user/selector.vue +416 -416
  180. package/src/domain/SysConfig.ts +74 -74
  181. package/src/platform/App.vue +7 -7
  182. package/src/platform/database/components/connection-detail.vue +1153 -1154
  183. package/src/platform/database/components/data-editor.vue +477 -477
  184. package/src/platform/database/components/database-detail.vue +1173 -1172
  185. package/src/platform/database/components/database-monitor.vue +1085 -1085
  186. package/src/platform/database/components/db-tools.vue +1264 -816
  187. package/src/platform/database/components/query-history.vue +1348 -1348
  188. package/src/platform/database/components/sql-executor.vue +737 -737
  189. package/src/platform/database/components/sql-query-editor.vue +1045 -1045
  190. package/src/platform/database/components/table-detail.vue +1375 -1376
  191. package/src/platform/database/components/table-editor.vue +916 -916
  192. package/src/platform/database/explorer.vue +1839 -1839
  193. package/src/platform/database/index.vue +1192 -1192
  194. package/src/platform/database/layout.vue +366 -366
  195. package/src/platform/database/router.ts +36 -36
  196. package/src/platform/database/styles/common.scss +601 -601
  197. package/src/platform/database/types/common.ts +444 -444
  198. package/src/platform/database/utils/export.ts +231 -231
  199. package/src/platform/database/utils/helpers.ts +436 -436
  200. package/src/platform/index.ts +32 -32
  201. package/src/platform/router.ts +40 -40
  202. package/src/platform/vscode/bridge.ts +121 -121
  203. package/src/platform/vscode/components/ConnectionPanel.vue +272 -272
  204. package/src/platform/vscode/components/DatabasePanel.vue +532 -532
  205. package/src/platform/vscode/components/QueryPanel.vue +371 -371
  206. package/src/platform/vscode/entry/connection.ts +13 -13
  207. package/src/platform/vscode/entry/database.ts +13 -13
  208. package/src/platform/vscode/entry/query.ts +13 -13
  209. package/src/platform/vscode/index.ts +5 -5
  210. package/src/service/base.ts +133 -127
  211. package/src/service/database.ts +505 -495
  212. package/src/service/login.ts +120 -120
  213. package/src/shims-vue.d.ts +6 -6
  214. package/src/stores/connection.ts +266 -266
  215. package/src/stores/session.ts +87 -87
  216. package/src/typings/database-types.ts +412 -412
  217. package/src/typings/database.ts +363 -363
  218. package/src/typings/global.d.ts +58 -58
  219. package/src/typings/pinia.d.ts +7 -7
  220. package/src/utils/clipboard.ts +29 -29
  221. package/src/utils/database-types.ts +242 -242
  222. package/src/utils/modal.ts +123 -123
  223. package/src/utils/request.ts +55 -55
  224. package/src/utils/sleep.ts +3 -3
  225. package/src/utils/toast.ts +73 -73
  226. package/src/utils/util.ts +171 -171
  227. package/src/utils/xlsx.ts +228 -228
  228. package/tsconfig.json +33 -33
  229. package/view/index.html +9 -9
  230. package/view/modules/header.tpl +13 -13
  231. package/view/modules/initial_state.tpl +19 -19
  232. package/vite.config.ts +424 -424
  233. package/vite.config.vscode.ts +47 -47
  234. package/fdb2.server.pid +0 -1
  235. package/server/backups/db_ai_breakout_2026-03-11T08-38-48-677Z.sql +0 -0
@@ -1,532 +1,532 @@
1
- <template>
2
- <div class="database-panel">
3
- <div class="panel-header">
4
- <h1 class="panel-title">数据库管理</h1>
5
- <div class="panel-actions">
6
- <button @click="handleAddConnection" class="btn btn-secondary">添加连接</button>
7
- </div>
8
- </div>
9
-
10
- <div style="margin-bottom: 20px;">
11
- <select v-model="selectedConnectionId" class="connection-selector" @change="handleConnectionChange">
12
- <option value="">选择数据库连接</option>
13
- <option v-for="conn in connections" :key="conn.id" :value="conn.id">
14
- {{ conn.name }} ({{ conn.host }}:{{ conn.port }})
15
- </option>
16
- </select>
17
- </div>
18
-
19
- <div id="contentArea">
20
- <div v-if="loading" class="loading">
21
- <div class="spinner"></div>
22
- <div>加载中...</div>
23
- </div>
24
- <div v-else-if="!selectedConnectionId" class="empty-state">
25
- <div class="empty-icon">🗄️</div>
26
- <div class="empty-title">未选择连接</div>
27
- <div class="empty-text">请从下拉列表中选择一个数据库连接</div>
28
- </div>
29
- <div v-else>
30
- <div class="tabs">
31
- <button
32
- v-for="tab in tabs"
33
- :key="tab.id"
34
- :class="['tab', { active: currentTab === tab.id }]"
35
- @click="currentTab = tab.id"
36
- >
37
- {{ tab.name }}
38
- </button>
39
- </div>
40
-
41
- <div v-show="currentTab === 'overview'" class="tab-content active">
42
- <h3 class="section-title">数据库概览</h3>
43
- <div class="database-grid">
44
- <div
45
- v-for="db in databases"
46
- :key="db.name"
47
- class="database-card"
48
- @click="selectDatabase(db)"
49
- >
50
- <div class="database-header">
51
- <div class="database-icon">🗄️</div>
52
- <div class="database-name">{{ db.name }}</div>
53
- </div>
54
- <div class="database-info">
55
- <span class="database-stat">📊 {{ db.tables }} 表</span>
56
- <span class="database-stat">💾 {{ db.size }}</span>
57
- </div>
58
- </div>
59
- </div>
60
- </div>
61
-
62
- <div v-show="currentTab === 'databases'" class="tab-content">
63
- <h3 class="section-title">数据库列表</h3>
64
- <div class="database-grid">
65
- <div
66
- v-for="db in databases"
67
- :key="db.name"
68
- class="database-card"
69
- @click="selectDatabase(db)"
70
- >
71
- <div class="database-header">
72
- <div class="database-icon">🗄️</div>
73
- <div class="database-name">{{ db.name }}</div>
74
- </div>
75
- <div class="database-info">
76
- <span class="database-stat">📊 {{ db.tables }} 表</span>
77
- <span class="database-stat">💾 {{ db.size }}</span>
78
- </div>
79
- </div>
80
- </div>
81
- </div>
82
-
83
- <div v-show="currentTab === 'tables'" class="tab-content">
84
- <h3 class="section-title">表列表</h3>
85
- <div class="table-list">
86
- <div
87
- v-for="table in tables"
88
- :key="table.name"
89
- class="table-item"
90
- >
91
- <div class="table-icon">{{ table.type === 'table' ? '📋' : '👁️' }}</div>
92
- <div class="table-info">
93
- <div class="table-name">{{ table.name }}</div>
94
- <div v-if="table.comment" class="table-comment">{{ table.comment }}</div>
95
- <div v-if="table.rows !== null" class="table-meta">
96
- <span>{{ table.rows.toLocaleString() }} 行</span>
97
- <span>{{ table.engine || '' }}</span>
98
- <span>{{ table.size || '' }}</span>
99
- </div>
100
- </div>
101
- <span :class="['table-type', table.type]">{{ table.type === 'table' ? 'TABLE' : 'VIEW' }}</span>
102
- <div class="table-actions">
103
- <button class="action-btn" @click.stop="handleViewData(table)">数据</button>
104
- <button class="action-btn" @click.stop="handleViewStructure(table)">结构</button>
105
- </div>
106
- </div>
107
- </div>
108
- </div>
109
- </div>
110
- </div>
111
- </div>
112
- </template>
113
-
114
- <script setup lang="ts">
115
- import { ref, reactive, computed, onMounted } from 'vue';
116
- import { getVSCodeBridge } from '../bridge';
117
-
118
- const vscode = getVSCodeBridge();
119
-
120
- const connections = ref<any[]>([]);
121
- const selectedConnectionId = ref('');
122
- const databases = ref<any[]>([]);
123
- const tables = ref<any[]>([]);
124
- const loading = ref(false);
125
- const currentTab = ref('overview');
126
-
127
- const selectedConnection = computed(() => {
128
- return connections.value.find(c => c.id === selectedConnectionId.value);
129
- });
130
-
131
- const tabs = [
132
- { id: 'overview', name: '概览' },
133
- { id: 'databases', name: '数据库' },
134
- { id: 'tables', name: '表' }
135
- ];
136
-
137
- onMounted(() => {
138
- // 请求连接列表
139
- vscode.postMessage({ command: 'getConnections' });
140
-
141
- // 监听来自扩展的消息
142
- vscode.onMessage((message: any) => {
143
- if (message.command === 'connections') {
144
- connections.value = message.data;
145
- } else if (message.command === 'selectConnection') {
146
- selectConnection(message.data);
147
- } else if (message.command === 'databases') {
148
- databases.value = message.data;
149
- loading.value = false;
150
- } else if (message.command === 'tables') {
151
- tables.value = message.data;
152
- loading.value = false;
153
- } else if (message.command === 'error') {
154
- handleError(message.data);
155
- }
156
- });
157
- });
158
-
159
- function handleAddConnection() {
160
- vscode.postMessage({ command: 'addConnection' });
161
- }
162
-
163
- function handleConnectionChange() {
164
- if (selectedConnectionId.value) {
165
- const connection = connections.value.find(c => c.id === selectedConnectionId.value);
166
- selectConnection(connection);
167
- } else {
168
- databases.value = [];
169
- tables.value = [];
170
- }
171
- }
172
-
173
- function selectConnection(connection: any) {
174
- if (!connection) return;
175
-
176
- selectedConnectionId.value = connection.id;
177
- loading.value = true;
178
-
179
- // 请求数据库列表
180
- vscode.postMessage({
181
- command: 'getDatabases',
182
- data: { connectionId: connection.id }
183
- });
184
- }
185
-
186
- function selectDatabase(database: any) {
187
- loading.value = true;
188
- currentTab.value = 'tables';
189
-
190
- // 请求表列表
191
- vscode.postMessage({
192
- command: 'getTables',
193
- data: {
194
- connectionId: selectedConnectionId.value,
195
- databaseName: database.name
196
- }
197
- });
198
- }
199
-
200
- function handleViewData(table: any) {
201
- vscode.postMessage({
202
- command: 'openTableData',
203
- data: {
204
- connectionId: selectedConnectionId.value,
205
- tableName: table.name
206
- }
207
- });
208
- }
209
-
210
- function handleViewStructure(table: any) {
211
- vscode.postMessage({
212
- command: 'openTableStructure',
213
- data: {
214
- connectionId: selectedConnectionId.value,
215
- tableName: table.name
216
- }
217
- });
218
- }
219
-
220
- function handleError(error: string) {
221
- loading.value = false;
222
- console.error('Error:', error);
223
- alert(`操作失败: ${error}`);
224
- }
225
- </script>
226
-
227
- <style scoped>
228
- .database-panel {
229
- padding: 16px;
230
- height: 100vh;
231
- overflow: auto;
232
- }
233
-
234
- .panel-header {
235
- display: flex;
236
- justify-content: space-between;
237
- align-items: center;
238
- margin-bottom: 20px;
239
- padding-bottom: 16px;
240
- border-bottom: 1px solid var(--vscode-panel-border);
241
- }
242
-
243
- .panel-title {
244
- font-size: 18px;
245
- font-weight: 600;
246
- margin: 0;
247
- color: var(--vscode-foreground);
248
- }
249
-
250
- .panel-actions {
251
- display: flex;
252
- gap: 8px;
253
- }
254
-
255
- .connection-selector {
256
- width: 300px;
257
- padding: 6px 10px;
258
- border: 1px solid var(--vscode-input-border);
259
- background-color: var(--vscode-input-background);
260
- color: var(--vscode-input-foreground);
261
- border-radius: 4px;
262
- font-size: 13px;
263
- }
264
-
265
- .btn {
266
- padding: 6px 14px;
267
- border: none;
268
- border-radius: 4px;
269
- font-size: 13px;
270
- cursor: pointer;
271
- transition: background-color 0.2s;
272
- }
273
-
274
- .btn-primary {
275
- background-color: var(--vscode-button-background);
276
- color: var(--vscode-button-foreground);
277
- }
278
-
279
- .btn-primary:hover {
280
- background-color: var(--vscode-button-hoverBackground);
281
- }
282
-
283
- .btn-secondary {
284
- background-color: var(--vscode-button-secondaryBackground);
285
- color: var(--vscode-button-secondaryForeground);
286
- }
287
-
288
- .btn-secondary:hover {
289
- background-color: var(--vscode-button-secondaryHoverBackground);
290
- }
291
-
292
- .database-grid {
293
- display: grid;
294
- grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
295
- gap: 16px;
296
- }
297
-
298
- .database-card {
299
- padding: 16px;
300
- background-color: var(--vscode-editor-background);
301
- border: 1px solid var(--vscode-panel-border);
302
- border-radius: 8px;
303
- cursor: pointer;
304
- transition: all 0.2s;
305
- }
306
-
307
- .database-card:hover {
308
- border-color: var(--vscode-focusBorder);
309
- box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
310
- transform: translateY(-2px);
311
- }
312
-
313
- .database-header {
314
- display: flex;
315
- align-items: center;
316
- gap: 12px;
317
- margin-bottom: 12px;
318
- }
319
-
320
- .database-icon {
321
- font-size: 28px;
322
- color: var(--vscode-textLink-foreground);
323
- }
324
-
325
- .database-name {
326
- flex: 1;
327
- font-size: 16px;
328
- font-weight: 600;
329
- color: var(--vscode-foreground);
330
- }
331
-
332
- .database-info {
333
- display: flex;
334
- gap: 16px;
335
- font-size: 12px;
336
- color: var(--vscode-descriptionForeground);
337
- }
338
-
339
- .database-stat {
340
- display: flex;
341
- align-items: center;
342
- gap: 4px;
343
- }
344
-
345
- .section-title {
346
- font-size: 16px;
347
- font-weight: 600;
348
- margin: 24px 0 16px 0;
349
- padding-bottom: 8px;
350
- border-bottom: 1px solid var(--vscode-panel-border);
351
- color: var(--vscode-foreground);
352
- }
353
-
354
- .tabs {
355
- display: flex;
356
- gap: 4px;
357
- margin-bottom: 16px;
358
- border-bottom: 1px solid var(--vscode-panel-border);
359
- }
360
-
361
- .tab {
362
- padding: 8px 16px;
363
- border: none;
364
- background-color: transparent;
365
- color: var(--vscode-foreground);
366
- font-size: 13px;
367
- cursor: pointer;
368
- border-bottom: 2px solid transparent;
369
- transition: all 0.2s;
370
- }
371
-
372
- .tab:hover {
373
- background-color: var(--vscode-list-hoverBackground);
374
- }
375
-
376
- .tab.active {
377
- border-bottom-color: var(--vscode-focusBorder);
378
- color: var(--vscode-textLink-foreground);
379
- }
380
-
381
- .tab-content {
382
- display: none;
383
- }
384
-
385
- .tab-content.active {
386
- display: block;
387
- }
388
-
389
- .table-list {
390
- display: flex;
391
- flex-direction: column;
392
- gap: 4px;
393
- }
394
-
395
- .table-item {
396
- display: flex;
397
- align-items: center;
398
- gap: 12px;
399
- padding: 12px;
400
- border-radius: 6px;
401
- cursor: pointer;
402
- transition: background-color 0.2s;
403
- }
404
-
405
- .table-item:hover {
406
- background-color: var(--vscode-list-hoverBackground);
407
- }
408
-
409
- .table-icon {
410
- font-size: 20px;
411
- color: var(--vscode-textLink-foreground);
412
- width: 24px;
413
- text-align: center;
414
- }
415
-
416
- .table-info {
417
- flex: 1;
418
- }
419
-
420
- .table-name {
421
- font-size: 14px;
422
- font-weight: 500;
423
- margin-bottom: 2px;
424
- color: var(--vscode-foreground);
425
- }
426
-
427
- .table-comment {
428
- font-size: 12px;
429
- color: var(--vscode-descriptionForeground);
430
- }
431
-
432
- .table-meta {
433
- display: flex;
434
- align-items: center;
435
- gap: 12px;
436
- font-size: 12px;
437
- color: var(--vscode-descriptionForeground);
438
- margin-top: 4px;
439
- }
440
-
441
- .table-type {
442
- padding: 2px 8px;
443
- border-radius: 12px;
444
- background-color: var(--vscode-badge-background);
445
- font-weight: 500;
446
- font-size: 12px;
447
- }
448
-
449
- .table-type.table {
450
- background-color: var(--vscode-badge-background);
451
- }
452
-
453
- .table-type.view {
454
- background-color: var(--vscode-gitDecorationModifiedResourceForeground);
455
- color: white;
456
- }
457
-
458
- .table-actions {
459
- display: flex;
460
- gap: 8px;
461
- opacity: 0;
462
- transition: opacity 0.2s;
463
- }
464
-
465
- .table-item:hover .table-actions {
466
- opacity: 1;
467
- }
468
-
469
- .action-btn {
470
- padding: 4px 8px;
471
- font-size: 12px;
472
- background-color: var(--vscode-button-secondaryBackground);
473
- color: var(--vscode-button-secondaryForeground);
474
- border: none;
475
- border-radius: 4px;
476
- cursor: pointer;
477
- }
478
-
479
- .action-btn:hover {
480
- background-color: var(--vscode-button-secondaryHoverBackground);
481
- }
482
-
483
- .empty-state {
484
- display: flex;
485
- flex-direction: column;
486
- align-items: center;
487
- justify-content: center;
488
- padding: 80px 20px;
489
- text-align: center;
490
- }
491
-
492
- .empty-icon {
493
- font-size: 64px;
494
- margin-bottom: 16px;
495
- opacity: 0.3;
496
- }
497
-
498
- .empty-title {
499
- font-size: 16px;
500
- font-weight: 600;
501
- margin-bottom: 8px;
502
- color: var(--vscode-foreground);
503
- }
504
-
505
- .empty-text {
506
- font-size: 14px;
507
- color: var(--vscode-descriptionForeground);
508
- }
509
-
510
- .loading {
511
- display: flex;
512
- flex-direction: column;
513
- align-items: center;
514
- justify-content: center;
515
- padding: 40px;
516
- }
517
-
518
- .spinner {
519
- width: 32px;
520
- height: 32px;
521
- border: 3px solid var(--vscode-progressBar-background);
522
- border-top-color: var(--vscode-progressBar-foreground);
523
- border-radius: 50%;
524
- animation: spin 0.6s linear infinite;
525
- }
526
-
527
- @keyframes spin {
528
- to {
529
- transform: rotate(360deg);
530
- }
531
- }
532
- </style>
1
+ <template>
2
+ <div class="database-panel">
3
+ <div class="panel-header">
4
+ <h1 class="panel-title">数据库管理</h1>
5
+ <div class="panel-actions">
6
+ <button @click="handleAddConnection" class="btn btn-secondary">添加连接</button>
7
+ </div>
8
+ </div>
9
+
10
+ <div style="margin-bottom: 20px;">
11
+ <select v-model="selectedConnectionId" class="connection-selector" @change="handleConnectionChange">
12
+ <option value="">选择数据库连接</option>
13
+ <option v-for="conn in connections" :key="conn.id" :value="conn.id">
14
+ {{ conn.name }} ({{ conn.host }}:{{ conn.port }})
15
+ </option>
16
+ </select>
17
+ </div>
18
+
19
+ <div id="contentArea">
20
+ <div v-if="loading" class="loading">
21
+ <div class="spinner"></div>
22
+ <div>加载中...</div>
23
+ </div>
24
+ <div v-else-if="!selectedConnectionId" class="empty-state">
25
+ <div class="empty-icon">🗄️</div>
26
+ <div class="empty-title">未选择连接</div>
27
+ <div class="empty-text">请从下拉列表中选择一个数据库连接</div>
28
+ </div>
29
+ <div v-else>
30
+ <div class="tabs">
31
+ <button
32
+ v-for="tab in tabs"
33
+ :key="tab.id"
34
+ :class="['tab', { active: currentTab === tab.id }]"
35
+ @click="currentTab = tab.id"
36
+ >
37
+ {{ tab.name }}
38
+ </button>
39
+ </div>
40
+
41
+ <div v-show="currentTab === 'overview'" class="tab-content active">
42
+ <h3 class="section-title">数据库概览</h3>
43
+ <div class="database-grid">
44
+ <div
45
+ v-for="db in databases"
46
+ :key="db.name"
47
+ class="database-card"
48
+ @click="selectDatabase(db)"
49
+ >
50
+ <div class="database-header">
51
+ <div class="database-icon">🗄️</div>
52
+ <div class="database-name">{{ db.name }}</div>
53
+ </div>
54
+ <div class="database-info">
55
+ <span class="database-stat">📊 {{ db.tables }} 表</span>
56
+ <span class="database-stat">💾 {{ db.size }}</span>
57
+ </div>
58
+ </div>
59
+ </div>
60
+ </div>
61
+
62
+ <div v-show="currentTab === 'databases'" class="tab-content">
63
+ <h3 class="section-title">数据库列表</h3>
64
+ <div class="database-grid">
65
+ <div
66
+ v-for="db in databases"
67
+ :key="db.name"
68
+ class="database-card"
69
+ @click="selectDatabase(db)"
70
+ >
71
+ <div class="database-header">
72
+ <div class="database-icon">🗄️</div>
73
+ <div class="database-name">{{ db.name }}</div>
74
+ </div>
75
+ <div class="database-info">
76
+ <span class="database-stat">📊 {{ db.tables }} 表</span>
77
+ <span class="database-stat">💾 {{ db.size }}</span>
78
+ </div>
79
+ </div>
80
+ </div>
81
+ </div>
82
+
83
+ <div v-show="currentTab === 'tables'" class="tab-content">
84
+ <h3 class="section-title">表列表</h3>
85
+ <div class="table-list">
86
+ <div
87
+ v-for="table in tables"
88
+ :key="table.name"
89
+ class="table-item"
90
+ >
91
+ <div class="table-icon">{{ table.type === 'table' ? '📋' : '👁️' }}</div>
92
+ <div class="table-info">
93
+ <div class="table-name">{{ table.name }}</div>
94
+ <div v-if="table.comment" class="table-comment">{{ table.comment }}</div>
95
+ <div v-if="table.rows !== null" class="table-meta">
96
+ <span>{{ table.rows.toLocaleString() }} 行</span>
97
+ <span>{{ table.engine || '' }}</span>
98
+ <span>{{ table.size || '' }}</span>
99
+ </div>
100
+ </div>
101
+ <span :class="['table-type', table.type]">{{ table.type === 'table' ? 'TABLE' : 'VIEW' }}</span>
102
+ <div class="table-actions">
103
+ <button class="action-btn" @click.stop="handleViewData(table)">数据</button>
104
+ <button class="action-btn" @click.stop="handleViewStructure(table)">结构</button>
105
+ </div>
106
+ </div>
107
+ </div>
108
+ </div>
109
+ </div>
110
+ </div>
111
+ </div>
112
+ </template>
113
+
114
+ <script setup lang="ts">
115
+ import { ref, reactive, computed, onMounted } from 'vue';
116
+ import { getVSCodeBridge } from '../bridge';
117
+
118
+ const vscode = getVSCodeBridge();
119
+
120
+ const connections = ref<any[]>([]);
121
+ const selectedConnectionId = ref('');
122
+ const databases = ref<any[]>([]);
123
+ const tables = ref<any[]>([]);
124
+ const loading = ref(false);
125
+ const currentTab = ref('overview');
126
+
127
+ const selectedConnection = computed(() => {
128
+ return connections.value.find(c => c.id === selectedConnectionId.value);
129
+ });
130
+
131
+ const tabs = [
132
+ { id: 'overview', name: '概览' },
133
+ { id: 'databases', name: '数据库' },
134
+ { id: 'tables', name: '表' }
135
+ ];
136
+
137
+ onMounted(() => {
138
+ // 请求连接列表
139
+ vscode.postMessage({ command: 'getConnections' });
140
+
141
+ // 监听来自扩展的消息
142
+ vscode.onMessage((message: any) => {
143
+ if (message.command === 'connections') {
144
+ connections.value = message.data;
145
+ } else if (message.command === 'selectConnection') {
146
+ selectConnection(message.data);
147
+ } else if (message.command === 'databases') {
148
+ databases.value = message.data;
149
+ loading.value = false;
150
+ } else if (message.command === 'tables') {
151
+ tables.value = message.data;
152
+ loading.value = false;
153
+ } else if (message.command === 'error') {
154
+ handleError(message.data);
155
+ }
156
+ });
157
+ });
158
+
159
+ function handleAddConnection() {
160
+ vscode.postMessage({ command: 'addConnection' });
161
+ }
162
+
163
+ function handleConnectionChange() {
164
+ if (selectedConnectionId.value) {
165
+ const connection = connections.value.find(c => c.id === selectedConnectionId.value);
166
+ selectConnection(connection);
167
+ } else {
168
+ databases.value = [];
169
+ tables.value = [];
170
+ }
171
+ }
172
+
173
+ function selectConnection(connection: any) {
174
+ if (!connection) return;
175
+
176
+ selectedConnectionId.value = connection.id;
177
+ loading.value = true;
178
+
179
+ // 请求数据库列表
180
+ vscode.postMessage({
181
+ command: 'getDatabases',
182
+ data: { connectionId: connection.id }
183
+ });
184
+ }
185
+
186
+ function selectDatabase(database: any) {
187
+ loading.value = true;
188
+ currentTab.value = 'tables';
189
+
190
+ // 请求表列表
191
+ vscode.postMessage({
192
+ command: 'getTables',
193
+ data: {
194
+ connectionId: selectedConnectionId.value,
195
+ databaseName: database.name
196
+ }
197
+ });
198
+ }
199
+
200
+ function handleViewData(table: any) {
201
+ vscode.postMessage({
202
+ command: 'openTableData',
203
+ data: {
204
+ connectionId: selectedConnectionId.value,
205
+ tableName: table.name
206
+ }
207
+ });
208
+ }
209
+
210
+ function handleViewStructure(table: any) {
211
+ vscode.postMessage({
212
+ command: 'openTableStructure',
213
+ data: {
214
+ connectionId: selectedConnectionId.value,
215
+ tableName: table.name
216
+ }
217
+ });
218
+ }
219
+
220
+ function handleError(error: string) {
221
+ loading.value = false;
222
+ console.error('Error:', error);
223
+ alert(`操作失败: ${error}`);
224
+ }
225
+ </script>
226
+
227
+ <style scoped>
228
+ .database-panel {
229
+ padding: 16px;
230
+ height: 100vh;
231
+ overflow: auto;
232
+ }
233
+
234
+ .panel-header {
235
+ display: flex;
236
+ justify-content: space-between;
237
+ align-items: center;
238
+ margin-bottom: 20px;
239
+ padding-bottom: 16px;
240
+ border-bottom: 1px solid var(--vscode-panel-border);
241
+ }
242
+
243
+ .panel-title {
244
+ font-size: 18px;
245
+ font-weight: 600;
246
+ margin: 0;
247
+ color: var(--vscode-foreground);
248
+ }
249
+
250
+ .panel-actions {
251
+ display: flex;
252
+ gap: 8px;
253
+ }
254
+
255
+ .connection-selector {
256
+ width: 300px;
257
+ padding: 6px 10px;
258
+ border: 1px solid var(--vscode-input-border);
259
+ background-color: var(--vscode-input-background);
260
+ color: var(--vscode-input-foreground);
261
+ border-radius: 4px;
262
+ font-size: 13px;
263
+ }
264
+
265
+ .btn {
266
+ padding: 6px 14px;
267
+ border: none;
268
+ border-radius: 4px;
269
+ font-size: 13px;
270
+ cursor: pointer;
271
+ transition: background-color 0.2s;
272
+ }
273
+
274
+ .btn-primary {
275
+ background-color: var(--vscode-button-background);
276
+ color: var(--vscode-button-foreground);
277
+ }
278
+
279
+ .btn-primary:hover {
280
+ background-color: var(--vscode-button-hoverBackground);
281
+ }
282
+
283
+ .btn-secondary {
284
+ background-color: var(--vscode-button-secondaryBackground);
285
+ color: var(--vscode-button-secondaryForeground);
286
+ }
287
+
288
+ .btn-secondary:hover {
289
+ background-color: var(--vscode-button-secondaryHoverBackground);
290
+ }
291
+
292
+ .database-grid {
293
+ display: grid;
294
+ grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
295
+ gap: 16px;
296
+ }
297
+
298
+ .database-card {
299
+ padding: 16px;
300
+ background-color: var(--vscode-editor-background);
301
+ border: 1px solid var(--vscode-panel-border);
302
+ border-radius: 8px;
303
+ cursor: pointer;
304
+ transition: all 0.2s;
305
+ }
306
+
307
+ .database-card:hover {
308
+ border-color: var(--vscode-focusBorder);
309
+ box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
310
+ transform: translateY(-2px);
311
+ }
312
+
313
+ .database-header {
314
+ display: flex;
315
+ align-items: center;
316
+ gap: 12px;
317
+ margin-bottom: 12px;
318
+ }
319
+
320
+ .database-icon {
321
+ font-size: 28px;
322
+ color: var(--vscode-textLink-foreground);
323
+ }
324
+
325
+ .database-name {
326
+ flex: 1;
327
+ font-size: 16px;
328
+ font-weight: 600;
329
+ color: var(--vscode-foreground);
330
+ }
331
+
332
+ .database-info {
333
+ display: flex;
334
+ gap: 16px;
335
+ font-size: 12px;
336
+ color: var(--vscode-descriptionForeground);
337
+ }
338
+
339
+ .database-stat {
340
+ display: flex;
341
+ align-items: center;
342
+ gap: 4px;
343
+ }
344
+
345
+ .section-title {
346
+ font-size: 16px;
347
+ font-weight: 600;
348
+ margin: 24px 0 16px 0;
349
+ padding-bottom: 8px;
350
+ border-bottom: 1px solid var(--vscode-panel-border);
351
+ color: var(--vscode-foreground);
352
+ }
353
+
354
+ .tabs {
355
+ display: flex;
356
+ gap: 4px;
357
+ margin-bottom: 16px;
358
+ border-bottom: 1px solid var(--vscode-panel-border);
359
+ }
360
+
361
+ .tab {
362
+ padding: 8px 16px;
363
+ border: none;
364
+ background-color: transparent;
365
+ color: var(--vscode-foreground);
366
+ font-size: 13px;
367
+ cursor: pointer;
368
+ border-bottom: 2px solid transparent;
369
+ transition: all 0.2s;
370
+ }
371
+
372
+ .tab:hover {
373
+ background-color: var(--vscode-list-hoverBackground);
374
+ }
375
+
376
+ .tab.active {
377
+ border-bottom-color: var(--vscode-focusBorder);
378
+ color: var(--vscode-textLink-foreground);
379
+ }
380
+
381
+ .tab-content {
382
+ display: none;
383
+ }
384
+
385
+ .tab-content.active {
386
+ display: block;
387
+ }
388
+
389
+ .table-list {
390
+ display: flex;
391
+ flex-direction: column;
392
+ gap: 4px;
393
+ }
394
+
395
+ .table-item {
396
+ display: flex;
397
+ align-items: center;
398
+ gap: 12px;
399
+ padding: 12px;
400
+ border-radius: 6px;
401
+ cursor: pointer;
402
+ transition: background-color 0.2s;
403
+ }
404
+
405
+ .table-item:hover {
406
+ background-color: var(--vscode-list-hoverBackground);
407
+ }
408
+
409
+ .table-icon {
410
+ font-size: 20px;
411
+ color: var(--vscode-textLink-foreground);
412
+ width: 24px;
413
+ text-align: center;
414
+ }
415
+
416
+ .table-info {
417
+ flex: 1;
418
+ }
419
+
420
+ .table-name {
421
+ font-size: 14px;
422
+ font-weight: 500;
423
+ margin-bottom: 2px;
424
+ color: var(--vscode-foreground);
425
+ }
426
+
427
+ .table-comment {
428
+ font-size: 12px;
429
+ color: var(--vscode-descriptionForeground);
430
+ }
431
+
432
+ .table-meta {
433
+ display: flex;
434
+ align-items: center;
435
+ gap: 12px;
436
+ font-size: 12px;
437
+ color: var(--vscode-descriptionForeground);
438
+ margin-top: 4px;
439
+ }
440
+
441
+ .table-type {
442
+ padding: 2px 8px;
443
+ border-radius: 12px;
444
+ background-color: var(--vscode-badge-background);
445
+ font-weight: 500;
446
+ font-size: 12px;
447
+ }
448
+
449
+ .table-type.table {
450
+ background-color: var(--vscode-badge-background);
451
+ }
452
+
453
+ .table-type.view {
454
+ background-color: var(--vscode-gitDecorationModifiedResourceForeground);
455
+ color: white;
456
+ }
457
+
458
+ .table-actions {
459
+ display: flex;
460
+ gap: 8px;
461
+ opacity: 0;
462
+ transition: opacity 0.2s;
463
+ }
464
+
465
+ .table-item:hover .table-actions {
466
+ opacity: 1;
467
+ }
468
+
469
+ .action-btn {
470
+ padding: 4px 8px;
471
+ font-size: 12px;
472
+ background-color: var(--vscode-button-secondaryBackground);
473
+ color: var(--vscode-button-secondaryForeground);
474
+ border: none;
475
+ border-radius: 4px;
476
+ cursor: pointer;
477
+ }
478
+
479
+ .action-btn:hover {
480
+ background-color: var(--vscode-button-secondaryHoverBackground);
481
+ }
482
+
483
+ .empty-state {
484
+ display: flex;
485
+ flex-direction: column;
486
+ align-items: center;
487
+ justify-content: center;
488
+ padding: 80px 20px;
489
+ text-align: center;
490
+ }
491
+
492
+ .empty-icon {
493
+ font-size: 64px;
494
+ margin-bottom: 16px;
495
+ opacity: 0.3;
496
+ }
497
+
498
+ .empty-title {
499
+ font-size: 16px;
500
+ font-weight: 600;
501
+ margin-bottom: 8px;
502
+ color: var(--vscode-foreground);
503
+ }
504
+
505
+ .empty-text {
506
+ font-size: 14px;
507
+ color: var(--vscode-descriptionForeground);
508
+ }
509
+
510
+ .loading {
511
+ display: flex;
512
+ flex-direction: column;
513
+ align-items: center;
514
+ justify-content: center;
515
+ padding: 40px;
516
+ }
517
+
518
+ .spinner {
519
+ width: 32px;
520
+ height: 32px;
521
+ border: 3px solid var(--vscode-progressBar-background);
522
+ border-top-color: var(--vscode-progressBar-foreground);
523
+ border-radius: 50%;
524
+ animation: spin 0.6s linear infinite;
525
+ }
526
+
527
+ @keyframes spin {
528
+ to {
529
+ transform: rotate(360deg);
530
+ }
531
+ }
532
+ </style>