@vib3code/sdk 2.0.1

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 (258) hide show
  1. package/CHANGELOG.md +118 -0
  2. package/DOCS/BLUEPRINT_EXECUTION_PLAN_2026-01-07.md +34 -0
  3. package/DOCS/CI_TESTING.md +38 -0
  4. package/DOCS/CLI_ONBOARDING.md +75 -0
  5. package/DOCS/CONTROL_REFERENCE.md +64 -0
  6. package/DOCS/DEV_TRACK_ANALYSIS.md +77 -0
  7. package/DOCS/DEV_TRACK_PLAN_2026-01-07.md +42 -0
  8. package/DOCS/DEV_TRACK_SESSION_2026-01-31.md +220 -0
  9. package/DOCS/ENV_SETUP.md +189 -0
  10. package/DOCS/EXPORT_FORMATS.md +417 -0
  11. package/DOCS/GPU_DISPOSAL_GUIDE.md +21 -0
  12. package/DOCS/LICENSING_TIERS.md +275 -0
  13. package/DOCS/MASTER_PLAN_2026-01-31.md +570 -0
  14. package/DOCS/OBS_SETUP_GUIDE.md +98 -0
  15. package/DOCS/PROJECT_SETUP.md +66 -0
  16. package/DOCS/RENDERER_LIFECYCLE.md +40 -0
  17. package/DOCS/REPO_MANIFEST.md +121 -0
  18. package/DOCS/SESSION_014_PLAN.md +195 -0
  19. package/DOCS/SESSION_LOG_2026-01-07.md +56 -0
  20. package/DOCS/STRATEGIC_BLUEPRINT_2026-01-07.md +72 -0
  21. package/DOCS/SYSTEM_AUDIT_2026-01-30.md +738 -0
  22. package/DOCS/SYSTEM_INVENTORY.md +520 -0
  23. package/DOCS/TELEMETRY_EXPORTS.md +34 -0
  24. package/DOCS/WEBGPU_STATUS.md +38 -0
  25. package/DOCS/XR_BENCHMARKS.md +608 -0
  26. package/LICENSE +21 -0
  27. package/README.md +426 -0
  28. package/docs/.nojekyll +0 -0
  29. package/docs/01-dissolution_of_euclidean_hegemony.html +346 -0
  30. package/docs/02-hyperspatial_ego_death.html +346 -0
  31. package/docs/03-post_cartesian_sublime.html +346 -0
  32. package/docs/04-crystalline_void_meditation.html +346 -0
  33. package/docs/05-quantum_decoherence_ballet.html +346 -0
  34. package/docs/06-dissolution_of_euclidean_hegemony.html +346 -0
  35. package/docs/07-hyperspatial_ego_death.html +346 -0
  36. package/docs/08-post_cartesian_sublime.html +346 -0
  37. package/docs/09-crystalline_void_meditation.html +346 -0
  38. package/docs/10-quantum_decoherence_ballet.html +346 -0
  39. package/docs/11-dissolution_of_euclidean_hegemony.html +346 -0
  40. package/docs/12-hyperspatial_ego_death.html +346 -0
  41. package/docs/13-post_cartesian_sublime.html +346 -0
  42. package/docs/index.html +794 -0
  43. package/docs/test-hub.html +441 -0
  44. package/docs/url-state.js +102 -0
  45. package/docs/vib3-exports/01-quantum-quantum-tetrahedron-lattice.html +489 -0
  46. package/docs/vib3-exports/02-quantum-quantum-hypersphere-matrix.html +489 -0
  47. package/docs/vib3-exports/03-quantum-quantum-hypertetra-fractal.html +489 -0
  48. package/docs/vib3-exports/04-faceted-faceted-crystal-structure.html +407 -0
  49. package/docs/vib3-exports/05-faceted-faceted-klein-bottle.html +407 -0
  50. package/docs/vib3-exports/06-faceted-faceted-hypertetra-torus.html +407 -0
  51. package/docs/vib3-exports/07-holographic-holographic-wave-field.html +457 -0
  52. package/docs/vib3-exports/08-holographic-holographic-hypersphere-sphere.html +457 -0
  53. package/docs/vib3-exports/09-holographic-holographic-hypertetra-crystal.html +457 -0
  54. package/docs/vib3-exports/index.html +238 -0
  55. package/docs/webgpu-live.html +702 -0
  56. package/package.json +367 -0
  57. package/src/advanced/AIPresetGenerator.js +777 -0
  58. package/src/advanced/MIDIController.js +703 -0
  59. package/src/advanced/OffscreenWorker.js +1051 -0
  60. package/src/advanced/WebGPUCompute.js +1051 -0
  61. package/src/advanced/WebXRRenderer.js +680 -0
  62. package/src/agent/cli/AgentCLI.js +615 -0
  63. package/src/agent/cli/index.js +14 -0
  64. package/src/agent/index.js +73 -0
  65. package/src/agent/mcp/MCPServer.js +950 -0
  66. package/src/agent/mcp/index.js +9 -0
  67. package/src/agent/mcp/tools.js +548 -0
  68. package/src/agent/telemetry/EventStream.js +669 -0
  69. package/src/agent/telemetry/Instrumentation.js +618 -0
  70. package/src/agent/telemetry/TelemetryExporters.js +427 -0
  71. package/src/agent/telemetry/TelemetryService.js +464 -0
  72. package/src/agent/telemetry/index.js +52 -0
  73. package/src/benchmarks/BenchmarkRunner.js +381 -0
  74. package/src/benchmarks/MetricsCollector.js +299 -0
  75. package/src/benchmarks/index.js +9 -0
  76. package/src/benchmarks/scenes.js +259 -0
  77. package/src/cli/index.js +675 -0
  78. package/src/config/ApiConfig.js +88 -0
  79. package/src/core/CanvasManager.js +217 -0
  80. package/src/core/ErrorReporter.js +117 -0
  81. package/src/core/ParameterMapper.js +333 -0
  82. package/src/core/Parameters.js +396 -0
  83. package/src/core/RendererContracts.js +200 -0
  84. package/src/core/UnifiedResourceManager.js +370 -0
  85. package/src/core/VIB3Engine.js +636 -0
  86. package/src/core/renderers/FacetedRendererAdapter.js +32 -0
  87. package/src/core/renderers/HolographicRendererAdapter.js +29 -0
  88. package/src/core/renderers/QuantumRendererAdapter.js +29 -0
  89. package/src/core/renderers/RendererLifecycleManager.js +63 -0
  90. package/src/creative/ColorPresetsSystem.js +980 -0
  91. package/src/creative/ParameterTimeline.js +1061 -0
  92. package/src/creative/PostProcessingPipeline.js +1113 -0
  93. package/src/creative/TransitionAnimator.js +683 -0
  94. package/src/export/CSSExporter.js +226 -0
  95. package/src/export/CardGeneratorBase.js +279 -0
  96. package/src/export/ExportManager.js +580 -0
  97. package/src/export/FacetedCardGenerator.js +279 -0
  98. package/src/export/HolographicCardGenerator.js +543 -0
  99. package/src/export/LottieExporter.js +552 -0
  100. package/src/export/QuantumCardGenerator.js +315 -0
  101. package/src/export/SVGExporter.js +519 -0
  102. package/src/export/ShaderExporter.js +903 -0
  103. package/src/export/TradingCardGenerator.js +3055 -0
  104. package/src/export/TradingCardManager.js +181 -0
  105. package/src/export/VIB3PackageExporter.js +559 -0
  106. package/src/export/index.js +14 -0
  107. package/src/export/systems/TradingCardSystemFaceted.js +494 -0
  108. package/src/export/systems/TradingCardSystemHolographic.js +452 -0
  109. package/src/export/systems/TradingCardSystemQuantum.js +411 -0
  110. package/src/faceted/FacetedSystem.js +963 -0
  111. package/src/features/CollectionManager.js +433 -0
  112. package/src/gallery/CollectionManager.js +240 -0
  113. package/src/gallery/GallerySystem.js +485 -0
  114. package/src/geometry/GeometryFactory.js +314 -0
  115. package/src/geometry/GeometryLibrary.js +72 -0
  116. package/src/geometry/buffers/BufferBuilder.js +338 -0
  117. package/src/geometry/buffers/index.js +18 -0
  118. package/src/geometry/generators/Crystal.js +420 -0
  119. package/src/geometry/generators/Fractal.js +298 -0
  120. package/src/geometry/generators/KleinBottle.js +197 -0
  121. package/src/geometry/generators/Sphere.js +192 -0
  122. package/src/geometry/generators/Tesseract.js +160 -0
  123. package/src/geometry/generators/Tetrahedron.js +225 -0
  124. package/src/geometry/generators/Torus.js +304 -0
  125. package/src/geometry/generators/Wave.js +341 -0
  126. package/src/geometry/index.js +142 -0
  127. package/src/geometry/warp/HypersphereCore.js +211 -0
  128. package/src/geometry/warp/HypertetraCore.js +386 -0
  129. package/src/geometry/warp/index.js +57 -0
  130. package/src/holograms/HolographicVisualizer.js +1073 -0
  131. package/src/holograms/RealHolographicSystem.js +966 -0
  132. package/src/holograms/variantRegistry.js +69 -0
  133. package/src/integrations/FigmaPlugin.js +854 -0
  134. package/src/integrations/OBSMode.js +754 -0
  135. package/src/integrations/ThreeJsPackage.js +660 -0
  136. package/src/integrations/TouchDesignerExport.js +552 -0
  137. package/src/integrations/frameworks/Vib3React.js +591 -0
  138. package/src/integrations/frameworks/Vib3Svelte.js +654 -0
  139. package/src/integrations/frameworks/Vib3Vue.js +628 -0
  140. package/src/llm/LLMParameterInterface.js +240 -0
  141. package/src/llm/LLMParameterUI.js +577 -0
  142. package/src/math/Mat4x4.js +708 -0
  143. package/src/math/Projection.js +341 -0
  144. package/src/math/Rotor4D.js +637 -0
  145. package/src/math/Vec4.js +476 -0
  146. package/src/math/constants.js +164 -0
  147. package/src/math/index.js +68 -0
  148. package/src/math/projections.js +54 -0
  149. package/src/math/rotations.js +196 -0
  150. package/src/quantum/QuantumEngine.js +906 -0
  151. package/src/quantum/QuantumVisualizer.js +1103 -0
  152. package/src/reactivity/ReactivityConfig.js +499 -0
  153. package/src/reactivity/ReactivityManager.js +586 -0
  154. package/src/reactivity/SpatialInputSystem.js +1783 -0
  155. package/src/reactivity/index.js +93 -0
  156. package/src/render/CommandBuffer.js +465 -0
  157. package/src/render/MultiCanvasBridge.js +340 -0
  158. package/src/render/RenderCommand.js +514 -0
  159. package/src/render/RenderResourceRegistry.js +523 -0
  160. package/src/render/RenderState.js +552 -0
  161. package/src/render/RenderTarget.js +512 -0
  162. package/src/render/ShaderLoader.js +253 -0
  163. package/src/render/ShaderProgram.js +599 -0
  164. package/src/render/UnifiedRenderBridge.js +496 -0
  165. package/src/render/backends/WebGLBackend.js +1108 -0
  166. package/src/render/backends/WebGPUBackend.js +1409 -0
  167. package/src/render/commands/CommandBufferExecutor.js +607 -0
  168. package/src/render/commands/RenderCommandBuffer.js +661 -0
  169. package/src/render/commands/index.js +17 -0
  170. package/src/render/index.js +367 -0
  171. package/src/scene/Disposable.js +498 -0
  172. package/src/scene/MemoryPool.js +618 -0
  173. package/src/scene/Node4D.js +697 -0
  174. package/src/scene/ResourceManager.js +599 -0
  175. package/src/scene/Scene4D.js +540 -0
  176. package/src/scene/index.js +98 -0
  177. package/src/schemas/error.schema.json +84 -0
  178. package/src/schemas/extension.schema.json +88 -0
  179. package/src/schemas/index.js +214 -0
  180. package/src/schemas/parameters.schema.json +142 -0
  181. package/src/schemas/tool-pack.schema.json +44 -0
  182. package/src/schemas/tool-response.schema.json +127 -0
  183. package/src/shaders/common/fullscreen.vert.glsl +5 -0
  184. package/src/shaders/common/fullscreen.vert.wgsl +17 -0
  185. package/src/shaders/common/geometry24.glsl +65 -0
  186. package/src/shaders/common/geometry24.wgsl +54 -0
  187. package/src/shaders/common/rotation4d.glsl +85 -0
  188. package/src/shaders/common/rotation4d.wgsl +86 -0
  189. package/src/shaders/common/uniforms.glsl +44 -0
  190. package/src/shaders/common/uniforms.wgsl +48 -0
  191. package/src/shaders/faceted/faceted.frag.glsl +129 -0
  192. package/src/shaders/faceted/faceted.frag.wgsl +164 -0
  193. package/src/shaders/holographic/holographic.frag.glsl +406 -0
  194. package/src/shaders/holographic/holographic.frag.wgsl +185 -0
  195. package/src/shaders/quantum/quantum.frag.glsl +513 -0
  196. package/src/shaders/quantum/quantum.frag.wgsl +361 -0
  197. package/src/testing/ParallelTestFramework.js +519 -0
  198. package/src/testing/__snapshots__/exportFormats.test.js.snap +24 -0
  199. package/src/testing/exportFormats.test.js +8 -0
  200. package/src/testing/projections.test.js +14 -0
  201. package/src/testing/rotations.test.js +37 -0
  202. package/src/ui/InteractivityMenu.js +516 -0
  203. package/src/ui/StatusManager.js +96 -0
  204. package/src/ui/adaptive/renderers/webgpu/BufferLayout.ts +252 -0
  205. package/src/ui/adaptive/renderers/webgpu/PolytopeInstanceBuffer.ts +144 -0
  206. package/src/ui/adaptive/renderers/webgpu/TripleBufferedUniform.ts +170 -0
  207. package/src/ui/adaptive/renderers/webgpu/WebGPURenderer.ts +735 -0
  208. package/src/ui/adaptive/renderers/webgpu/index.ts +112 -0
  209. package/src/variations/VariationManager.js +431 -0
  210. package/src/viewer/AudioReactivity.js +505 -0
  211. package/src/viewer/CardBending.js +481 -0
  212. package/src/viewer/GalleryUI.js +832 -0
  213. package/src/viewer/ReactivityManager.js +590 -0
  214. package/src/viewer/TradingCardExporter.js +600 -0
  215. package/src/viewer/ViewerPortal.js +374 -0
  216. package/src/viewer/index.js +12 -0
  217. package/src/wasm/WasmLoader.js +296 -0
  218. package/src/wasm/index.js +132 -0
  219. package/tools/agentic/mcpTools.js +88 -0
  220. package/tools/cli/agent-cli.js +92 -0
  221. package/tools/export/formats.js +24 -0
  222. package/tools/math/rotation-baseline.mjs +64 -0
  223. package/tools/shader-sync-verify.js +937 -0
  224. package/tools/telemetry/manifestPipeline.js +141 -0
  225. package/tools/telemetry/telemetryEvents.js +35 -0
  226. package/types/adaptive-sdk.d.ts +185 -0
  227. package/types/advanced/AIPresetGenerator.d.ts +81 -0
  228. package/types/advanced/MIDIController.d.ts +100 -0
  229. package/types/advanced/OffscreenWorker.d.ts +82 -0
  230. package/types/advanced/WebGPUCompute.d.ts +52 -0
  231. package/types/advanced/WebXRRenderer.d.ts +77 -0
  232. package/types/advanced/index.d.ts +46 -0
  233. package/types/core/ErrorReporter.d.ts +50 -0
  234. package/types/core/VIB3Engine.d.ts +204 -0
  235. package/types/creative/ColorPresetsSystem.d.ts +91 -0
  236. package/types/creative/ParameterTimeline.d.ts +74 -0
  237. package/types/creative/PostProcessingPipeline.d.ts +109 -0
  238. package/types/creative/TransitionAnimator.d.ts +71 -0
  239. package/types/creative/index.d.ts +35 -0
  240. package/types/integrations/FigmaPlugin.d.ts +46 -0
  241. package/types/integrations/OBSMode.d.ts +74 -0
  242. package/types/integrations/ThreeJsPackage.d.ts +62 -0
  243. package/types/integrations/TouchDesignerExport.d.ts +36 -0
  244. package/types/integrations/Vib3React.d.ts +74 -0
  245. package/types/integrations/Vib3Svelte.d.ts +63 -0
  246. package/types/integrations/Vib3Vue.d.ts +55 -0
  247. package/types/integrations/index.d.ts +52 -0
  248. package/types/reactivity/SpatialInputSystem.d.ts +173 -0
  249. package/types/reactivity/index.d.ts +394 -0
  250. package/types/render/CommandBuffer.d.ts +169 -0
  251. package/types/render/RenderCommand.d.ts +312 -0
  252. package/types/render/RenderState.d.ts +279 -0
  253. package/types/render/RenderTarget.d.ts +254 -0
  254. package/types/render/ShaderProgram.d.ts +277 -0
  255. package/types/render/UnifiedRenderBridge.d.ts +143 -0
  256. package/types/render/WebGLBackend.d.ts +168 -0
  257. package/types/render/WebGPUBackend.d.ts +186 -0
  258. package/types/render/index.d.ts +141 -0
