figma-console-mcp 0.1.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 (118) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +328 -0
  3. package/dist/browser/base.d.ts +50 -0
  4. package/dist/browser/base.d.ts.map +1 -0
  5. package/dist/browser/base.js +6 -0
  6. package/dist/browser/base.js.map +1 -0
  7. package/dist/browser/local.d.ts +66 -0
  8. package/dist/browser/local.d.ts.map +1 -0
  9. package/dist/browser/local.js +223 -0
  10. package/dist/browser/local.js.map +1 -0
  11. package/dist/cloudflare/browser/base.js +5 -0
  12. package/dist/cloudflare/browser/cloudflare.js +156 -0
  13. package/dist/cloudflare/browser-manager.js +157 -0
  14. package/dist/cloudflare/core/config.js +161 -0
  15. package/dist/cloudflare/core/console-monitor.js +382 -0
  16. package/dist/cloudflare/core/enrichment/enrichment-service.js +272 -0
  17. package/dist/cloudflare/core/enrichment/index.js +7 -0
  18. package/dist/cloudflare/core/enrichment/relationship-mapper.js +351 -0
  19. package/dist/cloudflare/core/enrichment/style-resolver.js +326 -0
  20. package/dist/cloudflare/core/figma-api.js +273 -0
  21. package/dist/cloudflare/core/figma-desktop-connector.js +383 -0
  22. package/dist/cloudflare/core/figma-style-extractor.js +311 -0
  23. package/dist/cloudflare/core/figma-tools.js +2299 -0
  24. package/dist/cloudflare/core/logger.js +53 -0
  25. package/dist/cloudflare/core/snippet-injector.js +96 -0
  26. package/dist/cloudflare/core/types/enriched.js +5 -0
  27. package/dist/cloudflare/core/types/index.js +4 -0
  28. package/dist/cloudflare/index.js +1059 -0
  29. package/dist/cloudflare/test-browser.js +88 -0
  30. package/dist/config.d.ts +17 -0
  31. package/dist/config.d.ts.map +1 -0
  32. package/dist/config.js +141 -0
  33. package/dist/config.js.map +1 -0
  34. package/dist/core/config.d.ts +17 -0
  35. package/dist/core/config.d.ts.map +1 -0
  36. package/dist/core/config.js +162 -0
  37. package/dist/core/config.js.map +1 -0
  38. package/dist/core/console-monitor.d.ts +81 -0
  39. package/dist/core/console-monitor.d.ts.map +1 -0
  40. package/dist/core/console-monitor.js +383 -0
  41. package/dist/core/console-monitor.js.map +1 -0
  42. package/dist/core/enrichment/enrichment-service.d.ts +52 -0
  43. package/dist/core/enrichment/enrichment-service.d.ts.map +1 -0
  44. package/dist/core/enrichment/enrichment-service.js +273 -0
  45. package/dist/core/enrichment/enrichment-service.js.map +1 -0
  46. package/dist/core/enrichment/index.d.ts +8 -0
  47. package/dist/core/enrichment/index.d.ts.map +1 -0
  48. package/dist/core/enrichment/index.js +8 -0
  49. package/dist/core/enrichment/index.js.map +1 -0
  50. package/dist/core/enrichment/relationship-mapper.d.ts +106 -0
  51. package/dist/core/enrichment/relationship-mapper.d.ts.map +1 -0
  52. package/dist/core/enrichment/relationship-mapper.js +352 -0
  53. package/dist/core/enrichment/relationship-mapper.js.map +1 -0
  54. package/dist/core/enrichment/style-resolver.d.ts +80 -0
  55. package/dist/core/enrichment/style-resolver.d.ts.map +1 -0
  56. package/dist/core/enrichment/style-resolver.js +327 -0
  57. package/dist/core/enrichment/style-resolver.js.map +1 -0
  58. package/dist/core/figma-api.d.ts +137 -0
  59. package/dist/core/figma-api.d.ts.map +1 -0
  60. package/dist/core/figma-api.js +274 -0
  61. package/dist/core/figma-api.js.map +1 -0
  62. package/dist/core/figma-desktop-connector.d.ts +52 -0
  63. package/dist/core/figma-desktop-connector.d.ts.map +1 -0
  64. package/dist/core/figma-desktop-connector.js +384 -0
  65. package/dist/core/figma-desktop-connector.js.map +1 -0
  66. package/dist/core/figma-style-extractor.d.ts +76 -0
  67. package/dist/core/figma-style-extractor.d.ts.map +1 -0
  68. package/dist/core/figma-style-extractor.js +312 -0
  69. package/dist/core/figma-style-extractor.js.map +1 -0
  70. package/dist/core/figma-tools.d.ts +15 -0
  71. package/dist/core/figma-tools.d.ts.map +1 -0
  72. package/dist/core/figma-tools.js +2300 -0
  73. package/dist/core/figma-tools.js.map +1 -0
  74. package/dist/core/logger.d.ts +22 -0
  75. package/dist/core/logger.d.ts.map +1 -0
  76. package/dist/core/logger.js +54 -0
  77. package/dist/core/logger.js.map +1 -0
  78. package/dist/core/snippet-injector.d.ts +24 -0
  79. package/dist/core/snippet-injector.d.ts.map +1 -0
  80. package/dist/core/snippet-injector.js +97 -0
  81. package/dist/core/snippet-injector.js.map +1 -0
  82. package/dist/core/types/enriched.d.ts +213 -0
  83. package/dist/core/types/enriched.d.ts.map +1 -0
  84. package/dist/core/types/enriched.js +6 -0
  85. package/dist/core/types/enriched.js.map +1 -0
  86. package/dist/core/types/index.d.ts +112 -0
  87. package/dist/core/types/index.d.ts.map +1 -0
  88. package/dist/core/types/index.js +5 -0
  89. package/dist/core/types/index.js.map +1 -0
  90. package/dist/index.d.ts +8 -0
  91. package/dist/index.d.ts.map +1 -0
  92. package/dist/index.js +72 -0
  93. package/dist/index.js.map +1 -0
  94. package/dist/local.d.ts +57 -0
  95. package/dist/local.d.ts.map +1 -0
  96. package/dist/local.js +668 -0
  97. package/dist/local.js.map +1 -0
  98. package/dist/logger.d.ts +22 -0
  99. package/dist/logger.d.ts.map +1 -0
  100. package/dist/logger.js +45 -0
  101. package/dist/logger.js.map +1 -0
  102. package/dist/server.d.ts +40 -0
  103. package/dist/server.d.ts.map +1 -0
  104. package/dist/server.js +99 -0
  105. package/dist/server.js.map +1 -0
  106. package/dist/tools/index.d.ts +15 -0
  107. package/dist/tools/index.d.ts.map +1 -0
  108. package/dist/tools/index.js +184 -0
  109. package/dist/tools/index.js.map +1 -0
  110. package/dist/types/index.d.ts +102 -0
  111. package/dist/types/index.d.ts.map +1 -0
  112. package/dist/types/index.js +6 -0
  113. package/dist/types/index.js.map +1 -0
  114. package/figma-desktop-bridge/README.md +232 -0
  115. package/figma-desktop-bridge/code.js +133 -0
  116. package/figma-desktop-bridge/manifest.json +13 -0
  117. package/figma-desktop-bridge/ui.html +200 -0
  118. package/package.json +77 -0
