@trishchuk/coolors-mcp 1.0.0

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 (197) hide show
  1. package/.claude/settings.local.json +39 -0
  2. package/.env +2 -0
  3. package/.github/ISSUE_TEMPLATE/bug_report.md +73 -0
  4. package/.github/ISSUE_TEMPLATE/feature_request.md +71 -0
  5. package/.github/pull_request_template.md +97 -0
  6. package/.github/workflows/ci.yml +127 -0
  7. package/.github/workflows/deploy-docs.yml +56 -0
  8. package/.github/workflows/release.yml +99 -0
  9. package/.mcp.json +12 -0
  10. package/.prettierignore +1 -0
  11. package/CLAUDE.md +201 -0
  12. package/DOCUMENTATION.md +274 -0
  13. package/GEMINI.md +54 -0
  14. package/LICENSE +21 -0
  15. package/README.md +401 -0
  16. package/demo/content_based_color.png +0 -0
  17. package/demo/music-player.html +621 -0
  18. package/demo/podcast-player.html +903 -0
  19. package/dist/bin/coolors-mcp.d.ts +1 -0
  20. package/dist/bin/coolors-mcp.js +154 -0
  21. package/dist/bin/coolors-mcp.js.map +1 -0
  22. package/dist/bin/server.d.ts +1 -0
  23. package/dist/bin/server.js +3292 -0
  24. package/dist/bin/server.js.map +1 -0
  25. package/dist/chunk-IQ7NN26V.js +114 -0
  26. package/dist/chunk-IQ7NN26V.js.map +1 -0
  27. package/dist/chunk-P3ARRKLS.js +1214 -0
  28. package/dist/chunk-P3ARRKLS.js.map +1 -0
  29. package/dist/color/index.d.ts +716 -0
  30. package/dist/color/index.js +153 -0
  31. package/dist/color/index.js.map +1 -0
  32. package/dist/coolors-mcp.d.ts +136 -0
  33. package/dist/coolors-mcp.js +7 -0
  34. package/dist/coolors-mcp.js.map +1 -0
  35. package/docs/.vitepress/cache/deps/@braintree_sanitize-url.js +93 -0
  36. package/docs/.vitepress/cache/deps/@braintree_sanitize-url.js.map +7 -0
  37. package/docs/.vitepress/cache/deps/_metadata.json +127 -0
  38. package/docs/.vitepress/cache/deps/chunk-BUSYA2B4.js +9 -0
  39. package/docs/.vitepress/cache/deps/chunk-BUSYA2B4.js.map +7 -0
  40. package/docs/.vitepress/cache/deps/chunk-JD3CXNQ6.js +12683 -0
  41. package/docs/.vitepress/cache/deps/chunk-JD3CXNQ6.js.map +7 -0
  42. package/docs/.vitepress/cache/deps/chunk-SYPOPCWC.js +9719 -0
  43. package/docs/.vitepress/cache/deps/chunk-SYPOPCWC.js.map +7 -0
  44. package/docs/.vitepress/cache/deps/cytoscape-cose-bilkent.js +4710 -0
  45. package/docs/.vitepress/cache/deps/cytoscape-cose-bilkent.js.map +7 -0
  46. package/docs/.vitepress/cache/deps/cytoscape.js +30278 -0
  47. package/docs/.vitepress/cache/deps/cytoscape.js.map +7 -0
  48. package/docs/.vitepress/cache/deps/dayjs.js +285 -0
  49. package/docs/.vitepress/cache/deps/dayjs.js.map +7 -0
  50. package/docs/.vitepress/cache/deps/debug.js +468 -0
  51. package/docs/.vitepress/cache/deps/debug.js.map +7 -0
  52. package/docs/.vitepress/cache/deps/package.json +3 -0
  53. package/docs/.vitepress/cache/deps/prismjs.js +1466 -0
  54. package/docs/.vitepress/cache/deps/prismjs.js.map +7 -0
  55. package/docs/.vitepress/cache/deps/prismjs_components_prism-bash.js +228 -0
  56. package/docs/.vitepress/cache/deps/prismjs_components_prism-bash.js.map +7 -0
  57. package/docs/.vitepress/cache/deps/prismjs_components_prism-javascript.js +142 -0
  58. package/docs/.vitepress/cache/deps/prismjs_components_prism-javascript.js.map +7 -0
  59. package/docs/.vitepress/cache/deps/prismjs_components_prism-json.js +27 -0
  60. package/docs/.vitepress/cache/deps/prismjs_components_prism-json.js.map +7 -0
  61. package/docs/.vitepress/cache/deps/prismjs_components_prism-python.js +65 -0
  62. package/docs/.vitepress/cache/deps/prismjs_components_prism-python.js.map +7 -0
  63. package/docs/.vitepress/cache/deps/prismjs_components_prism-typescript.js +53 -0
  64. package/docs/.vitepress/cache/deps/prismjs_components_prism-typescript.js.map +7 -0
  65. package/docs/.vitepress/cache/deps/prismjs_components_prism-yaml.js +73 -0
  66. package/docs/.vitepress/cache/deps/prismjs_components_prism-yaml.js.map +7 -0
  67. package/docs/.vitepress/cache/deps/vitepress___@vue_devtools-api.js +4507 -0
  68. package/docs/.vitepress/cache/deps/vitepress___@vue_devtools-api.js.map +7 -0
  69. package/docs/.vitepress/cache/deps/vitepress___@vueuse_core.js +584 -0
  70. package/docs/.vitepress/cache/deps/vitepress___@vueuse_core.js.map +7 -0
  71. package/docs/.vitepress/cache/deps/vitepress___@vueuse_integrations_useFocusTrap.js +1146 -0
  72. package/docs/.vitepress/cache/deps/vitepress___@vueuse_integrations_useFocusTrap.js.map +7 -0
  73. package/docs/.vitepress/cache/deps/vitepress___mark__js_src_vanilla__js.js +1667 -0
  74. package/docs/.vitepress/cache/deps/vitepress___mark__js_src_vanilla__js.js.map +7 -0
  75. package/docs/.vitepress/cache/deps/vitepress___minisearch.js +1814 -0
  76. package/docs/.vitepress/cache/deps/vitepress___minisearch.js.map +7 -0
  77. package/docs/.vitepress/cache/deps/vue.js +344 -0
  78. package/docs/.vitepress/cache/deps/vue.js.map +7 -0
  79. package/docs/.vitepress/components/ClientGrid.vue +125 -0
  80. package/docs/.vitepress/components/CodeBlock.vue +231 -0
  81. package/docs/.vitepress/components/ConfigModal.vue +477 -0
  82. package/docs/.vitepress/components/DiagramModal.vue +528 -0
  83. package/docs/.vitepress/components/TroubleshootingModal.vue +472 -0
  84. package/docs/.vitepress/config.js +162 -0
  85. package/docs/.vitepress/theme/FundingLayout.vue +251 -0
  86. package/docs/.vitepress/theme/Layout.vue +134 -0
  87. package/docs/.vitepress/theme/components/AdBanner.vue +317 -0
  88. package/docs/.vitepress/theme/components/AdPlaceholder.vue +78 -0
  89. package/docs/.vitepress/theme/components/FundingEffects.vue +322 -0
  90. package/docs/.vitepress/theme/components/FundingHero.vue +345 -0
  91. package/docs/.vitepress/theme/components/SupportSection.vue +511 -0
  92. package/docs/.vitepress/theme/custom-app.css +339 -0
  93. package/docs/.vitepress/theme/custom.css +699 -0
  94. package/docs/.vitepress/theme/index.js +25 -0
  95. package/docs/README.md +198 -0
  96. package/docs/concepts/accessibility.md +473 -0
  97. package/docs/concepts/color-spaces.md +222 -0
  98. package/docs/concepts/distance-metrics.md +384 -0
  99. package/docs/concepts/hct.md +261 -0
  100. package/docs/concepts/image-analysis.md +396 -0
  101. package/docs/concepts/material-design.md +306 -0
  102. package/docs/concepts/theme-matching.md +399 -0
  103. package/docs/examples/basic-colors.md +490 -0
  104. package/docs/examples/creating-themes.md +898 -0
  105. package/docs/examples/css-refactoring.md +824 -0
  106. package/docs/examples/image-extraction.md +882 -0
  107. package/docs/getting-started.md +366 -0
  108. package/docs/index.md +190 -0
  109. package/docs/installation.md +157 -0
  110. package/docs/tools/README.md +234 -0
  111. package/docs/tools/accessibility.md +614 -0
  112. package/docs/tools/color-operations.md +374 -0
  113. package/docs/tools/image-extraction.md +624 -0
  114. package/docs/tools/material-design.md +347 -0
  115. package/docs/tools/theme-matching.md +552 -0
  116. package/eslint.config.ts +14 -0
  117. package/examples/theme-matching.md +113 -0
  118. package/jsr.json +7 -0
  119. package/mcp-config.json +8 -0
  120. package/note.md +35 -0
  121. package/package.json +122 -0
  122. package/research_results.md +53 -0
  123. package/src/bin/coolors-mcp.ts +194 -0
  124. package/src/bin/server.ts +61 -0
  125. package/src/color/__tests__/conversions-argb.test.ts +198 -0
  126. package/src/color/__tests__/extract-colors.test.ts +360 -0
  127. package/src/color/__tests__/image-utils.test.ts +242 -0
  128. package/src/color/__tests__/reference-colors.test.ts +278 -0
  129. package/src/color/__tests__/round-trip.test.ts +197 -0
  130. package/src/color/conversions.test.ts +402 -0
  131. package/src/color/conversions.ts +393 -0
  132. package/src/color/dislike/__tests__/dislike-analyzer.test.ts +248 -0
  133. package/src/color/dislike/dislike-analyzer.ts +114 -0
  134. package/src/color/extract-colors.ts +228 -0
  135. package/src/color/hct/__tests__/hct-class.test.ts +232 -0
  136. package/src/color/hct/harmonization.ts +204 -0
  137. package/src/color/hct/hct-class.ts +109 -0
  138. package/src/color/hct/hct-solver.ts +168 -0
  139. package/src/color/hct/index.ts +39 -0
  140. package/src/color/hct/tonal-palette.ts +211 -0
  141. package/src/color/hct/types.ts +88 -0
  142. package/src/color/image-utils.ts +79 -0
  143. package/src/color/index.ts +87 -0
  144. package/src/color/material-theme.ts +157 -0
  145. package/src/color/metrics.test.ts +276 -0
  146. package/src/color/metrics.ts +281 -0
  147. package/src/color/quantize/__tests__/quantizer_celebi.test.ts +195 -0
  148. package/src/color/quantize/lab_point_provider.ts +55 -0
  149. package/src/color/quantize/point_provider.ts +27 -0
  150. package/src/color/quantize/quantizer_celebi.ts +51 -0
  151. package/src/color/quantize/quantizer_celebi_test.ts +71 -0
  152. package/src/color/quantize/quantizer_map.ts +47 -0
  153. package/src/color/quantize/quantizer_wsmeans.ts +232 -0
  154. package/src/color/quantize/quantizer_wu.ts +472 -0
  155. package/src/color/score/__tests__/score.test.ts +224 -0
  156. package/src/color/score/score.ts +175 -0
  157. package/src/color/types.ts +151 -0
  158. package/src/color/utils/color_utils.ts +292 -0
  159. package/src/color/utils/math_utils.ts +145 -0
  160. package/src/color/utils.test.ts +403 -0
  161. package/src/color/utils.ts +315 -0
  162. package/src/constants.ts +5 -0
  163. package/src/coolors-mcp.ts +37 -0
  164. package/src/examples/addition.ts +333 -0
  165. package/src/examples/color-demo.ts +125 -0
  166. package/src/examples/custom-logger.ts +201 -0
  167. package/src/examples/oauth-server.ts +113 -0
  168. package/src/examples/session-context.ts +269 -0
  169. package/src/session.ts +116 -0
  170. package/src/theme/__tests__/matcher.test.ts +180 -0
  171. package/src/theme/__tests__/parser.test.ts +148 -0
  172. package/src/theme/__tests__/refactor.test.ts +224 -0
  173. package/src/theme/index.ts +34 -0
  174. package/src/theme/matcher.ts +395 -0
  175. package/src/theme/parser.ts +392 -0
  176. package/src/theme/refactor.ts +360 -0
  177. package/src/theme/types.ts +152 -0
  178. package/src/tools/__tests__/gradient-generator.test.ts +206 -0
  179. package/src/tools/__tests__/palette-with-locks.test.ts +109 -0
  180. package/src/tools/color-conversion.tool.ts +54 -0
  181. package/src/tools/color-distance.tool.ts +41 -0
  182. package/src/tools/colors.ts +31 -0
  183. package/src/tools/contrast-checker.tool.ts +37 -0
  184. package/src/tools/dislike-analyzer.tool.ts +247 -0
  185. package/src/tools/gradient-generator.tool.ts +250 -0
  186. package/src/tools/image-extraction.tools.ts +289 -0
  187. package/src/tools/index.ts +39 -0
  188. package/src/tools/material-theme.tools.ts +250 -0
  189. package/src/tools/palette-generator.tool.ts +135 -0
  190. package/src/tools/palette-with-locks.tool.ts +221 -0
  191. package/src/tools/registry.ts +142 -0
  192. package/src/tools/simple-tools.ts +37 -0
  193. package/src/tools/theme-matching.tools.ts +334 -0
  194. package/src/types.ts +182 -0
  195. package/src/utils.ts +22 -0
  196. package/tsconfig.json +8 -0
  197. package/vitest.config.js +15 -0