@@ -0,0 +1,832 @@
1
+ /**
2
+ * GalleryUI.js - Browser for 100 Variations
3
+ *
4
+ * Provides a visual gallery interface for:
5
+ * - Browsing all 100 variation slots (30 default + 70 custom)
6
+ * - Filtering by system (quantum, faceted, holographic)
7
+ * - Searching and sorting variations
8
+ * - Preview thumbnails with live 4D rotation
9
+ * - Import/export collections
10
+ */
11
+
12
+ import { EventEmitter } from 'events';
13
+
14
+ /**
15
+ * Gallery view modes
16
+ */
17
+ export const ViewMode = {
18
+ GRID: 'grid',
19
+ LIST: 'list',
20
+ CAROUSEL: 'carousel'
21
+ };
22
+
23
+ /**
24
+ * Sort options
25
+ */
26
+ export const SortBy = {
27
+ INDEX: 'index',
28
+ NAME: 'name',
29
+ SYSTEM: 'system',
30
+ RECENTLY_MODIFIED: 'recentlyModified',
31
+ FAVORITES: 'favorites'
32
+ };
33
+
34
+ /**
35
+ * GalleryUI - Visual browser for 100 variations
36
+ */
37
+ export class GalleryUI extends EventEmitter {
38
+ constructor(options = {}) {
39
+ super();
40
+
41
+ this.container = null;
42
+ this.engine = null;
43
+ this.gallerySystem = null;
44
+
45
+ // View state
46
+ this.viewMode = options.viewMode || ViewMode.GRID;
47
+ this.sortBy = options.sortBy || SortBy.INDEX;
48
+ this.filterSystem = options.filterSystem || null;
49
+ this.searchQuery = '';
50
+ this.selectedIndex = 0;
51
+ this.pageSize = options.pageSize || 20;
52
+ this.currentPage = 0;
53
+
54
+ // Configuration
55
+ this.config = {
56
+ thumbnailSize: options.thumbnailSize || 150,
57
+ previewDelay: options.previewDelay || 300,
58
+ animatePreview: options.animatePreview !== false,
59
+ showLabels: options.showLabels !== false,
60
+ gridColumns: options.gridColumns || 5
61
+ };
62
+
63
+ // Internal state
64
+ this._variations = [];
65
+ this._filteredVariations = [];
66
+ this._thumbnailCache = new Map();
67
+ this._previewTimer = null;
68
+ this._previewCanvas = null;
69
+
70
+ // DOM elements
71
+ this._elements = {};
72
+
73
+ // Bind methods
74
+ this._onVariationClick = this._onVariationClick.bind(this);
75
+ this._onVariationHover = this._onVariationHover.bind(this);
76
+ this._onSearch = this._onSearch.bind(this);
77
+ this._onPageChange = this._onPageChange.bind(this);
78
+ }
79
+
80
+ /**
81
+ * Initialize gallery UI
82
+ */
83
+ initialize(container, engine, gallerySystem) {
84
+ this.container = container;
85
+ this.engine = engine;
86
+ this.gallerySystem = gallerySystem;
87
+
88
+ // Load variations
89
+ this._loadVariations();
90
+
91
+ // Build UI
92
+ this._buildUI();
93
+
94
+ // Render initial view
95
+ this.render();
96
+
97
+ this.emit('initialized');
98
+ return this;
99
+ }
100
+
101
+ /**
102
+ * Dispose and cleanup
103
+ */
104
+ dispose() {
105
+ if (this._previewTimer) {
106
+ clearTimeout(this._previewTimer);
107
+ }
108
+
109
+ this._thumbnailCache.clear();
110
+
111
+ if (this.container) {
112
+ this.container.innerHTML = '';
113
+ }
114
+
115
+ this.emit('disposed');
116
+ }
117
+
118
+ /**
119
+ * Load variations from gallery system
120
+ */
121
+ _loadVariations() {
122
+ this._variations = [];
123
+
124
+ // Load 100 variation slots
125
+ for (let i = 0; i < 100; i++) {
126
+ const variation = this.gallerySystem?.getVariation(i) || this._getDefaultVariation(i);
127
+ this._variations.push({
128
+ index: i,
129
+ ...variation,
130
+ isDefault: i < 30,
131
+ isEmpty: !variation.system
132
+ });
133
+ }
134
+
135
+ this._applyFilters();
136
+ }
137
+
138
+ /**
139
+ * Get default variation for slot
140
+ */
141
+ _getDefaultVariation(index) {
142
+ if (index >= 30) {
143
+ return { name: `Custom ${index - 29}`, system: null, geometry: 0 };
144
+ }
145
+
146
+ const systems = ['quantum', 'faceted', 'holographic'];
147
+ const system = systems[Math.floor(index / 10)];
148
+ const geometryIndex = index % 10;
149
+
150
+ const geometryNames = [
151
+ 'Tetrahedron', 'Hypercube', 'Sphere', 'Torus',
152
+ 'Klein Bottle', 'Fractal', 'Wave', 'Crystal',
153
+ 'Hypersphere Tetra', 'Hypersphere Cube'
154
+ ];
155
+
156
+ return {
157
+ name: `${geometryNames[geometryIndex]} (${system})`,
158
+ system,
159
+ geometry: geometryIndex,
160
+ rotation: { xy: 0, xz: 0, yz: 0, xw: 0, yw: 0, zw: 0 }
161
+ };
162
+ }
163
+
164
+ /**
165
+ * Build gallery UI elements
166
+ */
167
+ _buildUI() {
168
+ this.container.innerHTML = '';
169
+ this.container.className = 'vib3-gallery';
170
+
171
+ // Apply container styles
172
+ Object.assign(this.container.style, {
173
+ display: 'flex',
174
+ flexDirection: 'column',
175
+ height: '100%',
176
+ background: '#0a0a0f',
177
+ color: '#ffffff',
178
+ fontFamily: '"Segoe UI", Arial, sans-serif'
179
+ });
180
+
181
+ // Create header
182
+ this._elements.header = this._createHeader();
183
+ this.container.appendChild(this._elements.header);
184
+
185
+ // Create main content area
186
+ this._elements.content = document.createElement('div');
187
+ Object.assign(this._elements.content.style, {
188
+ flex: '1',
189
+ overflow: 'auto',
190
+ padding: '20px'
191
+ });
192
+ this.container.appendChild(this._elements.content);
193
+
194
+ // Create pagination
195
+ this._elements.pagination = this._createPagination();
196
+ this.container.appendChild(this._elements.pagination);
197
+
198
+ // Create preview overlay
199
+ this._elements.preview = this._createPreviewOverlay();
200
+ this.container.appendChild(this._elements.preview);
201
+ }
202
+
203
+ /**
204
+ * Create header with controls
205
+ */
206
+ _createHeader() {
207
+ const header = document.createElement('div');
208
+ Object.assign(header.style, {
209
+ display: 'flex',
210
+ alignItems: 'center',
211
+ gap: '15px',
212
+ padding: '15px 20px',
213
+ borderBottom: '1px solid #333',
214
+ background: '#111'
215
+ });
216
+
217
+ // Title
218
+ const title = document.createElement('h2');
219
+ title.textContent = 'VIB3+ Gallery';
220
+ Object.assign(title.style, {
221
+ margin: '0',
222
+ fontSize: '20px',
223
+ fontWeight: 'bold',
224
+ color: '#00ffff'
225
+ });
226
+ header.appendChild(title);
227
+
228
+ // Spacer
229
+ const spacer = document.createElement('div');
230
+ spacer.style.flex = '1';
231
+ header.appendChild(spacer);
232
+
233
+ // Search input
234
+ const search = document.createElement('input');
235
+ search.type = 'text';
236
+ search.placeholder = 'Search variations...';
237
+ Object.assign(search.style, {
238
+ padding: '8px 12px',
239
+ background: '#222',
240
+ border: '1px solid #444',
241
+ borderRadius: '4px',
242
+ color: '#fff',
243
+ width: '200px'
244
+ });
245
+ search.addEventListener('input', (e) => {
246
+ this.searchQuery = e.target.value;
247
+ this._applyFilters();
248
+ this.render();
249
+ });
250
+ header.appendChild(search);
251
+
252
+ // System filter
253
+ const systemFilter = document.createElement('select');
254
+ Object.assign(systemFilter.style, {
255
+ padding: '8px 12px',
256
+ background: '#222',
257
+ border: '1px solid #444',
258
+ borderRadius: '4px',
259
+ color: '#fff'
260
+ });
261
+ systemFilter.innerHTML = `
262
+ <option value="">All Systems</option>
263
+ <option value="quantum">Quantum</option>
264
+ <option value="faceted">Faceted</option>
265
+ <option value="holographic">Holographic</option>
266
+ `;
267
+ systemFilter.addEventListener('change', (e) => {
268
+ this.filterSystem = e.target.value || null;
269
+ this._applyFilters();
270
+ this.render();
271
+ });
272
+ header.appendChild(systemFilter);
273
+
274
+ // View mode toggle
275
+ const viewToggle = document.createElement('div');
276
+ Object.assign(viewToggle.style, {
277
+ display: 'flex',
278
+ gap: '5px'
279
+ });
280
+
281
+ ['grid', 'list', 'carousel'].forEach(mode => {
282
+ const btn = document.createElement('button');
283
+ btn.textContent = mode === 'grid' ? '⊞' : mode === 'list' ? '☰' : '◐';
284
+ Object.assign(btn.style, {
285
+ padding: '8px 12px',
286
+ background: this.viewMode === mode ? '#00ffff' : '#222',
287
+ border: 'none',
288
+ borderRadius: '4px',
289
+ color: this.viewMode === mode ? '#000' : '#fff',
290
+ cursor: 'pointer'
291
+ });
292
+ btn.addEventListener('click', () => {
293
+ this.viewMode = mode;
294
+ this.render();
295
+ });
296
+ viewToggle.appendChild(btn);
297
+ });
298
+ header.appendChild(viewToggle);
299
+
300
+ return header;
301
+ }
302
+
303
+ /**
304
+ * Create pagination controls
305
+ */
306
+ _createPagination() {
307
+ const pagination = document.createElement('div');
308
+ Object.assign(pagination.style, {
309
+ display: 'flex',
310
+ alignItems: 'center',
311
+ justifyContent: 'center',
312
+ gap: '10px',
313
+ padding: '15px',
314
+ borderTop: '1px solid #333',
315
+ background: '#111'
316
+ });
317
+
318
+ return pagination;
319
+ }
320
+
321
+ /**
322
+ * Create preview overlay
323
+ */
324
+ _createPreviewOverlay() {
325
+ const overlay = document.createElement('div');
326
+ Object.assign(overlay.style, {
327
+ position: 'fixed',
328
+ top: '0',
329
+ left: '0',
330
+ width: '100%',
331
+ height: '100%',
332
+ background: 'rgba(0, 0, 0, 0.9)',
333
+ display: 'none',
334
+ alignItems: 'center',
335
+ justifyContent: 'center',
336
+ zIndex: '1000'
337
+ });
338
+
339
+ overlay.addEventListener('click', () => {
340
+ this._hidePreview();
341
+ });
342
+
343
+ return overlay;
344
+ }
345
+
346
+ /**
347
+ * Apply current filters
348
+ */
349
+ _applyFilters() {
350
+ this._filteredVariations = this._variations.filter(v => {
351
+ // System filter
352
+ if (this.filterSystem && v.system !== this.filterSystem) {
353
+ return false;
354
+ }
355
+
356
+ // Search filter
357
+ if (this.searchQuery) {
358
+ const query = this.searchQuery.toLowerCase();
359
+ return v.name.toLowerCase().includes(query) ||
360
+ v.system?.toLowerCase().includes(query);
361
+ }
362
+
363
+ return true;
364
+ });
365
+
366
+ // Apply sorting
367
+ this._filteredVariations.sort((a, b) => {
368
+ switch (this.sortBy) {
369
+ case SortBy.NAME:
370
+ return a.name.localeCompare(b.name);
371
+ case SortBy.SYSTEM:
372
+ return (a.system || 'zzz').localeCompare(b.system || 'zzz');
373
+ case SortBy.FAVORITES:
374
+ return (b.isFavorite ? 1 : 0) - (a.isFavorite ? 1 : 0);
375
+ default:
376
+ return a.index - b.index;
377
+ }
378
+ });
379
+
380
+ this.currentPage = 0;
381
+ }
382
+
383
+ /**
384
+ * Render gallery
385
+ */
386
+ render() {
387
+ switch (this.viewMode) {
388
+ case ViewMode.LIST:
389
+ this._renderList();
390
+ break;
391
+ case ViewMode.CAROUSEL:
392
+ this._renderCarousel();
393
+ break;
394
+ default:
395
+ this._renderGrid();
396
+ }
397
+
398
+ this._updatePagination();
399
+ }
400
+
401
+ /**
402
+ * Render grid view
403
+ */
404
+ _renderGrid() {
405
+ const content = this._elements.content;
406
+ content.innerHTML = '';
407
+
408
+ Object.assign(content.style, {
409
+ display: 'grid',
410
+ gridTemplateColumns: `repeat(${this.config.gridColumns}, 1fr)`,
411
+ gap: '15px'
412
+ });
413
+
414
+ const start = this.currentPage * this.pageSize;
415
+ const end = Math.min(start + this.pageSize, this._filteredVariations.length);
416
+ const pageVariations = this._filteredVariations.slice(start, end);
417
+
418
+ pageVariations.forEach(variation => {
419
+ const card = this._createVariationCard(variation);
420
+ content.appendChild(card);
421
+ });
422
+ }
423
+
424
+ /**
425
+ * Render list view
426
+ */
427
+ _renderList() {
428
+ const content = this._elements.content;
429
+ content.innerHTML = '';
430
+
431
+ Object.assign(content.style, {
432
+ display: 'flex',
433
+ flexDirection: 'column',
434
+ gap: '10px'
435
+ });
436
+
437
+ const start = this.currentPage * this.pageSize;
438
+ const end = Math.min(start + this.pageSize, this._filteredVariations.length);
439
+ const pageVariations = this._filteredVariations.slice(start, end);
440
+
441
+ pageVariations.forEach(variation => {
442
+ const row = this._createVariationRow(variation);
443
+ content.appendChild(row);
444
+ });
445
+ }
446
+
447
+ /**
448
+ * Render carousel view
449
+ */
450
+ _renderCarousel() {
451
+ const content = this._elements.content;
452
+ content.innerHTML = '';
453
+
454
+ Object.assign(content.style, {
455
+ display: 'flex',
456
+ alignItems: 'center',
457
+ justifyContent: 'center',
458
+ gap: '20px'
459
+ });
460
+
461
+ // Show 5 items centered on selection
462
+ const total = this._filteredVariations.length;
463
+ const indices = [-2, -1, 0, 1, 2].map(offset => {
464
+ const idx = this.selectedIndex + offset;
465
+ return ((idx % total) + total) % total;
466
+ });
467
+
468
+ indices.forEach((idx, position) => {
469
+ const variation = this._filteredVariations[idx];
470
+ const card = this._createCarouselCard(variation, position === 2);
471
+ content.appendChild(card);
472
+ });
473
+ }
474
+
475
+ /**
476
+ * Create variation card for grid view
477
+ */
478
+ _createVariationCard(variation) {
479
+ const card = document.createElement('div');
480
+ Object.assign(card.style, {
481
+ background: '#1a1a2e',
482
+ borderRadius: '8px',
483
+ overflow: 'hidden',
484
+ cursor: 'pointer',
485
+ transition: 'transform 0.2s, box-shadow 0.2s',
486
+ border: variation.index === this.selectedIndex ? '2px solid #00ffff' : '2px solid transparent'
487
+ });
488
+
489
+ card.addEventListener('mouseenter', () => {
490
+ card.style.transform = 'scale(1.05)';
491
+ card.style.boxShadow = '0 4px 20px rgba(0, 255, 255, 0.3)';
492
+ this._onVariationHover(variation);
493
+ });
494
+
495
+ card.addEventListener('mouseleave', () => {
496
+ card.style.transform = 'scale(1)';
497
+ card.style.boxShadow = 'none';
498
+ });
499
+
500
+ card.addEventListener('click', () => {
501
+ this._onVariationClick(variation);
502
+ });
503
+
504
+ // Thumbnail
505
+ const thumb = document.createElement('div');
506
+ Object.assign(thumb.style, {
507
+ width: '100%',
508
+ aspectRatio: '1',
509
+ background: this._getSystemGradient(variation.system),
510
+ display: 'flex',
511
+ alignItems: 'center',
512
+ justifyContent: 'center',
513
+ fontSize: '24px',
514
+ color: '#fff'
515
+ });
516
+ thumb.textContent = variation.isEmpty ? '+' : variation.index;
517
+ card.appendChild(thumb);
518
+
519
+ // Info
520
+ if (this.config.showLabels) {
521
+ const info = document.createElement('div');
522
+ Object.assign(info.style, {
523
+ padding: '10px',
524
+ fontSize: '12px'
525
+ });
526
+
527
+ const name = document.createElement('div');
528
+ name.textContent = variation.name;
529
+ Object.assign(name.style, {
530
+ fontWeight: 'bold',
531
+ marginBottom: '4px',
532
+ whiteSpace: 'nowrap',
533
+ overflow: 'hidden',
534
+ textOverflow: 'ellipsis'
535
+ });
536
+ info.appendChild(name);
537
+
538
+ if (variation.system) {
539
+ const system = document.createElement('div');
540
+ system.textContent = variation.system;
541
+ Object.assign(system.style, {
542
+ color: this._getSystemColor(variation.system),
543
+ fontSize: '10px',
544
+ textTransform: 'uppercase'
545
+ });
546
+ info.appendChild(system);
547
+ }
548
+
549
+ card.appendChild(info);
550
+ }
551
+
552
+ return card;
553
+ }
554
+
555
+ /**
556
+ * Create variation row for list view
557
+ */
558
+ _createVariationRow(variation) {
559
+ const row = document.createElement('div');
560
+ Object.assign(row.style, {
561
+ display: 'flex',
562
+ alignItems: 'center',
563
+ gap: '15px',
564
+ padding: '10px 15px',
565
+ background: '#1a1a2e',
566
+ borderRadius: '8px',
567
+ cursor: 'pointer',
568
+ transition: 'background 0.2s',
569
+ border: variation.index === this.selectedIndex ? '2px solid #00ffff' : '2px solid transparent'
570
+ });
571
+
572
+ row.addEventListener('mouseenter', () => {
573
+ row.style.background = '#252545';
574
+ });
575
+
576
+ row.addEventListener('mouseleave', () => {
577
+ row.style.background = '#1a1a2e';
578
+ });
579
+
580
+ row.addEventListener('click', () => {
581
+ this._onVariationClick(variation);
582
+ });
583
+
584
+ // Index
585
+ const index = document.createElement('div');
586
+ index.textContent = String(variation.index).padStart(2, '0');
587
+ Object.assign(index.style, {
588
+ width: '30px',
589
+ color: '#666',
590
+ fontFamily: 'monospace'
591
+ });
592
+ row.appendChild(index);
593
+
594
+ // Thumbnail
595
+ const thumb = document.createElement('div');
596
+ Object.assign(thumb.style, {
597
+ width: '50px',
598
+ height: '50px',
599
+ borderRadius: '4px',
600
+ background: this._getSystemGradient(variation.system),
601
+ display: 'flex',
602
+ alignItems: 'center',
603
+ justifyContent: 'center'
604
+ });
605
+ row.appendChild(thumb);
606
+
607
+ // Name
608
+ const name = document.createElement('div');
609
+ name.textContent = variation.name;
610
+ name.style.flex = '1';
611
+ row.appendChild(name);
612
+
613
+ // System badge
614
+ if (variation.system) {
615
+ const badge = document.createElement('div');
616
+ badge.textContent = variation.system;
617
+ Object.assign(badge.style, {
618
+ padding: '4px 8px',
619
+ background: this._getSystemColor(variation.system),
620
+ borderRadius: '4px',
621
+ fontSize: '10px',
622
+ textTransform: 'uppercase',
623
+ color: '#000'
624
+ });
625
+ row.appendChild(badge);
626
+ }
627
+
628
+ return row;
629
+ }
630
+
631
+ /**
632
+ * Create carousel card
633
+ */
634
+ _createCarouselCard(variation, isCenter) {
635
+ const card = document.createElement('div');
636
+ const size = isCenter ? 300 : 150;
637
+ const opacity = isCenter ? 1 : 0.5;
638
+
639
+ Object.assign(card.style, {
640
+ width: `${size}px`,
641
+ height: `${size * 1.4}px`,
642
+ background: '#1a1a2e',
643
+ borderRadius: '12px',
644
+ overflow: 'hidden',
645
+ cursor: 'pointer',
646
+ transition: 'all 0.3s',
647
+ opacity: String(opacity),
648
+ transform: isCenter ? 'scale(1)' : 'scale(0.8)'
649
+ });
650
+
651
+ card.addEventListener('click', () => {
652
+ this._onVariationClick(variation);
653
+ });
654
+
655
+ // Content
656
+ const content = document.createElement('div');
657
+ Object.assign(content.style, {
658
+ width: '100%',
659
+ height: '80%',
660
+ background: this._getSystemGradient(variation.system),
661
+ display: 'flex',
662
+ alignItems: 'center',
663
+ justifyContent: 'center',
664
+ fontSize: isCenter ? '48px' : '24px',
665
+ color: '#fff'
666
+ });
667
+ content.textContent = variation.index;
668
+ card.appendChild(content);
669
+
670
+ // Label
671
+ const label = document.createElement('div');
672
+ Object.assign(label.style, {
673
+ padding: '10px',
674
+ textAlign: 'center',
675
+ fontSize: isCenter ? '16px' : '12px'
676
+ });
677
+ label.textContent = variation.name;
678
+ card.appendChild(label);
679
+
680
+ return card;
681
+ }
682
+
683
+ /**
684
+ * Update pagination
685
+ */
686
+ _updatePagination() {
687
+ const pagination = this._elements.pagination;
688
+ pagination.innerHTML = '';
689
+
690
+ const totalPages = Math.ceil(this._filteredVariations.length / this.pageSize);
691
+
692
+ if (totalPages <= 1) return;
693
+
694
+ // Previous button
695
+ const prevBtn = document.createElement('button');
696
+ prevBtn.textContent = '◀';
697
+ prevBtn.disabled = this.currentPage === 0;
698
+ Object.assign(prevBtn.style, {
699
+ padding: '8px 12px',
700
+ background: '#222',
701
+ border: 'none',
702
+ borderRadius: '4px',
703
+ color: '#fff',
704
+ cursor: prevBtn.disabled ? 'not-allowed' : 'pointer',
705
+ opacity: prevBtn.disabled ? '0.5' : '1'
706
+ });
707
+ prevBtn.addEventListener('click', () => {
708
+ if (this.currentPage > 0) {
709
+ this.currentPage--;
710
+ this.render();
711
+ }
712
+ });
713
+ pagination.appendChild(prevBtn);
714
+
715
+ // Page info
716
+ const pageInfo = document.createElement('span');
717
+ pageInfo.textContent = `${this.currentPage + 1} / ${totalPages}`;
718
+ pageInfo.style.padding = '0 15px';
719
+ pagination.appendChild(pageInfo);
720
+
721
+ // Next button
722
+ const nextBtn = document.createElement('button');
723
+ nextBtn.textContent = '▶';
724
+ nextBtn.disabled = this.currentPage >= totalPages - 1;
725
+ Object.assign(nextBtn.style, {
726
+ padding: '8px 12px',
727
+ background: '#222',
728
+ border: 'none',
729
+ borderRadius: '4px',
730
+ color: '#fff',
731
+ cursor: nextBtn.disabled ? 'not-allowed' : 'pointer',
732
+ opacity: nextBtn.disabled ? '0.5' : '1'
733
+ });
734
+ nextBtn.addEventListener('click', () => {
735
+ if (this.currentPage < totalPages - 1) {
736
+ this.currentPage++;
737
+ this.render();
738
+ }
739
+ });
740
+ pagination.appendChild(nextBtn);
741
+ }
742
+
743
+ /**
744
+ * Handle variation click
745
+ */
746
+ _onVariationClick(variation) {
747
+ this.selectedIndex = variation.index;
748
+ this.emit('select', variation);
749
+ this.render();
750
+ }
751
+
752
+ /**
753
+ * Handle variation hover
754
+ */
755
+ _onVariationHover(variation) {
756
+ if (this._previewTimer) {
757
+ clearTimeout(this._previewTimer);
758
+ }
759
+
760
+ this._previewTimer = setTimeout(() => {
761
+ this.emit('preview', variation);
762
+ }, this.config.previewDelay);
763
+ }
764
+
765
+ /**
766
+ * Show preview overlay
767
+ */
768
+ _showPreview(variation) {
769
+ this._elements.preview.style.display = 'flex';
770
+ // Preview content would be rendered here
771
+ }
772
+
773
+ /**
774
+ * Hide preview overlay
775
+ */
776
+ _hidePreview() {
777
+ this._elements.preview.style.display = 'none';
778
+ }
779
+
780
+ /**
781
+ * Get gradient for system
782
+ */
783
+ _getSystemGradient(system) {
784
+ const gradients = {
785
+ quantum: 'linear-gradient(135deg, #00ff88, #0088ff)',
786
+ faceted: 'linear-gradient(135deg, #ff4488, #ff8844)',
787
+ holographic: 'linear-gradient(135deg, #44aaff, #aa44ff)'
788
+ };
789
+ return gradients[system] || 'linear-gradient(135deg, #333, #555)';
790
+ }
791
+
792
+ /**
793
+ * Get color for system
794
+ */
795
+ _getSystemColor(system) {
796
+ const colors = {
797
+ quantum: '#00ff88',
798
+ faceted: '#ff4488',
799
+ holographic: '#44aaff'
800
+ };
801
+ return colors[system] || '#888';
802
+ }
803
+
804
+ /**
805
+ * Get current selection
806
+ */
807
+ getSelection() {
808
+ return this._variations[this.selectedIndex];
809
+ }
810
+
811
+ /**
812
+ * Set selection by index
813
+ */
814
+ setSelection(index) {
815
+ if (index >= 0 && index < this._variations.length) {
816
+ this.selectedIndex = index;
817
+ this.render();
818
+ this.emit('select', this._variations[index]);
819
+ }
820
+ }
821
+
822
+ /**
823
+ * Refresh gallery data
824
+ */
825
+ refresh() {
826
+ this._loadVariations();
827
+ this.render();
828
+ this.emit('refresh');
829
+ }
830
+ }
831
+
832
+ export default GalleryUI;