@@ -0,0 +1,273 @@
1
+ /**
2
+ * Enrichment Service
3
+ * Coordinates enrichment of Figma API responses
4
+ */
5
+ import { StyleValueResolver } from "./style-resolver.js";
6
+ import { RelationshipMapper } from "./relationship-mapper.js";
7
+ export class EnrichmentService {
8
+ constructor(logger) {
9
+ // Caches
10
+ this.fileDataCache = new Map();
11
+ this.lastEnrichmentTime = new Map();
12
+ this.CACHE_TTL_MS = 5 * 60 * 1000; // 5 minutes
13
+ this.logger = logger;
14
+ this.styleResolver = new StyleValueResolver(logger);
15
+ this.relationshipMapper = new RelationshipMapper(logger);
16
+ }
17
+ /**
18
+ * Enrich styles response
19
+ */
20
+ async enrichStyles(styles, fileKey, options = {}, fileData) {
21
+ if (!options.enrich) {
22
+ return styles; // Return as-is if enrichment not requested
23
+ }
24
+ this.logger.info({ fileKey, count: styles.length }, "Enriching styles");
25
+ try {
26
+ // Get file data for relationships (use provided data if available)
27
+ const data = fileData || await this.getFileDataForEnrichment(fileKey);
28
+ const variables = this.extractVariablesMap(data);
29
+ // Build relationships if requested
30
+ if (options.include_usage) {
31
+ await this.relationshipMapper.buildRelationships(data);
32
+ }
33
+ const enrichedStyles = [];
34
+ for (const style of styles) {
35
+ const enriched = {
36
+ ...style,
37
+ };
38
+ // Resolve value and variable reference
39
+ if (options.include_exports !== false) {
40
+ const { value, variableRef } = await this.styleResolver.resolveStyleValue(style, variables, options.max_depth);
41
+ enriched.resolved_value = value;
42
+ enriched.variable_reference = variableRef;
43
+ // Generate export formats
44
+ if (value && options.export_formats) {
45
+ enriched.export_formats = this.styleResolver.generateExportFormats(style.name, value, style.style_type, options.export_formats);
46
+ }
47
+ }
48
+ // Add usage information
49
+ if (options.include_usage) {
50
+ const styleId = style.node_id || style.key || style.id;
51
+ enriched.used_in_components =
52
+ this.relationshipMapper.getComponentsByStyle(styleId);
53
+ enriched.usage_count =
54
+ this.relationshipMapper.getStyleUsageCount(styleId);
55
+ }
56
+ enrichedStyles.push(enriched);
57
+ }
58
+ this.logger.info({ enrichedCount: enrichedStyles.length }, "Styles enrichment complete");
59
+ return enrichedStyles;
60
+ }
61
+ catch (error) {
62
+ this.logger.error({ error }, "Failed to enrich styles");
63
+ return styles; // Return original on error
64
+ }
65
+ }
66
+ /**
67
+ * Enrich variables response
68
+ */
69
+ async enrichVariables(variables, fileKey, options = {}, fileData) {
70
+ if (!options.enrich) {
71
+ return variables;
72
+ }
73
+ this.logger.info({ fileKey, count: variables.length }, "Enriching variables");
74
+ try {
75
+ const data = fileData || await this.getFileDataForEnrichment(fileKey);
76
+ const variablesMap = this.extractVariablesMap(data);
77
+ // Build relationships if requested
78
+ if (options.include_usage) {
79
+ await this.relationshipMapper.buildRelationships(data);
80
+ }
81
+ const enrichedVars = [];
82
+ for (const variable of variables) {
83
+ const enriched = {
84
+ ...variable,
85
+ };
86
+ // Resolve values for all modes
87
+ if (options.include_exports !== false) {
88
+ const resolved_values = {};
89
+ for (const [modeId, value] of Object.entries(variable.valuesByMode || {})) {
90
+ const resolvedValue = await this.styleResolver.resolveVariableValue(variable, variablesMap, options.max_depth);
91
+ resolved_values[modeId] = resolvedValue;
92
+ }
93
+ enriched.resolved_values = resolved_values;
94
+ // Generate export formats using first mode
95
+ const firstModeValue = Object.values(resolved_values)[0];
96
+ if (firstModeValue && options.export_formats) {
97
+ enriched.export_formats = this.styleResolver.generateExportFormats(variable.name, firstModeValue, variable.resolvedType, options.export_formats);
98
+ }
99
+ }
100
+ // Add usage information
101
+ if (options.include_usage) {
102
+ enriched.used_in_styles =
103
+ this.relationshipMapper.getStylesByVariable(variable.id);
104
+ enriched.used_in_components =
105
+ this.relationshipMapper.getComponentsByVariable(variable.id);
106
+ enriched.usage_count =
107
+ this.relationshipMapper.getVariableUsageCount(variable.id);
108
+ }
109
+ // Add dependencies
110
+ if (options.include_dependencies) {
111
+ enriched.dependencies =
112
+ this.relationshipMapper.getVariableDependencies(variable.id);
113
+ }
114
+ enrichedVars.push(enriched);
115
+ }
116
+ this.logger.info({ enrichedCount: enrichedVars.length }, "Variables enrichment complete");
117
+ return enrichedVars;
118
+ }
119
+ catch (error) {
120
+ this.logger.error({ error }, "Failed to enrich variables");
121
+ return variables;
122
+ }
123
+ }
124
+ /**
125
+ * Enrich component response
126
+ */
127
+ async enrichComponent(component, fileKey, options = {}, fileData) {
128
+ if (!options.enrich) {
129
+ return component;
130
+ }
131
+ this.logger.info({ fileKey, componentId: component.id }, "Enriching component");
132
+ try {
133
+ const enriched = {
134
+ ...component,
135
+ };
136
+ const data = fileData || await this.getFileDataForEnrichment(fileKey);
137
+ // Build relationships
138
+ await this.relationshipMapper.buildRelationships(data);
139
+ // Extract styles used
140
+ if (component.styles) {
141
+ const stylesUsed = [];
142
+ for (const [prop, styleId] of Object.entries(component.styles)) {
143
+ const style = data.styles?.[styleId];
144
+ if (style) {
145
+ stylesUsed.push({
146
+ style_id: styleId,
147
+ style_name: style.name,
148
+ style_type: style.style_type,
149
+ property: prop,
150
+ });
151
+ }
152
+ }
153
+ enriched.styles_used = stylesUsed;
154
+ }
155
+ // Extract variables used
156
+ if (component.boundVariables) {
157
+ const varsUsed = [];
158
+ // This would need actual variable resolution
159
+ enriched.variables_used = varsUsed;
160
+ }
161
+ // Detect hardcoded values (simplified for now)
162
+ // TODO: Implement full hardcoded value detection
163
+ enriched.hardcoded_values = [];
164
+ // Calculate token coverage
165
+ const totalProps = (enriched.styles_used?.length || 0) + (enriched.variables_used?.length || 0) + (enriched.hardcoded_values?.length || 0);
166
+ const tokenProps = (enriched.styles_used?.length || 0) + (enriched.variables_used?.length || 0);
167
+ enriched.token_coverage =
168
+ totalProps > 0 ? Math.round((tokenProps / totalProps) * 100) : 0;
169
+ this.logger.info({ componentId: component.id, coverage: enriched.token_coverage }, "Component enrichment complete");
170
+ return enriched;
171
+ }
172
+ catch (error) {
173
+ this.logger.error({ error }, "Failed to enrich component");
174
+ return component;
175
+ }
176
+ }
177
+ /**
178
+ * Enrich file data response
179
+ */
180
+ async enrichFileData(fileData, options = {}) {
181
+ if (!options.enrich) {
182
+ return fileData;
183
+ }
184
+ this.logger.info({ fileKey: fileData.fileKey }, "Enriching file data");
185
+ try {
186
+ const enriched = {
187
+ ...fileData,
188
+ };
189
+ // Build relationships for statistics
190
+ await this.relationshipMapper.buildRelationships(fileData);
191
+ // Calculate statistics
192
+ const styles = Object.values(fileData.styles || {});
193
+ const variables = Array.from(this.extractVariablesMap(fileData).values());
194
+ const unusedStyles = this.relationshipMapper.getUnusedStyles(styles);
195
+ const unusedVars = this.relationshipMapper.getUnusedVariables(variables);
196
+ enriched.statistics = {
197
+ total_variables: variables.length,
198
+ total_styles: styles.length,
199
+ total_components: Object.keys(fileData.components || {}).length,
200
+ unused_variables: unusedVars.length,
201
+ unused_styles: unusedStyles.length,
202
+ average_token_coverage: 0, // TODO: Calculate from components
203
+ total_hardcoded_values: 0, // TODO: Calculate from components
204
+ audit_issues_count: 0, // TODO: Calculate from audit
205
+ };
206
+ // Calculate health score (simple formula)
207
+ const usageRate = variables.length > 0
208
+ ? ((variables.length - unusedVars.length) / variables.length) * 100
209
+ : 100;
210
+ enriched.health_score = Math.round(usageRate);
211
+ this.logger.info({ health_score: enriched.health_score }, "File data enrichment complete");
212
+ return enriched;
213
+ }
214
+ catch (error) {
215
+ this.logger.error({ error }, "Failed to enrich file data");
216
+ return fileData;
217
+ }
218
+ }
219
+ /**
220
+ * Get file data for enrichment (with caching)
221
+ * NOTE: This is a placeholder that returns the data passed in
222
+ * In a full implementation, this would fetch fresh data from Figma API
223
+ */
224
+ async getFileDataForEnrichment(fileKey, providedData) {
225
+ const now = Date.now();
226
+ const lastEnrich = this.lastEnrichmentTime.get(fileKey) || 0;
227
+ // Return cached if fresh
228
+ if (this.fileDataCache.has(fileKey) &&
229
+ now - lastEnrich < this.CACHE_TTL_MS) {
230
+ return this.fileDataCache.get(fileKey);
231
+ }
232
+ // If data is provided, use it (from the tool that called us)
233
+ // This avoids redundant API calls since the tool already fetched the data
234
+ const fileData = providedData || {
235
+ styles: {},
236
+ variables: {},
237
+ document: {},
238
+ components: {},
239
+ };
240
+ this.fileDataCache.set(fileKey, fileData);
241
+ this.lastEnrichmentTime.set(fileKey, now);
242
+ return fileData;
243
+ }
244
+ /**
245
+ * Set file data in cache (called by tools that already have the data)
246
+ */
247
+ setFileDataCache(fileKey, fileData) {
248
+ this.fileDataCache.set(fileKey, fileData);
249
+ this.lastEnrichmentTime.set(fileKey, Date.now());
250
+ }
251
+ /**
252
+ * Extract variables as a Map for efficient lookup
253
+ */
254
+ extractVariablesMap(fileData) {
255
+ const variablesMap = new Map();
256
+ if (fileData.variables) {
257
+ for (const [id, variable] of Object.entries(fileData.variables)) {
258
+ variablesMap.set(id, variable);
259
+ }
260
+ }
261
+ return variablesMap;
262
+ }
263
+ /**
264
+ * Clear all caches
265
+ */
266
+ clearCache() {
267
+ this.fileDataCache.clear();
268
+ this.lastEnrichmentTime.clear();
269
+ this.styleResolver.clearCache();
270
+ this.relationshipMapper.clear();
271
+ }
272
+ }
273
+ //# sourceMappingURL=enrichment-service.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"enrichment-service.js","sourceRoot":"","sources":["../../../src/core/enrichment/enrichment-service.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Enrichment module exports
3
+ * Phase 5: Enriched Data Extraction & Design System Auditing
4
+ */
5
+ export { StyleValueResolver } from "./style-resolver.js";
6
+ export { RelationshipMapper } from "./relationship-mapper.js";
7
+ export { EnrichmentService } from "./enrichment-service.js";
8
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/core/enrichment/index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,0BAA0B,CAAC;AAC9D,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Enrichment module exports
3
+ * Phase 5: Enriched Data Extraction & Design System Auditing
4
+ */
5
+ export { StyleValueResolver } from "./style-resolver.js";
6
+ export { RelationshipMapper } from "./relationship-mapper.js";
7
+ export { EnrichmentService } from "./enrichment-service.js";
8
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/core/enrichment/index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,0BAA0B,CAAC;AAC9D,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC"}
@@ -0,0 +1,106 @@
1
+ /**
2
+ * Relationship Mapper
3
+ * Maps relationships between styles, variables, and components
4
+ * Tracks usage counts and builds reverse lookup indexes
5
+ */
6
+ import type { ComponentUsage, StyleUsage, VariableDependency } from "../types/enriched.js";
7
+ import type pino from "pino";
8
+ type Logger = pino.Logger;
9
+ export declare class RelationshipMapper {
10
+ private logger;
11
+ private componentsByStyle;
12
+ private componentsByVariable;
13
+ private stylesByVariable;
14
+ private variableDependencies;
15
+ private styleUsageCount;
16
+ private variableUsageCount;
17
+ constructor(logger: Logger);
18
+ /**
19
+ * Build all relationship indexes from file data
20
+ */
21
+ buildRelationships(fileData: any): Promise<void>;
22
+ /**
23
+ * Get components that use a specific style
24
+ */
25
+ getComponentsByStyle(styleId: string): ComponentUsage[];
26
+ /**
27
+ * Get components that use a specific variable
28
+ */
29
+ getComponentsByVariable(variableId: string): ComponentUsage[];
30
+ /**
31
+ * Get styles that use a specific variable
32
+ */
33
+ getStylesByVariable(variableId: string): StyleUsage[];
34
+ /**
35
+ * Get variable dependencies (what variables this variable references)
36
+ */
37
+ getVariableDependencies(variableId: string): VariableDependency[];
38
+ /**
39
+ * Get usage count for a style
40
+ */
41
+ getStyleUsageCount(styleId: string): number;
42
+ /**
43
+ * Get usage count for a variable
44
+ */
45
+ getVariableUsageCount(variableId: string): number;
46
+ /**
47
+ * Find unused styles (styles with zero usage)
48
+ */
49
+ getUnusedStyles(allStyles: any[]): StyleUsage[];
50
+ /**
51
+ * Find unused variables (variables with zero usage)
52
+ */
53
+ getUnusedVariables(allVariables: any[]): any[];
54
+ /**
55
+ * Build variable dependency graph (which variables reference which)
56
+ */
57
+ private buildVariableDependencies;
58
+ /**
59
+ * Extract dependencies from a single variable
60
+ */
61
+ private extractVariableDependencies;
62
+ /**
63
+ * Build relationships between styles and variables
64
+ */
65
+ private buildStyleVariableRelationships;
66
+ /**
67
+ * Extract variable references from a style
68
+ */
69
+ private extractStyleVariableReferences;
70
+ /**
71
+ * Build component relationships by traversing the document tree
72
+ */
73
+ private buildComponentRelationships;
74
+ /**
75
+ * Recursively traverse nodes to find component instances and their style/variable usage
76
+ */
77
+ private traverseNode;
78
+ /**
79
+ * Track which styles a node uses
80
+ */
81
+ private trackNodeStyleUsage;
82
+ /**
83
+ * Track which variables a node uses
84
+ */
85
+ private trackNodeVariableUsage;
86
+ /**
87
+ * Extract all variable references from a node's boundVariables
88
+ */
89
+ private extractNodeVariableReferences;
90
+ /**
91
+ * Detect circular variable references
92
+ */
93
+ detectCircularReferences(): Array<{
94
+ chain: string[];
95
+ }>;
96
+ /**
97
+ * DFS helper for detecting circular references
98
+ */
99
+ private detectCircularDFS;
100
+ /**
101
+ * Clear all indexes and caches
102
+ */
103
+ clear(): void;
104
+ }
105
+ export {};
106
+ //# sourceMappingURL=relationship-mapper.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"relationship-mapper.d.ts","sourceRoot":"","sources":["../../../src/core/enrichment/relationship-mapper.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,KAAK,EACX,cAAc,EACd,UAAU,EACV,kBAAkB,EAClB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAE7B,KAAK,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;AAE1B,qBAAa,kBAAkB;IAC9B,OAAO,CAAC,MAAM,CAAS;IAGvB,OAAO,CAAC,iBAAiB,CAA4C;IACrE,OAAO,CAAC,oBAAoB,CAA4C;IACxE,OAAO,CAAC,gBAAgB,CAAwC;IAChE,OAAO,CAAC,oBAAoB,CAAgD;IAG5E,OAAO,CAAC,eAAe,CAAkC;IACzD,OAAO,CAAC,kBAAkB,CAAkC;gBAEhD,MAAM,EAAE,MAAM;IAI1B;;OAEG;IACG,kBAAkB,CAAC,QAAQ,EAAE,GAAG,GAAG,OAAO,CAAC,IAAI,CAAC;IAmCtD;;OAEG;IACH,oBAAoB,CAAC,OAAO,EAAE,MAAM,GAAG,cAAc,EAAE;IAIvD;;OAEG;IACH,uBAAuB,CAAC,UAAU,EAAE,MAAM,GAAG,cAAc,EAAE;IAI7D;;OAEG;IACH,mBAAmB,CAAC,UAAU,EAAE,MAAM,GAAG,UAAU,EAAE;IAIrD;;OAEG;IACH,uBAAuB,CAAC,UAAU,EAAE,MAAM,GAAG,kBAAkB,EAAE;IAIjE;;OAEG;IACH,kBAAkB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM;IAI3C;;OAEG;IACH,qBAAqB,CAAC,UAAU,EAAE,MAAM,GAAG,MAAM;IAIjD;;OAEG;IACH,eAAe,CAAC,SAAS,EAAE,GAAG,EAAE,GAAG,UAAU,EAAE;IAoB/C;;OAEG;IACH,kBAAkB,CAAC,YAAY,EAAE,GAAG,EAAE,GAAG,GAAG,EAAE;IAkB9C;;OAEG;YACW,yBAAyB;IAWvC;;OAEG;IACH,OAAO,CAAC,2BAA2B;IAqCnC;;OAEG;YACW,+BAA+B;IA2B7C;;OAEG;IACH,OAAO,CAAC,8BAA8B;IAuBtC;;OAEG;YACW,2BAA2B;IASzC;;OAEG;YACW,YAAY;IAyB1B;;OAEG;IACH,OAAO,CAAC,mBAAmB;IA+B3B;;OAEG;IACH,OAAO,CAAC,sBAAsB;IA4B9B;;OAEG;IACH,OAAO,CAAC,6BAA6B;IAmBrC;;OAEG;IACH,wBAAwB,IAAI,KAAK,CAAC;QAAE,KAAK,EAAE,MAAM,EAAE,CAAA;KAAE,CAAC;IAYtD;;OAEG;IACH,OAAO,CAAC,iBAAiB;IA2BzB;;OAEG;IACH,KAAK,IAAI,IAAI;CAQb"}