@sarjallab09/figma-intelligence 1.0.1 → 1.2.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 (297) hide show
  1. package/README.md +67 -36
  2. package/dist/bin/cli.js +2 -0
  3. package/dist/design-bridge/bridge.js +2 -0
  4. package/dist/figma-bridge-plugin/bridge-relay.js +2 -0
  5. package/dist/figma-bridge-plugin/code.js +1 -0
  6. package/{figma-bridge-plugin → dist/figma-bridge-plugin}/package-lock.json +0 -3
  7. package/dist/figma-bridge-plugin/ui.html +4970 -0
  8. package/dist/figma-intelligence-layer/dist/index.js +2 -0
  9. package/dist/scripts/clean-existing-chunks.js +2 -0
  10. package/dist/scripts/connect-ai-tool.js +2 -0
  11. package/dist/scripts/convert-hub-pdfs.js +2 -0
  12. package/dist/scripts/figma-mcp-status.js +2 -0
  13. package/dist/scripts/register-codex-mcp.js +2 -0
  14. package/dist/scripts/test-copilot-chat.js +2 -0
  15. package/package.json +11 -8
  16. package/bin/cli.js +0 -859
  17. package/design-bridge/bridge.js +0 -196
  18. package/design-bridge/lib/assets.js +0 -367
  19. package/design-bridge/lib/prompt.js +0 -85
  20. package/design-bridge/lib/server.js +0 -66
  21. package/design-bridge/lib/stitch.js +0 -37
  22. package/design-bridge/lib/tokens.js +0 -82
  23. package/design-bridge/package-lock.json +0 -579
  24. package/figma-bridge-plugin/README.md +0 -97
  25. package/figma-bridge-plugin/anthropic-chat-runner.js +0 -192
  26. package/figma-bridge-plugin/bridge-relay.js +0 -2363
  27. package/figma-bridge-plugin/chat-runner.js +0 -459
  28. package/figma-bridge-plugin/code.js +0 -1528
  29. package/figma-bridge-plugin/codex-runner.js +0 -505
  30. package/figma-bridge-plugin/component-schemas.js +0 -110
  31. package/figma-bridge-plugin/content-context.js +0 -869
  32. package/figma-bridge-plugin/create-button.js +0 -216
  33. package/figma-bridge-plugin/gemini-cli-runner.js +0 -291
  34. package/figma-bridge-plugin/gemini-runner.js +0 -187
  35. package/figma-bridge-plugin/html-to-figma.js +0 -927
  36. package/figma-bridge-plugin/knowledge-hub/.gitkeep +0 -0
  37. package/figma-bridge-plugin/knowledge-hub/uspec-references/anatomy-spec.md +0 -159
  38. package/figma-bridge-plugin/knowledge-hub/uspec-references/api-spec.md +0 -162
  39. package/figma-bridge-plugin/knowledge-hub/uspec-references/color-spec.md +0 -148
  40. package/figma-bridge-plugin/knowledge-hub/uspec-references/full-spec-template.md +0 -314
  41. package/figma-bridge-plugin/knowledge-hub/uspec-references/property-spec.md +0 -175
  42. package/figma-bridge-plugin/knowledge-hub/uspec-references/screen-reader-spec.md +0 -180
  43. package/figma-bridge-plugin/knowledge-hub/uspec-references/structure-spec.md +0 -165
  44. package/figma-bridge-plugin/perplexity-runner.js +0 -188
  45. package/figma-bridge-plugin/references/SKILL.md +0 -178
  46. package/figma-bridge-plugin/references/anatomy-spec.md +0 -159
  47. package/figma-bridge-plugin/references/api-spec.md +0 -162
  48. package/figma-bridge-plugin/references/color-spec.md +0 -148
  49. package/figma-bridge-plugin/references/full-spec-template.md +0 -314
  50. package/figma-bridge-plugin/references/property-spec.md +0 -175
  51. package/figma-bridge-plugin/references/screen-reader-spec.md +0 -180
  52. package/figma-bridge-plugin/references/structure-spec.md +0 -165
  53. package/figma-bridge-plugin/shared-prompt-config.js +0 -604
  54. package/figma-bridge-plugin/spec-helpers/build-table.js +0 -269
  55. package/figma-bridge-plugin/spec-helpers/classify-elements.js +0 -189
  56. package/figma-bridge-plugin/spec-helpers/index.js +0 -35
  57. package/figma-bridge-plugin/spec-helpers/parse-figma-link.js +0 -49
  58. package/figma-bridge-plugin/spec-helpers/position-markers.js +0 -158
  59. package/figma-bridge-plugin/stitch-auth.js +0 -322
  60. package/figma-bridge-plugin/stitch-runner.js +0 -1427
  61. package/figma-bridge-plugin/token-resolver.js +0 -107
  62. package/figma-bridge-plugin/ui.html +0 -4467
  63. package/figma-intelligence-layer/.env.example +0 -39
  64. package/figma-intelligence-layer/docs/local-image-generation.md +0 -60
  65. package/figma-intelligence-layer/examples/comfyui-workflow-template.example.json +0 -101
  66. package/figma-intelligence-layer/jest.config.js +0 -14
  67. package/figma-intelligence-layer/mcp-config.json +0 -19
  68. package/figma-intelligence-layer/package-lock.json +0 -5892
  69. package/figma-intelligence-layer/scripts/setup-comfyui-local.sh +0 -67
  70. package/figma-intelligence-layer/scripts/start-comfyui.sh +0 -33
  71. package/figma-intelligence-layer/src/index.ts +0 -2233
  72. package/figma-intelligence-layer/src/shared/auto-layout-validator.ts +0 -404
  73. package/figma-intelligence-layer/src/shared/cache.ts +0 -187
  74. package/figma-intelligence-layer/src/shared/color-operations.ts +0 -533
  75. package/figma-intelligence-layer/src/shared/color-utils.ts +0 -138
  76. package/figma-intelligence-layer/src/shared/component-script-builder.ts +0 -413
  77. package/figma-intelligence-layer/src/shared/component-templates.ts +0 -2767
  78. package/figma-intelligence-layer/src/shared/concept-taxonomy.ts +0 -694
  79. package/figma-intelligence-layer/src/shared/decision-log.ts +0 -128
  80. package/figma-intelligence-layer/src/shared/design-system-context.ts +0 -568
  81. package/figma-intelligence-layer/src/shared/design-system-intelligence.ts +0 -131
  82. package/figma-intelligence-layer/src/shared/design-system-matcher.ts +0 -184
  83. package/figma-intelligence-layer/src/shared/design-system-normalizers.ts +0 -196
  84. package/figma-intelligence-layer/src/shared/design-system-tokens.ts +0 -295
  85. package/figma-intelligence-layer/src/shared/dtcg-validator.ts +0 -530
  86. package/figma-intelligence-layer/src/shared/enrichment-pipeline.ts +0 -671
  87. package/figma-intelligence-layer/src/shared/figma-bridge.ts +0 -1408
  88. package/figma-intelligence-layer/src/shared/font-config.ts +0 -126
  89. package/figma-intelligence-layer/src/shared/icon-catalog.ts +0 -360
  90. package/figma-intelligence-layer/src/shared/icon-fetch.ts +0 -80
  91. package/figma-intelligence-layer/src/shared/prototype-script-builder.ts +0 -162
  92. package/figma-intelligence-layer/src/shared/response-compression.ts +0 -440
  93. package/figma-intelligence-layer/src/shared/semantic-token-catalog.ts +0 -324
  94. package/figma-intelligence-layer/src/shared/token-binder.ts +0 -505
  95. package/figma-intelligence-layer/src/shared/token-math.ts +0 -427
  96. package/figma-intelligence-layer/src/shared/token-naming.ts +0 -468
  97. package/figma-intelligence-layer/src/shared/token-utils.ts +0 -420
  98. package/figma-intelligence-layer/src/shared/types.ts +0 -346
  99. package/figma-intelligence-layer/src/shared/typography-presets.ts +0 -94
  100. package/figma-intelligence-layer/src/shared/unsplash.ts +0 -165
  101. package/figma-intelligence-layer/src/shared/vision-client.ts +0 -607
  102. package/figma-intelligence-layer/src/shared/vision-provider-anthropic.ts +0 -334
  103. package/figma-intelligence-layer/src/shared/vision-provider-openai.ts +0 -446
  104. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotate-handler.ts +0 -782
  105. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotate-renderer.ts +0 -496
  106. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/a11y-annotation-kit.ts +0 -230
  107. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/colorblind-sim.ts +0 -66
  108. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/index.ts +0 -810
  109. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-analyzer.ts +0 -1191
  110. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-figma-page.ts +0 -1346
  111. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/keyboard-sr-order-handler.ts +0 -148
  112. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/vpat-figma-page.ts +0 -499
  113. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/vpat-report.ts +0 -910
  114. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/wcag-checker.ts +0 -989
  115. package/figma-intelligence-layer/src/tools/phase1-vision/a11y-audit/wcag-criteria.ts +0 -1160
  116. package/figma-intelligence-layer/src/tools/phase1-vision/design-from-ref/index.ts +0 -424
  117. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/component-recognizer.ts +0 -38
  118. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/ds-matcher.ts +0 -111
  119. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/font-matcher.ts +0 -114
  120. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/icon-resolver.ts +0 -103
  121. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/index.ts +0 -1060
  122. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/layout-segmenter.ts +0 -18
  123. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/token-inferencer.ts +0 -39
  124. package/figma-intelligence-layer/src/tools/phase1-vision/screen-cloner/vision-pipeline.ts +0 -58
  125. package/figma-intelligence-layer/src/tools/phase1-vision/sketch-to-design/index.ts +0 -298
  126. package/figma-intelligence-layer/src/tools/phase1-vision/visual-audit/index.ts +0 -197
  127. package/figma-intelligence-layer/src/tools/phase2-accuracy/component-audit/index.ts +0 -494
  128. package/figma-intelligence-layer/src/tools/phase2-accuracy/intent-translator/index.ts +0 -356
  129. package/figma-intelligence-layer/src/tools/phase2-accuracy/layout-intelligence/container-patterns.ts +0 -123
  130. package/figma-intelligence-layer/src/tools/phase2-accuracy/layout-intelligence/index.ts +0 -663
  131. package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/built-in-rules.yaml +0 -56
  132. package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/index.ts +0 -614
  133. package/figma-intelligence-layer/src/tools/phase2-accuracy/lint-rules/rule-engine.ts +0 -113
  134. package/figma-intelligence-layer/src/tools/phase2-accuracy/theme-generator/color-theory.ts +0 -178
  135. package/figma-intelligence-layer/src/tools/phase2-accuracy/theme-generator/index.ts +0 -470
  136. package/figma-intelligence-layer/src/tools/phase2-accuracy/variant-expander/index.ts +0 -429
  137. package/figma-intelligence-layer/src/tools/phase2-accuracy/variant-expander/token-override-maps.ts +0 -226
  138. package/figma-intelligence-layer/src/tools/phase3-generation/ai-image-insert/index.ts +0 -535
  139. package/figma-intelligence-layer/src/tools/phase3-generation/component-archaeologist/index.ts +0 -660
  140. package/figma-intelligence-layer/src/tools/phase3-generation/component-archaeologist/pattern-fingerprints.ts +0 -209
  141. package/figma-intelligence-layer/src/tools/phase3-generation/composition-builder/index.ts +0 -540
  142. package/figma-intelligence-layer/src/tools/phase3-generation/figma-animated-build.ts +0 -391
  143. package/figma-intelligence-layer/src/tools/phase3-generation/page-architect/index.ts +0 -2019
  144. package/figma-intelligence-layer/src/tools/phase3-generation/page-architect/screen-templates.ts +0 -131
  145. package/figma-intelligence-layer/src/tools/phase3-generation/prototype-map/index.ts +0 -381
  146. package/figma-intelligence-layer/src/tools/phase3-generation/prototype-wire/index.ts +0 -565
  147. package/figma-intelligence-layer/src/tools/phase3-generation/swarm-build/index.ts +0 -764
  148. package/figma-intelligence-layer/src/tools/phase3-generation/system-drift/index.ts +0 -535
  149. package/figma-intelligence-layer/src/tools/phase3-generation/unsplash-search/index.ts +0 -84
  150. package/figma-intelligence-layer/src/tools/phase3-generation/url-to-frame/index.ts +0 -401
  151. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/css-animations.ts +0 -68
  152. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/framer-motion.ts +0 -78
  153. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/code-generators/swift-animations.ts +0 -93
  154. package/figma-intelligence-layer/src/tools/phase4-sync/animation-specifier/index.ts +0 -596
  155. package/figma-intelligence-layer/src/tools/phase4-sync/ci-check/index.ts +0 -462
  156. package/figma-intelligence-layer/src/tools/phase4-sync/export-tokens/index.ts +0 -1470
  157. package/figma-intelligence-layer/src/tools/phase4-sync/generate-component-code/index.ts +0 -829
  158. package/figma-intelligence-layer/src/tools/phase4-sync/handoff-spec/index.ts +0 -702
  159. package/figma-intelligence-layer/src/tools/phase4-sync/icon-library-sync/index.ts +0 -483
  160. package/figma-intelligence-layer/src/tools/phase4-sync/sync-from-code/index.ts +0 -501
  161. package/figma-intelligence-layer/src/tools/phase4-sync/sync-from-code/storybook-parser.ts +0 -106
  162. package/figma-intelligence-layer/src/tools/phase4-sync/watch-docs/index.ts +0 -676
  163. package/figma-intelligence-layer/src/tools/phase4-sync/webhook-listener/index.ts +0 -560
  164. package/figma-intelligence-layer/src/tools/phase5-governance/apg-doc/index.ts +0 -1043
  165. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/component-detection.ts +0 -620
  166. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/anatomy.ts +0 -331
  167. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/color-tokens.ts +0 -77
  168. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/properties.ts +0 -54
  169. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/snapshot.ts +0 -287
  170. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/spacing.ts +0 -71
  171. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/states.ts +0 -43
  172. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/extractors/typography.ts +0 -71
  173. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/index.ts +0 -221
  174. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/_default.ts +0 -166
  175. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/accordion.ts +0 -232
  176. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/alert.ts +0 -234
  177. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/avatar-group.ts +0 -270
  178. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/avatar.ts +0 -249
  179. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/badge.ts +0 -231
  180. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/banner.ts +0 -293
  181. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/breadcrumb.ts +0 -240
  182. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/button.ts +0 -243
  183. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/calendar.ts +0 -307
  184. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/card.ts +0 -143
  185. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/checkbox.ts +0 -227
  186. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/chip.ts +0 -233
  187. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/combobox.ts +0 -282
  188. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/datepicker.ts +0 -276
  189. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/divider.ts +0 -223
  190. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/drawer.ts +0 -255
  191. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/dropdown-menu.ts +0 -289
  192. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/empty-state.ts +0 -261
  193. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/file-uploader.ts +0 -290
  194. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/form.ts +0 -265
  195. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/grid.ts +0 -238
  196. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/icon.ts +0 -255
  197. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/index.ts +0 -128
  198. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/inline-edit.ts +0 -286
  199. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/inline-message.ts +0 -255
  200. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/input.ts +0 -330
  201. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/link.ts +0 -247
  202. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/list.ts +0 -250
  203. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/menu.ts +0 -247
  204. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/modal.ts +0 -144
  205. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/navbar.ts +0 -264
  206. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/navigation.ts +0 -251
  207. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/number-input.ts +0 -261
  208. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/pagination.ts +0 -248
  209. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/popover.ts +0 -270
  210. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/progress.ts +0 -251
  211. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/radio.ts +0 -142
  212. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/range-slider.ts +0 -282
  213. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/rating.ts +0 -250
  214. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/search.ts +0 -258
  215. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/segmented-control.ts +0 -265
  216. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/select.ts +0 -319
  217. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/skeleton.ts +0 -256
  218. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/slider.ts +0 -232
  219. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/spinner.ts +0 -239
  220. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/status-dot.ts +0 -252
  221. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/stepper.ts +0 -270
  222. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/table.ts +0 -244
  223. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tabs.ts +0 -143
  224. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tag.ts +0 -243
  225. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/textarea.ts +0 -259
  226. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/time-picker.ts +0 -293
  227. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toast.ts +0 -144
  228. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toggle.ts +0 -289
  229. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/toolbar.ts +0 -267
  230. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/tooltip.ts +0 -232
  231. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/treeview.ts +0 -257
  232. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/knowledge/typography.ts +0 -319
  233. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/legacy-compat.ts +0 -121
  234. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/anatomy-diagram.ts +0 -430
  235. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/figma-page.ts +0 -312
  236. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/json.ts +0 -129
  237. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/markdown.ts +0 -78
  238. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/renderers/visual-doc.ts +0 -2333
  239. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/accessibility.ts +0 -100
  240. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/anatomy.ts +0 -32
  241. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/color-tokens.ts +0 -59
  242. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/content-guidance.ts +0 -18
  243. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/design-tokens.ts +0 -53
  244. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/interaction-rules.ts +0 -19
  245. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/overview.ts +0 -91
  246. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/properties-api.ts +0 -71
  247. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/qa-criteria.ts +0 -19
  248. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/related-components.ts +0 -110
  249. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/responsive.ts +0 -19
  250. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/size-specs.ts +0 -67
  251. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/spacing-structure.ts +0 -58
  252. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/state-specs.ts +0 -79
  253. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/states.ts +0 -50
  254. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/type-hierarchy.ts +0 -33
  255. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/typography.ts +0 -55
  256. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/usage-guidelines.ts +0 -73
  257. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/sections/variants.ts +0 -81
  258. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec/types.ts +0 -409
  259. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/index.ts +0 -198
  260. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/renderer.ts +0 -701
  261. package/figma-intelligence-layer/src/tools/phase5-governance/component-spec-sheet/types.ts +0 -88
  262. package/figma-intelligence-layer/src/tools/phase5-governance/decision-log/index.ts +0 -135
  263. package/figma-intelligence-layer/src/tools/phase5-governance/design-decision-log/index.ts +0 -491
  264. package/figma-intelligence-layer/src/tools/phase5-governance/ds-primitives/index.ts +0 -416
  265. package/figma-intelligence-layer/src/tools/phase5-governance/ds-scaffolder/index.ts +0 -722
  266. package/figma-intelligence-layer/src/tools/phase5-governance/ds-variables/index.ts +0 -449
  267. package/figma-intelligence-layer/src/tools/phase5-governance/health-report/index.ts +0 -393
  268. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/index.ts +0 -406
  269. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/figma-page.ts +0 -292
  270. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/json.ts +0 -24
  271. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/markdown.ts +0 -172
  272. package/figma-intelligence-layer/src/tools/phase5-governance/taxonomy-docs/renderers/naming-guide.ts +0 -409
  273. package/figma-intelligence-layer/src/tools/phase5-governance/token-analytics/index.ts +0 -594
  274. package/figma-intelligence-layer/src/tools/phase5-governance/token-docs/index.ts +0 -710
  275. package/figma-intelligence-layer/src/tools/phase5-governance/token-migrate/index.ts +0 -458
  276. package/figma-intelligence-layer/src/tools/phase5-governance/token-naming/index.ts +0 -134
  277. package/figma-intelligence-layer/tests/apg-doc.test.ts +0 -101
  278. package/figma-intelligence-layer/tests/design-system-context.test.ts +0 -152
  279. package/figma-intelligence-layer/tests/design-system-matcher.test.ts +0 -144
  280. package/figma-intelligence-layer/tests/figma-bridge.test.ts +0 -83
  281. package/figma-intelligence-layer/tests/generate-image-and-insert.test.ts +0 -56
  282. package/figma-intelligence-layer/tests/screen-cloner-regression.test.ts +0 -69
  283. package/figma-intelligence-layer/tests/smoke.test.ts +0 -174
  284. package/figma-intelligence-layer/tests/spec-generator.test.ts +0 -127
  285. package/figma-intelligence-layer/tests/token-migrate.test.ts +0 -21
  286. package/figma-intelligence-layer/tests/token-naming.test.ts +0 -30
  287. package/figma-intelligence-layer/tsconfig.json +0 -19
  288. package/scripts/clean-existing-chunks.js +0 -179
  289. package/scripts/connect-ai-tool.js +0 -490
  290. package/scripts/convert-hub-pdfs.js +0 -425
  291. package/scripts/figma-mcp-status.js +0 -349
  292. package/scripts/register-codex-mcp.js +0 -96
  293. /package/{design-bridge → dist/design-bridge}/.env.example +0 -0
  294. /package/{design-bridge → dist/design-bridge}/package.json +0 -0
  295. /package/{figma-bridge-plugin → dist/figma-bridge-plugin}/manifest.json +0 -0
  296. /package/{figma-bridge-plugin → dist/figma-bridge-plugin}/package.json +0 -0
  297. /package/{figma-intelligence-layer → dist/figma-intelligence-layer}/package.json +0 -0