@@ -0,0 +1,528 @@
1
+ <template>
2
+ <div class="diagram-wrapper">
3
+ <!-- Always visible diagram -->
4
+ <div
5
+ class="diagram-container"
6
+ @click="openModal"
7
+ :style="containerStyle"
8
+ >
9
+ <div class="diagram-preview">
10
+ <slot />
11
+ </div>
12
+ <div class="zoom-hint">
13
+ <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
14
+ <circle cx="11" cy="11" r="8"/>
15
+ <path d="21 21l-4.35-4.35"/>
16
+ <path d="15 11h-8"/>
17
+ <path d="11 15v-8"/>
18
+ </svg>
19
+ <span>Click to enlarge</span>
20
+ </div>
21
+ </div>
22
+
23
+ <!-- Modal overlay -->
24
+ <div
25
+ v-if="isOpen"
26
+ class="diagram-modal"
27
+ @click="closeModal"
28
+ >
29
+ <div class="modal-content" @click.stop>
30
+ <div class="modal-header">
31
+ <div class="modal-controls">
32
+ <div class="zoom-controls">
33
+ <button @click="zoomOut" class="control-btn" title="Zoom out" :disabled="scale <= 0.1">
34
+ <span class="zoom-symbol">−</span>
35
+ </button>
36
+ <span class="zoom-info">{{ Math.round(scale * 100) }}%</span>
37
+ <button @click="zoomIn" class="control-btn" title="Zoom in" :disabled="scale >= maxZoom">
38
+ <span class="zoom-symbol">+</span>
39
+ </button>
40
+ </div>
41
+ <div class="action-controls">
42
+ <button @click="fitToScreen" class="control-btn" title="Fit to screen">
43
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
44
+ <path d="M8 3H5a2 2 0 0 0-2 2v3m18 0V5a2 2 0 0 0-2-2h-3m0 18h3a2 2 0 0 0 2-2v-3M3 16v3a2 2 0 0 0 2 2h3"/>
45
+ </svg>
46
+ </button>
47
+ <button @click="closeModal" class="close-btn" title="Close">&times;</button>
48
+ </div>
49
+ </div>
50
+ </div>
51
+ <div
52
+ class="diagram-zoom-container"
53
+ ref="zoomContainer"
54
+ @wheel="handleZoom"
55
+ @mousedown="startPan"
56
+ @mousemove="handlePan"
57
+ @mouseup="endPan"
58
+ @touchstart="startPan"
59
+ @touchmove="handlePan"
60
+ @touchend="endPan"
61
+ >
62
+ <div class="diagram-content">
63
+ <div
64
+ class="diagram-transform-wrapper"
65
+ :style="contentStyle"
66
+ ref="modalContent"
67
+ >
68
+ <slot />
69
+ </div>
70
+ </div>
71
+ </div>
72
+ </div>
73
+ </div>
74
+ </div>
75
+ </template>
76
+
77
+ <script setup>
78
+ import { ref, computed, onMounted, onUnmounted } from 'vue'
79
+
80
+ const isOpen = ref(false)
81
+ const scale = ref(1)
82
+ const translateX = ref(0)
83
+ const translateY = ref(0)
84
+ const fitScale = ref(1)
85
+ const isPanning = ref(false)
86
+ const startX = ref(0)
87
+ const startY = ref(0)
88
+ const zoomContainer = ref(null)
89
+
90
+ const maxZoom = computed(() => {
91
+ return Math.max(5, fitScale.value * 5) // At least 5x, or 5x the fit scale
92
+ })
93
+
94
+ const containerStyle = computed(() => ({
95
+ cursor: 'pointer',
96
+ border: '1px solid #ddd',
97
+ borderRadius: '8px',
98
+ padding: '10px',
99
+ margin: '10px 0',
100
+ transition: 'all 0.2s ease',
101
+ ':hover': {
102
+ borderColor: '#999'
103
+ }
104
+ }))
105
+
106
+ const contentStyle = computed(() => ({
107
+ transform: `translate(${translateX.value}px, ${translateY.value}px) scale(${scale.value})`,
108
+ transformOrigin: 'center',
109
+ transition: isPanning.value ? 'none' : 'transform 0.2s ease'
110
+ }))
111
+
112
+ const openModal = () => {
113
+ isOpen.value = true
114
+ document.body.style.overflow = 'hidden'
115
+ // Start by fitting to screen
116
+ setTimeout(() => {
117
+ calculateFitScale()
118
+ fitToScreen()
119
+ }, 100)
120
+ }
121
+
122
+ const closeModal = () => {
123
+ isOpen.value = false
124
+ document.body.style.overflow = ''
125
+ // Reset zoom and pan for next time
126
+ scale.value = 1
127
+ translateX.value = 0
128
+ translateY.value = 0
129
+ }
130
+
131
+
132
+ const zoomIn = () => {
133
+ scale.value = Math.min(maxZoom.value, scale.value + 0.2)
134
+ }
135
+
136
+ const zoomOut = () => {
137
+ scale.value = Math.max(0.1, scale.value - 0.2)
138
+ }
139
+
140
+ const fitToScreen = () => {
141
+ const container = zoomContainer.value
142
+ if (!container) return
143
+
144
+ // Only calculate fit scale once per modal session
145
+ if (fitScale.value === 1) {
146
+ calculateFitScale()
147
+ }
148
+
149
+ // Always use the stored fit scale
150
+ scale.value = fitScale.value
151
+ translateX.value = 0
152
+ translateY.value = 0
153
+ }
154
+
155
+ const calculateFitScale = () => {
156
+ const container = zoomContainer.value
157
+ if (!container) return
158
+
159
+ try {
160
+ const containerRect = container.getBoundingClientRect()
161
+
162
+ // Try multiple selectors to find the diagram
163
+ const selectors = [
164
+ 'svg',
165
+ '.mermaid',
166
+ '.mermaid svg',
167
+ '[data-processed="true"]',
168
+ 'pre[class*="mermaid"]',
169
+ 'div[class*="mermaid"]'
170
+ ]
171
+
172
+ let diagramElement = null
173
+ let diagramRect = null
174
+
175
+ for (const selector of selectors) {
176
+ diagramElement = container.querySelector(selector)
177
+ if (diagramElement) {
178
+ diagramRect = diagramElement.getBoundingClientRect()
179
+ // Make sure we found a valid element with dimensions
180
+ if (diagramRect.width > 0 && diagramRect.height > 0) {
181
+ break
182
+ }
183
+ }
184
+ }
185
+
186
+ if (!diagramElement || !diagramRect || diagramRect.width === 0 || diagramRect.height === 0) {
187
+ // Final fallback: use the container content
188
+ const content = container.querySelector('.diagram-content')
189
+ if (content) {
190
+ diagramRect = content.getBoundingClientRect()
191
+ }
192
+ }
193
+
194
+ if (!diagramRect || diagramRect.width === 0 || diagramRect.height === 0) {
195
+ // Ultimate fallback
196
+ fitScale.value = 1.2
197
+ return
198
+ }
199
+
200
+ // Calculate scale to fit with padding
201
+ const padding = 40
202
+ const availableWidth = containerRect.width - padding
203
+ const availableHeight = containerRect.height - padding
204
+
205
+ const scaleX = availableWidth / diagramRect.width
206
+ const scaleY = availableHeight / diagramRect.height
207
+
208
+ // Use the smaller scale to ensure it fits both dimensions
209
+ const optimalScale = Math.min(scaleX, scaleY)
210
+
211
+ // Apply reasonable bounds
212
+ fitScale.value = Math.max(0.3, Math.min(optimalScale, 4))
213
+
214
+ } catch (error) {
215
+ console.warn('Error calculating fit scale:', error)
216
+ fitScale.value = 1.2
217
+ }
218
+ }
219
+
220
+ const handleZoom = (e) => {
221
+ e.preventDefault()
222
+ const delta = e.deltaY > 0 ? -0.1 : 0.1
223
+ scale.value = Math.max(0.1, Math.min(maxZoom.value, scale.value + delta))
224
+ }
225
+
226
+ const startPan = (e) => {
227
+ isPanning.value = true
228
+ const clientX = e.clientX || e.touches[0].clientX
229
+ const clientY = e.clientY || e.touches[0].clientY
230
+ startX.value = clientX - translateX.value
231
+ startY.value = clientY - translateY.value
232
+ }
233
+
234
+ const handlePan = (e) => {
235
+ if (!isPanning.value) return
236
+ e.preventDefault()
237
+ const clientX = e.clientX || e.touches[0].clientX
238
+ const clientY = e.clientY || e.touches[0].clientY
239
+ translateX.value = clientX - startX.value
240
+ translateY.value = clientY - startY.value
241
+ }
242
+
243
+ const endPan = () => {
244
+ isPanning.value = false
245
+ }
246
+
247
+ const handleKeydown = (e) => {
248
+ if (e.key === 'Escape' && isOpen.value) {
249
+ closeModal()
250
+ }
251
+ }
252
+
253
+ onMounted(() => {
254
+ document.addEventListener('keydown', handleKeydown)
255
+ })
256
+
257
+ onUnmounted(() => {
258
+ document.removeEventListener('keydown', handleKeydown)
259
+ document.body.style.overflow = ''
260
+ })
261
+ </script>
262
+
263
+ <style scoped>
264
+ .diagram-wrapper {
265
+ position: relative;
266
+ }
267
+
268
+ .diagram-container {
269
+ position: relative;
270
+ cursor: pointer;
271
+ border: 1px solid #ddd;
272
+ border-radius: 8px;
273
+ padding: 10px;
274
+ margin: 10px 0;
275
+ transition: all 0.2s ease;
276
+ background: var(--vp-c-bg);
277
+ }
278
+
279
+ .diagram-container:hover {
280
+ border-color: #999;
281
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
282
+ }
283
+
284
+ .diagram-container:hover .zoom-hint {
285
+ opacity: 1;
286
+ }
287
+
288
+ .diagram-preview {
289
+ position: relative;
290
+ overflow: hidden;
291
+ }
292
+
293
+ .diagram-content {
294
+ width: 100%;
295
+ height: 100%;
296
+ user-select: none;
297
+ display: flex;
298
+ align-items: center;
299
+ justify-content: center;
300
+ }
301
+
302
+ .diagram-transform-wrapper {
303
+ transform-origin: center;
304
+ display: flex;
305
+ align-items: center;
306
+ justify-content: center;
307
+ }
308
+
309
+ .zoom-hint {
310
+ position: absolute;
311
+ top: 10px;
312
+ right: 10px;
313
+ background: rgba(0, 0, 0, 0.7);
314
+ color: white;
315
+ padding: 6px 10px;
316
+ border-radius: 20px;
317
+ font-size: 12px;
318
+ align-items: center;
319
+ gap: 5px;
320
+ opacity: 0;
321
+ transition: opacity 0.2s ease;
322
+ pointer-events: none;
323
+ display: none;
324
+ }
325
+
326
+ @media (hover: hover) {
327
+ .zoom-hint {
328
+ display: flex;
329
+ }
330
+ }
331
+
332
+ .diagram-modal {
333
+ position: fixed;
334
+ top: 0;
335
+ left: 0;
336
+ width: 100vw;
337
+ height: 100vh;
338
+ background: rgba(0, 0, 0, 0.9);
339
+ display: flex;
340
+ justify-content: center;
341
+ align-items: center;
342
+ z-index: 9999;
343
+ backdrop-filter: blur(3px);
344
+ }
345
+
346
+ .modal-content {
347
+ position: relative;
348
+ background: var(--vp-c-bg);
349
+ border-radius: 12px;
350
+ width: 95vw;
351
+ height: 95vh;
352
+ max-width: 1200px;
353
+ max-height: 900px;
354
+ overflow: hidden;
355
+ box-shadow: 0 25px 50px rgba(0, 0, 0, 0.5);
356
+ display: flex;
357
+ flex-direction: column;
358
+ }
359
+
360
+ .modal-header {
361
+ padding: 15px 20px;
362
+ border-bottom: 1px solid var(--vp-c-border);
363
+ background: var(--vp-c-bg-alt);
364
+ flex-shrink: 0;
365
+ }
366
+
367
+ .modal-controls {
368
+ display: flex;
369
+ align-items: center;
370
+ justify-content: space-between;
371
+ gap: 15px;
372
+ flex-wrap: wrap;
373
+ }
374
+
375
+ .zoom-controls {
376
+ display: flex;
377
+ align-items: center;
378
+ gap: 10px;
379
+ }
380
+
381
+ .action-controls {
382
+ display: flex;
383
+ align-items: center;
384
+ gap: 10px;
385
+ }
386
+
387
+ .control-btn {
388
+ background: var(--vp-c-bg);
389
+ border: 1px solid var(--vp-c-border);
390
+ border-radius: 6px;
391
+ padding: 8px;
392
+ cursor: pointer;
393
+ display: flex;
394
+ align-items: center;
395
+ justify-content: center;
396
+ transition: all 0.2s ease;
397
+ color: var(--vp-c-text-1);
398
+ }
399
+
400
+ .control-btn:hover:not(:disabled) {
401
+ background: var(--vp-c-bg-soft);
402
+ border-color: var(--vp-c-brand);
403
+ }
404
+
405
+ .control-btn:disabled {
406
+ opacity: 0.5;
407
+ cursor: not-allowed;
408
+ }
409
+
410
+ .zoom-symbol {
411
+ font-size: 18px;
412
+ font-weight: bold;
413
+ line-height: 1;
414
+ }
415
+
416
+ .zoom-info {
417
+ font-size: 14px;
418
+ color: var(--vp-c-text-2);
419
+ font-weight: 500;
420
+ min-width: 50px;
421
+ text-align: center;
422
+ }
423
+
424
+ .close-btn {
425
+ background: none;
426
+ border: none;
427
+ font-size: 24px;
428
+ cursor: pointer;
429
+ color: var(--vp-c-text-2);
430
+ width: 32px;
431
+ height: 32px;
432
+ display: flex;
433
+ align-items: center;
434
+ justify-content: center;
435
+ border-radius: 50%;
436
+ transition: all 0.2s ease;
437
+ }
438
+
439
+ .close-btn:hover {
440
+ background: var(--vp-c-bg-soft);
441
+ color: var(--vp-c-text-1);
442
+ }
443
+
444
+ .diagram-zoom-container {
445
+ flex: 1;
446
+ overflow: hidden;
447
+ cursor: grab;
448
+ position: relative;
449
+ background: var(--vp-c-bg);
450
+ background-image: radial-gradient(circle, var(--vp-c-border) 1px, transparent 1px);
451
+ background-size: 20px 20px;
452
+ background-position: 0 0;
453
+ }
454
+
455
+ .diagram-zoom-container:active {
456
+ cursor: grabbing;
457
+ }
458
+
459
+ .diagram-content {
460
+ width: 100%;
461
+ height: 100%;
462
+ user-select: none;
463
+ display: flex;
464
+ align-items: center;
465
+ justify-content: center;
466
+ }
467
+
468
+ /* Mobile optimizations */
469
+ @media (max-width: 768px) {
470
+ .modal-content {
471
+ width: 100vw;
472
+ height: 100vh;
473
+ max-width: none;
474
+ max-height: none;
475
+ border-radius: 0;
476
+ }
477
+
478
+ .modal-header {
479
+ padding: 10px 15px;
480
+ }
481
+
482
+ .modal-controls {
483
+ justify-content: center;
484
+ flex-wrap: wrap;
485
+ gap: 10px;
486
+ }
487
+
488
+ .zoom-controls {
489
+ order: 1;
490
+ }
491
+
492
+ .action-controls {
493
+ order: 2;
494
+ }
495
+
496
+ .zoom-hint {
497
+ top: 5px;
498
+ right: 5px;
499
+ padding: 4px 8px;
500
+ font-size: 11px;
501
+ }
502
+
503
+ .zoom-hint span {
504
+ display: none;
505
+ }
506
+ }
507
+
508
+ @media (max-width: 480px) {
509
+ .modal-controls {
510
+ gap: 8px;
511
+ }
512
+
513
+ .zoom-controls, .action-controls {
514
+ gap: 8px;
515
+ }
516
+
517
+ .control-btn {
518
+ padding: 10px;
519
+ min-width: 44px;
520
+ min-height: 44px;
521
+ }
522
+
523
+ .close-btn {
524
+ min-width: 44px;
525
+ min-height: 44px;
526
+ }
527
+ }
528
+ </style>