@@ -1,1408 +0,0 @@
1
- import WebSocket, { WebSocketServer } from "ws";
2
- import {
3
- FigmaNode,
4
- ExecuteResult,
5
- ComponentSet,
6
- Token,
7
- FigmaBridgeEvent,
8
- FigmaBridgeEventType,
9
- FigmaContextSnapshot,
10
- FigmaDocumentChangeSummary,
11
- FigmaPageSummary,
12
- FigmaSelectionItem,
13
- } from "./types.js";
14
- import { BridgeCache } from "./cache.js";
15
- import { compressResponse, CompressedResponse, CompressionTier } from "./response-compression.js";
16
- import { enrichDesignSystem, EnrichedDesignSystem, resolveStyles, ResolvedStyle } from "./enrichment-pipeline.js";
17
-
18
- const WS_PORT = parseInt(process.env.FIGMA_BRIDGE_PORT || "9001", 10);
19
- const REQUEST_TIMEOUT = parseInt(process.env.FIGMA_REQUEST_TIMEOUT || "30000", 10);
20
-
21
- let relayServer: WebSocketServer | null = null;
22
- let relayPluginSocket: WebSocket | null = null;
23
- const relayMcpSockets = new Set<WebSocket>();
24
- const relayPendingRequests = new Map<string, WebSocket>();
25
- let relayStartupPromise: Promise<void> | null = null;
26
-
27
- type RelayMessage = {
28
- id?: string;
29
- method?: string;
30
- params?: Record<string, unknown>;
31
- result?: unknown;
32
- error?: string;
33
- type?: string;
34
- fileName?: string;
35
- eventType?: FigmaBridgeEventType;
36
- payload?: unknown;
37
- timestamp?: number;
38
- };
39
-
40
- function sendRelayStatus(ws: WebSocket | null, mcpConnected: boolean) {
41
- if (!ws || ws.readyState !== WebSocket.OPEN) return;
42
-
43
- ws.send(JSON.stringify({
44
- type: "bridge-status",
45
- mcpConnected,
46
- }));
47
- }
48
-
49
- function hasConnectedMcpSocket() {
50
- for (const socket of relayMcpSockets) {
51
- if (socket.readyState === WebSocket.OPEN) return true;
52
- }
53
- return false;
54
- }
55
-
56
- function broadcastToMcpSockets(raw: string) {
57
- for (const socket of relayMcpSockets) {
58
- if (socket.readyState === WebSocket.OPEN) {
59
- socket.send(raw);
60
- }
61
- }
62
- }
63
-
64
- function setupRelayRouting(wss: WebSocketServer) {
65
- wss.on("connection", (ws, req) => {
66
- const path = req.url || "/";
67
- const isPlugin = path.includes("/plugin");
68
-
69
- if (isPlugin) {
70
- relayPluginSocket = ws;
71
- process.stderr.write("Figma bridge plugin connected\n");
72
- sendRelayStatus(ws, hasConnectedMcpSocket());
73
- } else {
74
- relayMcpSockets.add(ws);
75
- process.stderr.write("Figma bridge MCP socket connected\n");
76
- sendRelayStatus(relayPluginSocket, true);
77
- }
78
-
79
- ws.on("message", (data) => {
80
- const raw = data.toString();
81
- let msg: RelayMessage;
82
-
83
- try {
84
- msg = JSON.parse(raw);
85
- } catch {
86
- return;
87
- }
88
-
89
- if (isPlugin) {
90
- if (msg.type === "plugin-hello") {
91
- process.stderr.write(`Figma bridge plugin ready (${msg.fileName || "unknown file"})\n`);
92
- return;
93
- }
94
-
95
- if (msg.type === "bridge-event") {
96
- broadcastToMcpSockets(raw);
97
- return;
98
- }
99
-
100
- if (msg.id && !msg.method) {
101
- const targetSocket = relayPendingRequests.get(msg.id);
102
- if (targetSocket && targetSocket.readyState === WebSocket.OPEN) {
103
- targetSocket.send(raw);
104
- } else {
105
- broadcastToMcpSockets(raw);
106
- }
107
- relayPendingRequests.delete(msg.id);
108
- }
109
- return;
110
- }
111
-
112
- if (msg.id && msg.method) {
113
- if (relayPluginSocket && relayPluginSocket.readyState === WebSocket.OPEN) {
114
- relayPendingRequests.set(msg.id, ws);
115
- relayPluginSocket.send(JSON.stringify({
116
- type: "bridge-request",
117
- id: msg.id,
118
- method: msg.method,
119
- params: msg.params || {},
120
- }));
121
- } else {
122
- ws.send(JSON.stringify({
123
- id: msg.id,
124
- error: "Figma plugin is not connected. Open Figma and run the Intelligence Bridge plugin.",
125
- }));
126
- }
127
- }
128
- });
129
-
130
- ws.on("close", () => {
131
- if (isPlugin && relayPluginSocket === ws) {
132
- relayPluginSocket = null;
133
- process.stderr.write("Figma bridge plugin disconnected\n");
134
- }
135
-
136
- if (!isPlugin) {
137
- relayMcpSockets.delete(ws);
138
- for (const [requestId, requestSocket] of relayPendingRequests.entries()) {
139
- if (requestSocket === ws) {
140
- relayPendingRequests.delete(requestId);
141
- }
142
- }
143
- process.stderr.write("Figma bridge MCP socket disconnected\n");
144
- sendRelayStatus(relayPluginSocket, hasConnectedMcpSocket());
145
- }
146
- });
147
- });
148
- }
149
-
150
- // P3: Port fallback range
151
- const PORT_FALLBACK_RANGE = 10;
152
-
153
- export async function ensureRelayServer(): Promise<void> {
154
- if (relayServer) return;
155
- if (relayStartupPromise) return relayStartupPromise;
156
-
157
- relayStartupPromise = new Promise<void>((resolve, reject) => {
158
- // Try to create our own relay server on the configured port
159
- const wss = new WebSocketServer({ port: WS_PORT });
160
- let settled = false;
161
-
162
- const finish = (callback: () => unknown) => {
163
- if (settled) return;
164
- settled = true;
165
- callback();
166
- };
167
-
168
- setupRelayRouting(wss);
169
-
170
- wss.on("listening", () => {
171
- relayServer = wss;
172
- process.stderr.write(`Figma bridge relay listening on ws://localhost:${WS_PORT}\n`);
173
- finish(() => resolve());
174
- });
175
-
176
- wss.on("error", (error: NodeJS.ErrnoException) => {
177
- if (error.code === "EADDRINUSE") {
178
- // Port in use — bridge-relay is already running externally.
179
- // Don't try other ports; just connect as a client to the existing relay.
180
- process.stderr.write(`Figma bridge relay already running on ws://localhost:${WS_PORT} — connecting as client\n`);
181
- finish(() => resolve());
182
- return;
183
- }
184
- finish(() => reject(error));
185
- });
186
- }).finally(() => {
187
- relayStartupPromise = null;
188
- });
189
-
190
- return relayStartupPromise ?? Promise.resolve();
191
- }
192
-
193
- // ─────────────────────────────────────────────────────────────────────────────
194
- // FigmaBridge
195
- // Wraps the Desktop Bridge WebSocket connection with typed helpers
196
- // ─────────────────────────────────────────────────────────────────────────────
197
-
198
- export class FigmaBridge {
199
- private ws: WebSocket | null = null;
200
- private pendingRequests = new Map<
201
- string,
202
- { resolve: (v: unknown) => void; reject: (e: Error) => void }
203
- >();
204
- private eventListeners = new Map<FigmaBridgeEventType | "*", Set<(event: FigmaBridgeEvent) => void>>();
205
- private context: FigmaContextSnapshot = {
206
- status: "disconnected",
207
- selection: [],
208
- };
209
- private hasHydratedStatus = false;
210
- private hasHydratedSelection = false;
211
- private connected = false;
212
- private connectPromise: Promise<void> | null = null;
213
- private capabilitiesCache: Record<string, unknown> | null = null;
214
- private _activeDesignSystemId: string | null = null;
215
-
216
- // ─── Taxonomy auto-sync ─────────────────────────────────────────────────
217
- private _taxonomyAutoSyncTimer: ReturnType<typeof setTimeout> | null = null;
218
- private _taxonomyAutoSyncCallback: (() => Promise<void>) | null = null;
219
-
220
- /**
221
- * Register a callback to run when variables change (for taxonomy docs auto-sync).
222
- * The callback is debounced with a 2-second delay.
223
- */
224
- setTaxonomyAutoSyncHandler(handler: (() => Promise<void>) | null): void {
225
- this._taxonomyAutoSyncCallback = handler;
226
- }
227
-
228
- private scheduleTaxonomyAutoSync(): void {
229
- if (!this._taxonomyAutoSyncCallback) return;
230
- if (this._taxonomyAutoSyncTimer) clearTimeout(this._taxonomyAutoSyncTimer);
231
- this._taxonomyAutoSyncTimer = setTimeout(async () => {
232
- this._taxonomyAutoSyncTimer = null;
233
- try {
234
- await this._taxonomyAutoSyncCallback?.();
235
- } catch (e) {
236
- process.stderr.write(`Taxonomy auto-sync error: ${e}\n`);
237
- }
238
- }, 2000);
239
- }
240
-
241
- // ─── P1: Caching Layer ──────────────────────────────────────────────────
242
- private _cache: BridgeCache | null = null;
243
- get cache(): BridgeCache {
244
- if (!this._cache) this._cache = new BridgeCache(this);
245
- return this._cache;
246
- }
247
-
248
- isConnected(): boolean {
249
- return this.connected && this.ws?.readyState === WebSocket.OPEN;
250
- }
251
-
252
- /**
253
- * Get the currently selected design system ID (e.g. "antd", "carbon", "mui").
254
- * Returns null if no design system is selected in the UI.
255
- */
256
- getActiveDesignSystemId(): string | null {
257
- return this._activeDesignSystemId;
258
- }
259
-
260
- /**
261
- * Request the current design system ID from the relay and cache it locally.
262
- */
263
- private async hydrateDesignSystemId(): Promise<void> {
264
- try {
265
- const result = await this.send<string | null>("getActiveDesignSystemId");
266
- this._activeDesignSystemId = result ?? null;
267
- } catch {
268
- // Non-critical — may not be supported by older relays
269
- }
270
- }
271
-
272
- private rejectAllPending(reason: Error) {
273
- const pending = Array.from(this.pendingRequests.values());
274
- this.pendingRequests.clear();
275
- for (const request of pending) {
276
- request.reject(reason);
277
- }
278
- }
279
-
280
- private invalidateConnection(reason: Error) {
281
- const socket = this.ws;
282
- this.connected = false;
283
- this.ws = null;
284
- // Clear ALL cached state so reconnect always fetches fresh data
285
- this.context = {
286
- status: "disconnected",
287
- fileName: undefined,
288
- currentPage: undefined,
289
- pageCount: 0,
290
- selection: [],
291
- lastUpdatedAt: Date.now(),
292
- lastDocumentChange: undefined,
293
- };
294
- this.hasHydratedStatus = false;
295
- this.hasHydratedSelection = false;
296
- this.capabilitiesCache = null;
297
- this.rejectAllPending(reason);
298
- if (socket && socket.readyState === WebSocket.OPEN) {
299
- socket.close();
300
- }
301
- }
302
-
303
- async connect(): Promise<void> {
304
- if (this.isConnected()) return;
305
- if (this.connectPromise) return this.connectPromise;
306
- await ensureRelayServer();
307
- this.connectPromise = new Promise<void>((resolve, reject) => {
308
- const socket = new WebSocket(`ws://localhost:${WS_PORT}`);
309
- const handleFailure = (err: Error) => {
310
- this.connected = false;
311
- this.ws = null;
312
- reject(err);
313
- };
314
-
315
- socket.on("open", () => {
316
- this.ws = socket;
317
- this.connected = true;
318
- this.context = {
319
- ...this.context,
320
- status: "connected",
321
- lastUpdatedAt: Date.now(),
322
- };
323
- // Proactively hydrate status + selection in background so next getStatus() is instant
324
- this.hydrateAfterConnect();
325
- resolve();
326
- });
327
- socket.on("error", (err) => {
328
- handleFailure(err instanceof Error ? err : new Error(String(err)));
329
- });
330
- socket.on("message", (data) => this.handleMessage(String(data)));
331
- socket.on("close", () => {
332
- this.invalidateConnection(new Error("FigmaBridge: socket closed"));
333
- });
334
- }).finally((): void => {
335
- this.connectPromise = null;
336
- });
337
-
338
- return this.connectPromise ?? Promise.resolve();
339
- }
340
-
341
- /** Fire-and-forget hydration after (re)connect — populates context cache quickly */
342
- private hydrateAfterConnect(): void {
343
- // Hydrate active design system ID from relay
344
- this.hydrateDesignSystemId();
345
- if (this.hasHydratedStatus) return;
346
- // Use a short timeout for the hydration RPC (5s instead of default 30s)
347
- const FAST_TIMEOUT = 5000;
348
- const id = `req_${Date.now()}_${Math.random().toString(36).slice(2)}`;
349
- if (!this.ws || this.ws.readyState !== WebSocket.OPEN) return;
350
-
351
- const timeout = setTimeout(() => {
352
- this.pendingRequests.delete(id);
353
- // Don't invalidate connection on hydration timeout — it's best-effort
354
- }, FAST_TIMEOUT);
355
-
356
- this.pendingRequests.set(id, {
357
- resolve: (v) => {
358
- clearTimeout(timeout);
359
- const status = v as { fileName?: string; currentPage?: FigmaPageSummary; pageCount?: number; timestamp?: number };
360
- if (status?.fileName) {
361
- this.context = {
362
- ...this.context,
363
- status: "connected",
364
- fileName: status.fileName,
365
- currentPage: status.currentPage,
366
- pageCount: status.pageCount || 0,
367
- lastUpdatedAt: status.timestamp || Date.now(),
368
- };
369
- this.hasHydratedStatus = true;
370
- }
371
- },
372
- reject: (e) => {
373
- clearTimeout(timeout);
374
- // Swallow — hydration is best-effort
375
- },
376
- });
377
-
378
- this.ws.send(JSON.stringify({ id, method: "getStatus", params: {} }));
379
- }
380
-
381
- private emitEvent(event: FigmaBridgeEvent) {
382
- const specificListeners = this.eventListeners.get(event.eventType);
383
- const catchAllListeners = this.eventListeners.get("*");
384
-
385
- for (const listener of specificListeners || []) {
386
- listener(event);
387
- }
388
-
389
- for (const listener of catchAllListeners || []) {
390
- listener(event);
391
- }
392
- }
393
-
394
- private updateContextFromEvent(event: FigmaBridgeEvent) {
395
- const nextContext: FigmaContextSnapshot = {
396
- ...this.context,
397
- status: "connected",
398
- lastUpdatedAt: event.timestamp,
399
- };
400
-
401
- if (event.eventType === "bridge.ready") {
402
- const payload = event.payload as Partial<FigmaContextSnapshot>;
403
- nextContext.fileName = payload.fileName || nextContext.fileName;
404
- nextContext.currentPage = payload.currentPage || nextContext.currentPage;
405
- nextContext.pageCount = payload.pageCount || nextContext.pageCount;
406
- nextContext.selection = payload.selection || nextContext.selection;
407
- this.hasHydratedStatus = true;
408
- this.hasHydratedSelection = true;
409
- }
410
-
411
- if (event.eventType === "selectionchange") {
412
- nextContext.selection = (event.payload as { selection?: FigmaSelectionItem[] }).selection || [];
413
- this.hasHydratedSelection = true;
414
- }
415
-
416
- if (event.eventType === "currentpagechange") {
417
- const payload = event.payload as { currentPage?: FigmaPageSummary; selection?: FigmaSelectionItem[] };
418
- nextContext.currentPage = payload.currentPage || nextContext.currentPage;
419
- if (payload.selection) {
420
- nextContext.selection = payload.selection;
421
- this.hasHydratedSelection = true;
422
- }
423
- this.hasHydratedStatus = true;
424
- }
425
-
426
- if (event.eventType === "documentchange") {
427
- const payload = event.payload as FigmaDocumentChangeSummary;
428
- nextContext.lastDocumentChange = payload;
429
- // P1: Invalidate cache on document changes
430
- if (this._cache) this._cache.onDocumentChange();
431
- }
432
-
433
- // Auto-sync: trigger taxonomy docs re-render on variable changes
434
- if ((event.eventType as string) === "variable-change") {
435
- this.scheduleTaxonomyAutoSync();
436
- }
437
-
438
- this.context = nextContext;
439
- }
440
-
441
- private handleMessage(raw: string) {
442
- let msg: RelayMessage;
443
- try {
444
- msg = JSON.parse(raw);
445
- } catch {
446
- return;
447
- }
448
- // Handle design system change broadcasts from relay
449
- if (msg.type === "design-system-changed") {
450
- this._activeDesignSystemId = (msg as Record<string, unknown>).designSystemId as string | null ?? null;
451
- return;
452
- }
453
-
454
- if (msg.type === "bridge-event" && msg.eventType) {
455
- const event: FigmaBridgeEvent = {
456
- type: "bridge-event",
457
- eventType: msg.eventType,
458
- payload: msg.payload,
459
- timestamp: msg.timestamp || Date.now(),
460
- };
461
- this.updateContextFromEvent(event);
462
- this.emitEvent(event);
463
- return;
464
- }
465
- if (!msg.id) return;
466
- const pending = this.pendingRequests.get(msg.id);
467
- if (!pending) return;
468
- this.pendingRequests.delete(msg.id);
469
- if (msg.error) {
470
- pending.reject(new Error(msg.error));
471
- } else {
472
- pending.resolve(msg.result);
473
- }
474
- }
475
-
476
- private send<T>(method: string, params: Record<string, unknown> = {}): Promise<T> {
477
- return this.sendWithTimeout<T>(method, params, REQUEST_TIMEOUT);
478
- }
479
-
480
- private sendWithTimeout<T>(method: string, params: Record<string, unknown> = {}, timeoutMs: number = REQUEST_TIMEOUT): Promise<T> {
481
- return new Promise((resolve, reject) => {
482
- if (!this.isConnected() || !this.ws) {
483
- reject(new Error("FigmaBridge: not connected"));
484
- return;
485
- }
486
- const id = `req_${Date.now()}_${Math.random().toString(36).slice(2)}`;
487
- const timeout = setTimeout(() => {
488
- this.pendingRequests.delete(id);
489
- const error = new Error(`FigmaBridge: timeout on method ${method}`);
490
- this.invalidateConnection(error);
491
- reject(error);
492
- }, timeoutMs);
493
-
494
- this.pendingRequests.set(id, {
495
- resolve: (v) => { clearTimeout(timeout); resolve(v as T); },
496
- reject: (e) => { clearTimeout(timeout); reject(e); },
497
- });
498
-
499
- this.ws.send(JSON.stringify({ id, method, params }));
500
- });
501
- }
502
-
503
- async execute(script: string, timeoutMs?: number): Promise<ExecuteResult> {
504
- try {
505
- const result = await this.sendWithTimeout<unknown>("execute", { code: script }, timeoutMs || REQUEST_TIMEOUT);
506
- return { success: true, result };
507
- } catch (err) {
508
- return {
509
- success: false,
510
- error: err instanceof Error ? err.message : String(err),
511
- };
512
- }
513
- }
514
-
515
- async getNode(nodeId: string): Promise<FigmaNode> {
516
- const result = await this.execute(`
517
- const node = await figma.getNodeByIdAsync(${JSON.stringify(nodeId)});
518
- if (!node) throw new Error("Node not found: " + ${JSON.stringify(nodeId)});
519
- return JSON.parse(JSON.stringify(node));
520
- `);
521
- if (!result.success) throw new Error(result.error);
522
- return result.result as FigmaNode;
523
- }
524
-
525
- async takeScreenshot(nodeId: string): Promise<string> {
526
- // Use the plugin's dedicated "screenshot" handler — it exports PNG bytes
527
- // in the plugin sandbox, then the UI layer (browser) converts to base64.
528
- // This avoids calling btoa() inside the Figma plugin sandbox where it doesn't exist.
529
- const dataUri = await this.sendWithTimeout<string>("screenshot", { nodeId, scale: 2 }, 60000);
530
- return dataUri;
531
- }
532
-
533
- async importImage(imageDataUri: string): Promise<{ imageHash: string; byteLength: number }> {
534
- const result = await this.send<{ imageHash: string; byteLength: number }>("importImage", {
535
- imageDataUri,
536
- });
537
- return result;
538
- }
539
-
540
- async getComponentSets(): Promise<ComponentSet[]> {
541
- const result = await this.execute(`
542
- var page = figma.currentPage;
543
- var sets = page.findAll(function(n) { return n.type === 'COMPONENT_SET'; });
544
- var allSets = [];
545
- for (var j = 0; j < sets.length; j++) {
546
- var s = sets[j];
547
- try {
548
- allSets.push({
549
- id: s.id,
550
- name: s.name,
551
- description: s.description || '',
552
- variantGroupProperties: s.variantGroupProperties || {},
553
- children: s.children.map(function(c) { return { id: c.id, name: c.name, type: c.type, description: c.description || '' }; })
554
- });
555
- } catch (err) {
556
- // Skip component sets with corrupted variant metadata so clone generation can continue.
557
- }
558
- if (allSets.length > 200) break;
559
- }
560
- return allSets;
561
- `);
562
- if (!result.success) throw new Error(result.error);
563
- return result.result as ComponentSet[];
564
- }
565
-
566
- async getTokens(collectionId?: string): Promise<Token[]> {
567
- const filter = collectionId
568
- ? `collections.filter(c => c.id === ${JSON.stringify(collectionId)})`
569
- : "collections";
570
- const result = await this.execute(`
571
- const collections = await figma.variables.getLocalVariableCollectionsAsync();
572
- const filtered = ${filter};
573
- const tokens = [];
574
- for (const col of filtered) {
575
- const vars = [];
576
- for (const id of col.variableIds) {
577
- const v = await figma.variables.getVariableByIdAsync(id);
578
- if (v) vars.push(v);
579
- }
580
- for (const v of vars) {
581
- const modeValues = {};
582
- for (const [modeId, val] of Object.entries(v.valuesByMode)) {
583
- const mode = col.modes.find(m => m.modeId === modeId);
584
- modeValues[mode ? mode.name : modeId] = val;
585
- }
586
- var firstVal = Object.values(modeValues)[0];
587
- tokens.push({
588
- id: v.id,
589
- name: v.name,
590
- type: v.resolvedType,
591
- value: firstVal !== undefined ? firstVal : null,
592
- collectionId: col.id,
593
- modeValues,
594
- description: v.description || '',
595
- });
596
- }
597
- }
598
- return tokens;
599
- `);
600
- if (!result.success) throw new Error(result.error);
601
- return result.result as Token[];
602
- }
603
-
604
- async getAllPages(): Promise<Array<{ id: string; name: string }>> {
605
- const result = await this.execute(`
606
- return figma.root.children.map(p => ({ id: p.id, name: p.name }));
607
- `);
608
- if (!result.success) throw new Error(result.error);
609
- return result.result as Array<{ id: string; name: string }>;
610
- }
611
-
612
- async createPage(name: string): Promise<string> {
613
- const result = await this.execute(`
614
- const page = figma.createPage();
615
- page.name = ${JSON.stringify(name)};
616
- return page.id;
617
- `);
618
- if (!result.success) throw new Error(result.error);
619
- return result.result as string;
620
- }
621
-
622
- async navigateToNode(nodeId: string): Promise<void> {
623
- const result = await this.execute(`
624
- const node = await figma.getNodeByIdAsync(${JSON.stringify(nodeId)});
625
- if (node) figma.viewport.scrollAndZoomIntoView([node]);
626
- `);
627
- if (!result.success) throw new Error(result.error);
628
- }
629
-
630
- async getAllInstances(): Promise<Array<{ id: string; mainComponentId: string; name: string; pageId: string }>> {
631
- const result = await this.execute(`
632
- const instances = [];
633
- const page = figma.currentPage;
634
- const found = page.findAll(n => n.type === 'INSTANCE');
635
- for (const inst of found) {
636
- instances.push({
637
- id: inst.id,
638
- mainComponentId: inst.mainComponent ? inst.mainComponent.id : '',
639
- name: inst.name,
640
- pageId: page.id,
641
- });
642
- }
643
- return instances;
644
- `);
645
- if (!result.success) throw new Error(result.error);
646
- return result.result as Array<{ id: string; mainComponentId: string; name: string; pageId: string }>;
647
- }
648
-
649
- async getPrototypeConnections(): Promise<Array<{
650
- fromId: string;
651
- fromName: string;
652
- toId: string;
653
- toName: string;
654
- trigger: Record<string, unknown>;
655
- action: Record<string, unknown>;
656
- }>> {
657
- const result = await this.execute(`
658
- const connections = [];
659
- const page = figma.currentPage;
660
- const frames = page.findAll(n => n.type === 'FRAME' || n.type === 'COMPONENT');
661
- for (const frame of frames) {
662
- const reactions = frame.reactions || [];
663
- for (const r of reactions) {
664
- if (r.action && r.action.type === 'NODE') {
665
- const destNode = await figma.getNodeByIdAsync(r.action.destinationId);
666
- connections.push({
667
- fromId: frame.id,
668
- fromName: frame.name,
669
- toId: r.action.destinationId,
670
- toName: destNode ? destNode.name : 'Unknown',
671
- trigger: r.trigger || {},
672
- action: r.action || {},
673
- });
674
- }
675
- }
676
- }
677
- return connections;
678
- `);
679
- if (!result.success) throw new Error(result.error);
680
- return result.result as Array<{
681
- fromId: string;
682
- fromName: string;
683
- toId: string;
684
- toName: string;
685
- trigger: Record<string, unknown>;
686
- action: Record<string, unknown>;
687
- }>;
688
- }
689
-
690
- async disconnect(): Promise<void> {
691
- if (this.ws) {
692
- this.ws.close();
693
- this.ws = null;
694
- this.connected = false;
695
- this.context = {
696
- ...this.context,
697
- status: "disconnected",
698
- lastUpdatedAt: Date.now(),
699
- };
700
- }
701
- }
702
-
703
- async hydrateContext(): Promise<FigmaContextSnapshot> {
704
- if (!this.context.fileName || !this.context.currentPage) {
705
- const status = await this.send<{
706
- status: "connected";
707
- fileName: string;
708
- currentPage: FigmaPageSummary;
709
- pageCount: number;
710
- timestamp: number;
711
- }>("getStatus");
712
- this.context = {
713
- ...this.context,
714
- status: "connected",
715
- fileName: status.fileName,
716
- currentPage: status.currentPage,
717
- pageCount: status.pageCount,
718
- lastUpdatedAt: status.timestamp,
719
- };
720
- this.hasHydratedStatus = true;
721
- }
722
-
723
- if (!this.hasHydratedSelection) {
724
- const selection = await this.send<FigmaSelectionItem[]>("getSelection");
725
- this.context = {
726
- ...this.context,
727
- status: "connected",
728
- selection,
729
- lastUpdatedAt: Date.now(),
730
- };
731
- this.hasHydratedSelection = true;
732
- }
733
-
734
- return this.getContextSnapshot();
735
- }
736
-
737
- getContextSnapshot(): FigmaContextSnapshot {
738
- return {
739
- ...this.context,
740
- selection: [...this.context.selection],
741
- currentPage: this.context.currentPage ? { ...this.context.currentPage } : undefined,
742
- lastDocumentChange: this.context.lastDocumentChange
743
- ? {
744
- documentChanges: this.context.lastDocumentChange.documentChanges.map((change) => ({ ...change })),
745
- }
746
- : undefined,
747
- };
748
- }
749
-
750
- onEvent(
751
- eventType: FigmaBridgeEventType | "*",
752
- listener: (event: FigmaBridgeEvent) => void
753
- ): () => void {
754
- const listeners = this.eventListeners.get(eventType) || new Set<(event: FigmaBridgeEvent) => void>();
755
- listeners.add(listener);
756
- this.eventListeners.set(eventType, listeners);
757
-
758
- return () => {
759
- const existing = this.eventListeners.get(eventType);
760
- if (!existing) return;
761
- existing.delete(listener);
762
- if (existing.size === 0) {
763
- this.eventListeners.delete(eventType);
764
- }
765
- };
766
- }
767
-
768
- handleIncomingMessage(raw: string) {
769
- this.handleMessage(raw);
770
- }
771
-
772
- // ─── Navigation & Status ─────────────────────────────────────────────────
773
-
774
- async getStatus(): Promise<Record<string, unknown>> {
775
- // Return cached status only if context has been hydrated and looks fresh
776
- if (this.hasHydratedStatus && this.context.fileName && this.context.currentPage) {
777
- return {
778
- status: this.context.status,
779
- fileName: this.context.fileName,
780
- currentPage: this.context.currentPage,
781
- pageCount: this.context.pageCount,
782
- timestamp: this.context.lastUpdatedAt || Date.now(),
783
- };
784
- }
785
-
786
- // Use shorter timeout (5s) for status — don't wait 30s for a simple ping
787
- const STATUS_TIMEOUT = 5000;
788
- const status = await this.sendWithTimeout<{
789
- status: "connected";
790
- fileName: string;
791
- currentPage: FigmaPageSummary;
792
- pageCount: number;
793
- timestamp: number;
794
- }>("getStatus", {}, STATUS_TIMEOUT);
795
-
796
- this.context = {
797
- ...this.context,
798
- status: "connected",
799
- fileName: status.fileName,
800
- currentPage: status.currentPage,
801
- pageCount: status.pageCount,
802
- lastUpdatedAt: status.timestamp,
803
- };
804
- this.hasHydratedStatus = true;
805
-
806
- return status;
807
- }
808
-
809
- async navigate(nodeId: string): Promise<Record<string, unknown>> {
810
- return this.send("navigate", { nodeId });
811
- }
812
-
813
- async getSelection(): Promise<Array<{ id: string; name: string; type: string }>> {
814
- if (this.hasHydratedSelection) {
815
- return this.context.selection.map((item) => ({ ...item }));
816
- }
817
-
818
- // Use shorter timeout (5s) for selection queries
819
- const selection = await this.sendWithTimeout<FigmaSelectionItem[]>("getSelection", {}, 5000);
820
- this.context = {
821
- ...this.context,
822
- status: "connected",
823
- selection,
824
- lastUpdatedAt: Date.now(),
825
- };
826
- this.hasHydratedSelection = true;
827
-
828
- return selection;
829
- }
830
-
831
- async getCapabilities(force = false): Promise<Record<string, unknown>> {
832
- if (!force && this.capabilitiesCache) {
833
- return { ...this.capabilitiesCache };
834
- }
835
-
836
- const capabilities = await this.send<Record<string, unknown>>("getCapabilities");
837
- this.capabilitiesCache = capabilities;
838
- return { ...capabilities };
839
- }
840
-
841
- private async ensureVariablesApi(): Promise<void> {
842
- const capabilities = await this.getCapabilities();
843
- if (!capabilities.variablesApi || !capabilities.localVariablesApi) {
844
- throw new Error(
845
- "Figma Variables API is unavailable in the current plugin runtime. Reopen the bridge plugin in a Variables-capable Figma editor context."
846
- );
847
- }
848
- }
849
-
850
- // ─── Variable CRUD ───────────────────────────────────────────────────────
851
-
852
- async createVariableCollection(name: string, initialModeName?: string): Promise<Record<string, unknown>> {
853
- await this.ensureVariablesApi();
854
- return this.send("createVariableCollection", { name, initialModeName });
855
- }
856
-
857
- async createVariable(
858
- name: string,
859
- collectionId: string,
860
- resolvedType: string,
861
- valuesByMode?: Record<string, unknown>,
862
- description?: string,
863
- scopes?: string[],
864
- codeSyntax?: Record<string, string>
865
- ): Promise<Record<string, unknown>> {
866
- await this.ensureVariablesApi();
867
- return this.send("createVariable", { name, collectionId, resolvedType, valuesByMode, description, scopes, codeSyntax });
868
- }
869
-
870
- /**
871
- * Set variable scoping (which properties this variable can be applied to).
872
- * Scopes: ALL_SCOPES, ALL_FILLS, FRAME_FILL, SHAPE_FILL, TEXT_FILL, STROKE_COLOR,
873
- * EFFECT_COLOR, WIDTH_HEIGHT, GAP, CORNER_RADIUS, OPACITY, STROKE_FLOAT,
874
- * EFFECT_FLOAT, FONT_SIZE, LINE_HEIGHT, LETTER_SPACING, PARAGRAPH_SPACING,
875
- * PARAGRAPH_INDENT, FONT_WEIGHT, FONT_FAMILY, FONT_STYLE, TEXT_CONTENT
876
- */
877
- async setVariableScopes(variableId: string, scopes: string[]): Promise<Record<string, unknown>> {
878
- await this.ensureVariablesApi();
879
- return this.send("setVariableScopes", { variableId, scopes });
880
- }
881
-
882
- /**
883
- * Set code syntax for a variable (platform-specific code identifiers).
884
- * e.g. { WEB: "--color-brand-500", ANDROID: "colorBrand500", iOS: "Color.brand500" }
885
- */
886
- async setVariableCodeSyntax(variableId: string, codeSyntax: Record<string, string>): Promise<Record<string, unknown>> {
887
- await this.ensureVariablesApi();
888
- return this.send("setVariableCodeSyntax", { variableId, codeSyntax });
889
- }
890
-
891
- /**
892
- * Set variable description.
893
- */
894
- async setVariableDescription(variableId: string, description: string): Promise<Record<string, unknown>> {
895
- await this.ensureVariablesApi();
896
- return this.send("setVariableDescription", { variableId, description });
897
- }
898
-
899
- async updateVariable(variableId: string, modeId: string, value: unknown): Promise<Record<string, unknown>> {
900
- await this.ensureVariablesApi();
901
- return this.send("updateVariable", { variableId, modeId, value });
902
- }
903
-
904
- async deleteVariable(variableId: string): Promise<Record<string, unknown>> {
905
- await this.ensureVariablesApi();
906
- return this.send("deleteVariable", { variableId });
907
- }
908
-
909
- async renameVariable(variableId: string, newName: string): Promise<Record<string, unknown>> {
910
- await this.ensureVariablesApi();
911
- return this.send("renameVariable", { variableId, newName });
912
- }
913
-
914
- async deleteVariableCollection(collectionId: string): Promise<Record<string, unknown>> {
915
- await this.ensureVariablesApi();
916
- return this.send("deleteVariableCollection", { collectionId });
917
- }
918
-
919
- async addMode(collectionId: string, modeName: string): Promise<Record<string, unknown>> {
920
- await this.ensureVariablesApi();
921
- return this.send("addMode", { collectionId, modeName });
922
- }
923
-
924
- async renameMode(collectionId: string, modeId: string, newName: string): Promise<Record<string, unknown>> {
925
- await this.ensureVariablesApi();
926
- return this.send("renameMode", { collectionId, modeId, newName });
927
- }
928
-
929
- async batchCreateVariables(
930
- variables: Array<{
931
- name: string;
932
- collectionId: string;
933
- resolvedType: string;
934
- valuesByMode?: Record<string, unknown>;
935
- description?: string;
936
- scopes?: string[];
937
- codeSyntax?: Record<string, string>;
938
- }>
939
- ): Promise<Record<string, unknown>> {
940
- await this.ensureVariablesApi();
941
- return this.send("batchCreateVariables", { variables });
942
- }
943
-
944
- async batchUpdateVariables(
945
- updates: Array<{ variableId: string; modeId: string; value: unknown }>
946
- ): Promise<Record<string, unknown>> {
947
- await this.ensureVariablesApi();
948
- return this.send("batchUpdateVariables", { updates });
949
- }
950
-
951
- // ─── Node Operations ─────────────────────────────────────────────────────
952
-
953
- async cloneNode(nodeId: string, x?: number, y?: number): Promise<Record<string, unknown>> {
954
- return this.send("cloneNode", { nodeId, x, y });
955
- }
956
-
957
- async deleteNode(nodeId: string): Promise<Record<string, unknown>> {
958
- return this.send("deleteNode", { nodeId });
959
- }
960
-
961
- async moveNode(nodeId: string, x?: number, y?: number, parentId?: string): Promise<Record<string, unknown>> {
962
- return this.send("moveNode", { nodeId, x, y, parentId });
963
- }
964
-
965
- async resizeNode(nodeId: string, width: number, height: number): Promise<Record<string, unknown>> {
966
- return this.send("resizeNode", { nodeId, width, height });
967
- }
968
-
969
- async renameNode(nodeId: string, newName: string): Promise<Record<string, unknown>> {
970
- return this.send("renameNode", { nodeId, newName });
971
- }
972
-
973
- async setFills(nodeId: string, fills: unknown[]): Promise<Record<string, unknown>> {
974
- return this.send("setFills", { nodeId, fills });
975
- }
976
-
977
- async setStrokes(nodeId: string, strokes: unknown[], strokeWeight?: number): Promise<Record<string, unknown>> {
978
- return this.send("setStrokes", { nodeId, strokes, strokeWeight });
979
- }
980
-
981
- async setText(nodeId: string, characters: string, fontSize?: number): Promise<Record<string, unknown>> {
982
- return this.send("setText", { nodeId, characters, fontSize });
983
- }
984
-
985
- // ─── Component Operations ────────────────────────────────────────────────
986
-
987
- async searchComponents(query: string, limit?: number): Promise<Array<Record<string, unknown>>> {
988
- return this.send("searchComponents", { query, limit });
989
- }
990
-
991
- async instantiateComponent(
992
- nodeId: string,
993
- variant?: Record<string, string>,
994
- x?: number,
995
- y?: number,
996
- parentId?: string
997
- ): Promise<Record<string, unknown>> {
998
- return this.send("instantiateComponent", { nodeId, variant, x, y, parentId });
999
- }
1000
-
1001
- async setDescription(nodeId: string, description: string): Promise<Record<string, unknown>> {
1002
- return this.send("setDescription", { nodeId, description });
1003
- }
1004
-
1005
- // ─── Variable Binding ───────────────────────────────────────────────────
1006
-
1007
- async bindVariables(
1008
- bindings: Array<{ nodeId: string; field: string; variableId: string; fillIndex?: number }>
1009
- ): Promise<{ bound: number; total: number }> {
1010
- if (bindings.length === 0) return { bound: 0, total: 0 };
1011
-
1012
- await this.ensureVariablesApi();
1013
-
1014
- const script = `
1015
- (async () => {
1016
- const bindings = ${JSON.stringify(bindings)};
1017
- let bound = 0;
1018
-
1019
- for (const b of bindings) {
1020
- const node = await figma.getNodeByIdAsync(b.nodeId);
1021
- if (!node) continue;
1022
-
1023
- const variable = await figma.variables.getVariableByIdAsync(b.variableId);
1024
- if (!variable) continue;
1025
-
1026
- if (b.field === 'fills' || b.field === 'strokes') {
1027
- const idx = b.fillIndex ?? 0;
1028
- if (node[b.field] && node[b.field].length > idx) {
1029
- const paints = [...node[b.field]];
1030
- paints[idx] = figma.variables.setBoundVariableForPaint(paints[idx], 'color', variable);
1031
- node[b.field] = paints;
1032
- bound++;
1033
- }
1034
- } else {
1035
- try {
1036
- node.setBoundVariable(b.field, variable.id);
1037
- bound++;
1038
- } catch (e) { /* skip unsupported fields */ }
1039
- }
1040
- }
1041
-
1042
- return { bound, total: bindings.length };
1043
- })();
1044
- `.trim();
1045
-
1046
- const result = await this.execute(script);
1047
- if (!result.success) {
1048
- return { bound: 0, total: bindings.length };
1049
- }
1050
- return result.result as { bound: number; total: number };
1051
- }
1052
-
1053
- // ─── Design System Extraction ────────────────────────────────────────────
1054
-
1055
- async getVariables(collectionId?: string, verbosity?: string): Promise<unknown[]> {
1056
- await this.ensureVariablesApi();
1057
- return this.send("getVariables", { collectionId, verbosity });
1058
- }
1059
-
1060
- async getStyles(): Promise<Record<string, unknown>> {
1061
- return this.send("getStyles");
1062
- }
1063
-
1064
- // ─── Deep Node Serialization & Batch Read ──────────────────────────────
1065
-
1066
- /**
1067
- * Get a node with full recursive child data up to maxDepth.
1068
- * Unlike getNode() which returns 1-level children as {id, name, type},
1069
- * this returns the full property set for every descendant.
1070
- */
1071
- async getNodeDeep(nodeId: string, maxDepth: number = 10): Promise<unknown> {
1072
- const result = await this.execute(`
1073
- (async () => {
1074
- const maxDepth = ${Math.min(maxDepth, 20)};
1075
-
1076
- function serializeNode(node, depth) {
1077
- const data = {
1078
- id: node.id,
1079
- name: node.name,
1080
- type: node.type,
1081
- visible: node.visible !== false,
1082
- };
1083
-
1084
- // Geometry
1085
- if ('x' in node) { data.x = node.x; data.y = node.y; }
1086
- if ('width' in node) { data.width = node.width; data.height = node.height; }
1087
- if ('absoluteBoundingBox' in node) data.absoluteBoundingBox = node.absoluteBoundingBox;
1088
- if ('absoluteRenderBounds' in node) data.absoluteRenderBounds = node.absoluteRenderBounds;
1089
-
1090
- // Visual (fills/strokes/effects can be figma.mixed Symbol — guard with try-catch)
1091
- try { if ('fills' in node && node.fills && typeof node.fills !== 'symbol') data.fills = JSON.parse(JSON.stringify(node.fills)); } catch (e) {}
1092
- try { if ('strokes' in node && node.strokes && typeof node.strokes !== 'symbol') data.strokes = JSON.parse(JSON.stringify(node.strokes)); } catch (e) {}
1093
- try { if ('effects' in node && node.effects && typeof node.effects !== 'symbol') data.effects = JSON.parse(JSON.stringify(node.effects)); } catch (e) {}
1094
- if ('opacity' in node) data.opacity = node.opacity;
1095
- try { if ('cornerRadius' in node && typeof node.cornerRadius !== 'symbol') data.cornerRadius = node.cornerRadius; } catch (e) {}
1096
- try { if ('strokeWeight' in node && typeof node.strokeWeight !== 'symbol') data.strokeWeight = node.strokeWeight; } catch (e) {}
1097
-
1098
- // Layout
1099
- if ('layoutMode' in node) {
1100
- data.layoutMode = node.layoutMode;
1101
- data.primaryAxisSizingMode = node.primaryAxisSizingMode;
1102
- data.counterAxisSizingMode = node.counterAxisSizingMode;
1103
- data.paddingTop = node.paddingTop;
1104
- data.paddingRight = node.paddingRight;
1105
- data.paddingBottom = node.paddingBottom;
1106
- data.paddingLeft = node.paddingLeft;
1107
- data.itemSpacing = node.itemSpacing;
1108
- if (node.primaryAxisAlignItems) data.primaryAxisAlignItems = node.primaryAxisAlignItems;
1109
- if (node.counterAxisAlignItems) data.counterAxisAlignItems = node.counterAxisAlignItems;
1110
- }
1111
-
1112
- // Text (all text properties can be figma.mixed Symbol for mixed-style text)
1113
- if (node.type === 'TEXT') {
1114
- data.characters = node.characters;
1115
- try {
1116
- if (typeof node.fontSize !== 'symbol') data.fontSize = node.fontSize;
1117
- if (typeof node.fontName !== 'symbol') data.fontName = JSON.parse(JSON.stringify(node.fontName));
1118
- if (typeof node.lineHeight !== 'symbol') data.lineHeight = node.lineHeight;
1119
- if (typeof node.letterSpacing !== 'symbol') data.letterSpacing = node.letterSpacing;
1120
- if (typeof node.textAlignHorizontal !== 'symbol') data.textAlignHorizontal = node.textAlignHorizontal;
1121
- if (typeof node.textAlignVertical !== 'symbol') data.textAlignVertical = node.textAlignVertical;
1122
- } catch (e) { /* mixed styles */ }
1123
- }
1124
-
1125
- // Component metadata
1126
- if ('componentPropertyReferences' in node) data.componentPropertyReferences = node.componentPropertyReferences;
1127
- if ('variantProperties' in node && node.variantProperties) data.variantProperties = node.variantProperties;
1128
- if (node.type === 'INSTANCE' && node.mainComponent) {
1129
- data.mainComponentId = node.mainComponent.id;
1130
- data.mainComponentName = node.mainComponent.name;
1131
- }
1132
- if (node.description) data.description = node.description;
1133
-
1134
- // Variable bindings
1135
- if ('boundVariables' in node && node.boundVariables) {
1136
- try {
1137
- const bv = {};
1138
- for (const [prop, binding] of Object.entries(node.boundVariables)) {
1139
- if (binding && typeof binding === 'object' && 'id' in binding) {
1140
- bv[prop] = { id: binding.id, type: binding.type };
1141
- } else if (Array.isArray(binding)) {
1142
- bv[prop] = binding.map(b => b && typeof b === 'object' && 'id' in b ? { id: b.id } : null).filter(Boolean);
1143
- }
1144
- }
1145
- if (Object.keys(bv).length > 0) data.boundVariables = bv;
1146
- } catch (e) { /* skip */ }
1147
- }
1148
-
1149
- // Recursive children
1150
- if ('children' in node && node.children && depth < maxDepth) {
1151
- data.children = [];
1152
- for (const child of node.children) {
1153
- data.children.push(serializeNode(child, depth + 1));
1154
- }
1155
- data.childCount = node.children.length;
1156
- } else if ('children' in node && node.children) {
1157
- data.childCount = node.children.length;
1158
- data.children = node.children.map(c => ({ id: c.id, name: c.name, type: c.type }));
1159
- data.truncatedAtDepth = depth;
1160
- }
1161
-
1162
- return data;
1163
- }
1164
-
1165
- const root = await figma.getNodeByIdAsync(${JSON.stringify(nodeId)});
1166
- if (!root) throw new Error('Node not found: ' + ${JSON.stringify(nodeId)});
1167
- return serializeNode(root, 0);
1168
- })();
1169
- `, 60000); // 60s timeout for deep trees
1170
-
1171
- if (!result.success) throw new Error(result.error);
1172
- return result.result;
1173
- }
1174
-
1175
- /**
1176
- * Batch read multiple nodes in a single round-trip.
1177
- * Returns a map of nodeId → serialized node data (1-level deep children).
1178
- */
1179
- async batchGetNodes(nodeIds: string[], includeChildren: boolean = true): Promise<Record<string, unknown>> {
1180
- if (nodeIds.length === 0) return {};
1181
-
1182
- const result = await this.execute(`
1183
- (async () => {
1184
- const ids = ${JSON.stringify(nodeIds.slice(0, 200))};
1185
- const includeChildren = ${includeChildren};
1186
- const results = {};
1187
-
1188
- for (const id of ids) {
1189
- const node = await figma.getNodeByIdAsync(id);
1190
- if (!node) { results[id] = null; continue; }
1191
-
1192
- const data = {
1193
- id: node.id,
1194
- name: node.name,
1195
- type: node.type,
1196
- visible: node.visible !== false,
1197
- };
1198
-
1199
- if ('x' in node) { data.x = node.x; data.y = node.y; }
1200
- if ('width' in node) { data.width = node.width; data.height = node.height; }
1201
- if ('fills' in node && node.fills) data.fills = JSON.parse(JSON.stringify(node.fills));
1202
- if ('strokes' in node && node.strokes) data.strokes = JSON.parse(JSON.stringify(node.strokes));
1203
- if ('effects' in node && node.effects) data.effects = JSON.parse(JSON.stringify(node.effects));
1204
- if ('opacity' in node) data.opacity = node.opacity;
1205
- if ('cornerRadius' in node) data.cornerRadius = node.cornerRadius;
1206
-
1207
- if ('layoutMode' in node) {
1208
- data.layoutMode = node.layoutMode;
1209
- data.paddingTop = node.paddingTop;
1210
- data.paddingRight = node.paddingRight;
1211
- data.paddingBottom = node.paddingBottom;
1212
- data.paddingLeft = node.paddingLeft;
1213
- data.itemSpacing = node.itemSpacing;
1214
- }
1215
-
1216
- if (node.type === 'TEXT') {
1217
- data.characters = node.characters;
1218
- try {
1219
- data.fontSize = node.fontSize;
1220
- data.fontName = JSON.parse(JSON.stringify(node.fontName));
1221
- } catch (e) {}
1222
- }
1223
-
1224
- if (includeChildren && 'children' in node && node.children) {
1225
- data.childCount = node.children.length;
1226
- data.children = node.children.map(c => ({
1227
- id: c.id,
1228
- name: c.name,
1229
- type: c.type,
1230
- visible: c.visible !== false,
1231
- x: 'x' in c ? c.x : undefined,
1232
- y: 'y' in c ? c.y : undefined,
1233
- width: 'width' in c ? c.width : undefined,
1234
- height: 'height' in c ? c.height : undefined,
1235
- }));
1236
- }
1237
-
1238
- results[id] = data;
1239
- }
1240
-
1241
- return results;
1242
- })();
1243
- `, 60000);
1244
-
1245
- if (!result.success) throw new Error(result.error);
1246
- return result.result as Record<string, unknown>;
1247
- }
1248
-
1249
- // ─── Multi-Mode Variable Binding ──────────────────────────────────────────
1250
-
1251
- /**
1252
- * Bind semantic variables to nodes AND set the explicit variable mode on a
1253
- * container frame. This is the key method for theme-switching support.
1254
- *
1255
- * Unlike bindVariables() which just binds variables without mode awareness,
1256
- * this method:
1257
- * 1. Binds semantic variables (which have Light/Dark mode values)
1258
- * 2. Sets the explicit mode on the target frame so children resolve correctly
1259
- */
1260
- async bindVariablesMultiMode(
1261
- bindings: Array<{ nodeId: string; field: string; variableId: string; fillIndex?: number }>,
1262
- targetFrameId: string,
1263
- collectionId: string,
1264
- activeModeId: string
1265
- ): Promise<{ bound: number; total: number; modeSet: boolean; errors?: string[] }> {
1266
- if (bindings.length === 0 && !targetFrameId) {
1267
- return { bound: 0, total: 0, modeSet: false };
1268
- }
1269
-
1270
- await this.ensureVariablesApi();
1271
-
1272
- // Import from token-binder for script generation
1273
- const { buildMultiModeBindingScript } = await import("./token-binder.js");
1274
- const script = buildMultiModeBindingScript(
1275
- bindings.map((b) => ({
1276
- nodeId: b.nodeId,
1277
- field: b.field,
1278
- fillIndex: b.fillIndex,
1279
- semanticVariableId: b.variableId,
1280
- })),
1281
- targetFrameId,
1282
- collectionId,
1283
- activeModeId
1284
- );
1285
-
1286
- const result = await this.execute(script);
1287
- if (!result.success) {
1288
- return { bound: 0, total: bindings.length, modeSet: false, errors: [result.error ?? "Unknown error"] };
1289
- }
1290
- return result.result as { bound: number; total: number; modeSet: boolean; errors?: string[] };
1291
- }
1292
-
1293
- /**
1294
- * Switch a frame's variable mode (theme switching).
1295
- * All children with bound variables will resolve to the new mode's values.
1296
- */
1297
- async switchMode(
1298
- frameId: string,
1299
- collectionId: string,
1300
- modeId: string
1301
- ): Promise<{ success: boolean; error?: string }> {
1302
- await this.ensureVariablesApi();
1303
- const { buildModeSwitchScript } = await import("./token-binder.js");
1304
- const script = buildModeSwitchScript(frameId, collectionId, modeId);
1305
- const result = await this.execute(script);
1306
- if (!result.success) {
1307
- return { success: false, error: result.error };
1308
- }
1309
- return result.result as { success: boolean; error?: string };
1310
- }
1311
-
1312
- /**
1313
- * List all modes for a variable collection.
1314
- * Returns mode IDs and names so callers can pick one for switchMode().
1315
- */
1316
- async listModes(collectionId: string): Promise<unknown> {
1317
- await this.ensureVariablesApi();
1318
- const { buildListModesScript } = await import("./token-binder.js");
1319
- const script = buildListModesScript(collectionId);
1320
- const result = await this.execute(script);
1321
- if (!result.success) throw new Error(result.error);
1322
- return result.result;
1323
- }
1324
-
1325
- // ─── P0: Enrichment & Compression ─────────────────────────────────────
1326
-
1327
- /**
1328
- * Get enriched design system data — tokens organized semantically,
1329
- * components categorized, relationships mapped. Cached for 5 minutes.
1330
- */
1331
- async getEnrichedDesignSystem(): Promise<EnrichedDesignSystem> {
1332
- return this.cache.getEnrichedDesignSystem();
1333
- }
1334
-
1335
- /**
1336
- * Get a node with resolved styles — fills as hex, typography categorized,
1337
- * spacing grid-snapped, radius categorized.
1338
- */
1339
- async getNodeEnriched(nodeId: string): Promise<{ node: FigmaNode; styles: ResolvedStyle }> {
1340
- const [node, tokens] = await Promise.all([
1341
- this.cache.getNode(nodeId),
1342
- this.cache.getTokens(),
1343
- ]);
1344
- const styles = resolveStyles(node, tokens);
1345
- return { node, styles };
1346
- }
1347
-
1348
- /**
1349
- * Compress any response payload to fit within AI context window limits.
1350
- * Automatically selects compression tier based on byte size.
1351
- */
1352
- compressForAI(data: unknown, forceTier?: CompressionTier): CompressedResponse {
1353
- return compressResponse(data, { forceTier });
1354
- }
1355
-
1356
- // ─── Create Child Node ───────────────────────────────────────────────────
1357
-
1358
- async createChild(
1359
- childType: string,
1360
- parentId?: string,
1361
- name?: string,
1362
- width?: number,
1363
- height?: number,
1364
- x?: number,
1365
- y?: number,
1366
- characters?: string
1367
- ): Promise<Record<string, unknown>> {
1368
- return this.send("createChild", { childType, parentId, name, width, height, x, y, characters });
1369
- }
1370
-
1371
- // ─── Swarm Agent Cursor Methods ──────────────────────────────────────────
1372
-
1373
- async spawnAgentCursor(agentId: string, x: number, y: number): Promise<void> {
1374
- await this.send("spawnAgentCursor", { agentId, x, y });
1375
- }
1376
-
1377
- async moveAgentCursor(agentId: string, x: number, y: number, animate = true, durationMs = 250): Promise<void> {
1378
- await this.send("moveAgentCursor", { agentId, x, y, animate, durationMs });
1379
- }
1380
-
1381
- async updateAgentLabel(agentId: string, label: string): Promise<void> {
1382
- await this.send("updateAgentLabel", { agentId, label });
1383
- }
1384
-
1385
- async removeAgentCursor(agentId: string): Promise<void> {
1386
- await this.send("removeAgentCursor", { agentId });
1387
- }
1388
-
1389
- async postAgentChat(agentId: string, message: string, x: number, y: number): Promise<string> {
1390
- const result = await this.send<{ noteId: string }>("agentChat", { agentId, message, x, y });
1391
- return result.noteId;
1392
- }
1393
-
1394
- async cleanupSwarm(): Promise<void> {
1395
- await this.send("cleanupAgentCursors", {});
1396
- }
1397
- }
1398
-
1399
- // Singleton bridge instance
1400
- let bridgeInstance: FigmaBridge | null = null;
1401
-
1402
- export async function getBridge(): Promise<FigmaBridge> {
1403
- if (!bridgeInstance) {
1404
- bridgeInstance = new FigmaBridge();
1405
- }
1406
- await bridgeInstance.connect();
1407
- return bridgeInstance;
1408
- }