@customviews-js/customviews 1.0.3 → 1.1.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 (52) hide show
  1. package/README.md +362 -103
  2. package/dist/{custom-views.cjs.js → custom-views.core.cjs.js} +1708 -891
  3. package/dist/custom-views.core.cjs.js.map +1 -0
  4. package/dist/custom-views.core.esm.js +2413 -0
  5. package/dist/custom-views.core.esm.js.map +1 -0
  6. package/dist/custom-views.esm.js +1708 -890
  7. package/dist/custom-views.esm.js.map +1 -1
  8. package/dist/{custom-views.umd.js → custom-views.js} +1707 -890
  9. package/dist/custom-views.js.map +1 -0
  10. package/dist/custom-views.min.js +7 -0
  11. package/dist/custom-views.min.js.map +1 -0
  12. package/dist/types/{models/AssetsManager.d.ts → core/assets-manager.d.ts} +1 -1
  13. package/dist/types/{models/AssetsManager.d.ts.map → core/assets-manager.d.ts.map} +1 -1
  14. package/dist/types/core/core.d.ts +25 -9
  15. package/dist/types/core/core.d.ts.map +1 -1
  16. package/dist/types/core/custom-elements.d.ts +8 -0
  17. package/dist/types/core/custom-elements.d.ts.map +1 -0
  18. package/dist/types/core/render.d.ts +4 -1
  19. package/dist/types/core/render.d.ts.map +1 -1
  20. package/dist/types/core/tab-manager.d.ts +35 -0
  21. package/dist/types/core/tab-manager.d.ts.map +1 -0
  22. package/dist/types/core/toggle-manager.d.ts +28 -0
  23. package/dist/types/core/toggle-manager.d.ts.map +1 -0
  24. package/dist/types/core/url-state-manager.d.ts.map +1 -1
  25. package/dist/types/core/visibility-manager.d.ts +1 -1
  26. package/dist/types/core/widget.d.ts +2 -0
  27. package/dist/types/core/widget.d.ts.map +1 -1
  28. package/dist/types/entry/browser-entry.d.ts +13 -0
  29. package/dist/types/entry/browser-entry.d.ts.map +1 -0
  30. package/dist/types/index.d.ts +11 -21
  31. package/dist/types/index.d.ts.map +1 -1
  32. package/dist/types/lib/custom-views.d.ts +29 -0
  33. package/dist/types/lib/custom-views.d.ts.map +1 -0
  34. package/dist/types/styles/styles.d.ts +2 -0
  35. package/dist/types/styles/styles.d.ts.map +1 -1
  36. package/dist/types/styles/tab-styles.d.ts +5 -0
  37. package/dist/types/styles/tab-styles.d.ts.map +1 -0
  38. package/dist/types/styles/toggle-styles.d.ts +5 -0
  39. package/dist/types/styles/toggle-styles.d.ts.map +1 -0
  40. package/dist/types/styles/widget-styles.d.ts +1 -1
  41. package/dist/types/styles/widget-styles.d.ts.map +1 -1
  42. package/dist/types/types/types.d.ts +85 -0
  43. package/dist/types/types/types.d.ts.map +1 -1
  44. package/dist/types/utils/url-utils.d.ts +8 -0
  45. package/dist/types/utils/url-utils.d.ts.map +1 -0
  46. package/package.json +61 -57
  47. package/dist/custom-views.cjs.js.map +0 -1
  48. package/dist/custom-views.umd.js.map +0 -1
  49. package/dist/custom-views.umd.min.js +0 -7
  50. package/dist/custom-views.umd.min.js.map +0 -1
  51. package/dist/types/models/Config.d.ts +0 -10
  52. package/dist/types/models/Config.d.ts.map +0 -1
@@ -1,97 +1,8 @@
1
1
  /*!
2
- * @customviews-js/customviews v1.0.2
2
+ * @customviews-js/customviews v1.1.1
3
3
  * (c) 2025 Chan Ger Teck
4
4
  * Released under the MIT License.
5
5
  */
6
- /** --- Basic renderers --- */
7
- function renderImage(el, asset) {
8
- if (!asset.src)
9
- return;
10
- el.innerHTML = '';
11
- const img = document.createElement('img');
12
- img.src = asset.src;
13
- img.alt = asset.alt || '';
14
- // Apply custom styling if provided
15
- if (asset.className) {
16
- img.className = asset.className;
17
- }
18
- if (asset.style) {
19
- img.setAttribute('style', asset.style);
20
- }
21
- // Default styles (can be overridden by asset.style)
22
- img.style.maxWidth = img.style.maxWidth || '100%';
23
- img.style.height = img.style.height || 'auto';
24
- img.style.display = img.style.display || 'block';
25
- el.appendChild(img);
26
- }
27
- function renderText(el, asset) {
28
- if (asset.content != null) {
29
- el.textContent = asset.content;
30
- }
31
- // Apply custom styling if provided
32
- if (asset.className) {
33
- el.className = asset.className;
34
- }
35
- if (asset.style) {
36
- el.setAttribute('style', asset.style);
37
- }
38
- }
39
- function renderHtml(el, asset) {
40
- if (asset.content != null) {
41
- el.innerHTML = asset.content;
42
- }
43
- // Apply custom styling if provided
44
- if (asset.className) {
45
- el.className = asset.className;
46
- }
47
- if (asset.style) {
48
- el.setAttribute('style', asset.style);
49
- }
50
- }
51
- /** --- Unified asset renderer --- */
52
- function detectAssetType(asset) {
53
- // If src exists, it's an image
54
- if (asset.src)
55
- return 'image';
56
- // If content contains HTML tags, it's HTML
57
- if (asset.content && /<[^>]+>/.test(asset.content)) {
58
- return 'html';
59
- }
60
- return 'text';
61
- }
62
- function renderAssetInto(el, assetId, assetsManager) {
63
- const asset = assetsManager.get(assetId);
64
- if (!asset)
65
- return;
66
- const type = asset.type || detectAssetType(asset);
67
- switch (type) {
68
- case 'image':
69
- renderImage(el, asset);
70
- break;
71
- case 'text':
72
- renderText(el, asset);
73
- break;
74
- case 'html':
75
- renderHtml(el, asset);
76
- break;
77
- default:
78
- el.innerHTML = asset.content || String(asset);
79
- console.warn('[CustomViews] Unknown asset type:', type);
80
- }
81
- }
82
-
83
- /**
84
- * Configuration for the site, has default state and list of toggles
85
- */
86
- class Config {
87
- defaultState;
88
- allToggles;
89
- constructor(defaultState, allToggles) {
90
- this.defaultState = defaultState;
91
- this.allToggles = allToggles;
92
- }
93
- }
94
-
95
6
  /**
96
7
  * Manages persistence of custom views state using browser localStorage
97
8
  */
@@ -221,6 +132,10 @@ class URLStateManager {
221
132
  const compact = {
222
133
  t: state.toggles
223
134
  };
135
+ // Add tab groups if present
136
+ if (state.tabs && Object.keys(state.tabs).length > 0) {
137
+ compact.g = Object.entries(state.tabs);
138
+ }
224
139
  // Convert to JSON and encode
225
140
  const json = JSON.stringify(compact);
226
141
  let encoded;
@@ -267,9 +182,19 @@ class URLStateManager {
267
182
  if (!compact || typeof compact !== 'object') {
268
183
  throw new Error('Invalid compact state structure');
269
184
  }
270
- return {
185
+ const state = {
271
186
  toggles: Array.isArray(compact.t) ? compact.t : []
272
187
  };
188
+ // Reconstruct tabs from compact format
189
+ if (Array.isArray(compact.g)) {
190
+ state.tabs = {};
191
+ for (const [groupId, tabId] of compact.g) {
192
+ if (typeof groupId === 'string' && typeof tabId === 'string') {
193
+ state.tabs[groupId] = tabId;
194
+ }
195
+ }
196
+ }
197
+ return state;
273
198
  }
274
199
  catch (error) {
275
200
  console.warn('Failed to decode view state:', error);
@@ -324,7 +249,7 @@ class VisibilityManager {
324
249
  }
325
250
  /**
326
251
  * Apply simple class-based visibility to a toggle element.
327
- * The element is assumed to have data-customviews-toggle.
252
+ * The element is assumed to have data-cv-toggle or data-customviews-toggle.
328
253
  */
329
254
  applyElementVisibility(el, visible) {
330
255
  if (visible) {
@@ -338,34 +263,581 @@ class VisibilityManager {
338
263
  }
339
264
  }
340
265
 
341
- const CORE_STYLES = `
342
- [data-customviews-toggle] {
343
- transition: opacity 150ms ease,
344
- transform 150ms ease,
345
- max-height 200ms ease,
346
- margin 150ms ease;
347
- will-change: opacity, transform, max-height, margin;
348
- }
349
-
350
- .cv-visible {
351
- opacity: 1 !important;
352
- transform: translateY(0) !important;
353
- max-height: var(--cv-max-height, 9999px) !important;
354
- }
355
-
356
- .cv-hidden {
357
- opacity: 0 !important;
358
- transform: translateY(-4px) !important;
359
- pointer-events: none !important;
360
- padding-top: 0 !important;
361
- padding-bottom: 0 !important;
362
- border-top-width: 0 !important;
363
- border-bottom-width: 0 !important;
364
- max-height: 0 !important;
365
- margin-top: 0 !important;
366
- margin-bottom: 0 !important;
367
- overflow: hidden !important;
368
- }
266
+ /** --- Icon utilities --- */
267
+ function ensureFontAwesomeInjected() {
268
+ const isFontAwesomeLoaded = Array.from(document.styleSheets).some(sheet => sheet.href && (sheet.href.includes('font-awesome') || sheet.href.includes('fontawesome')));
269
+ if (isFontAwesomeLoaded)
270
+ return;
271
+ const link = document.createElement('link');
272
+ link.rel = 'stylesheet';
273
+ link.href = 'https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css';
274
+ link.setAttribute('data-customviews-fontawesome', 'true');
275
+ document.head.appendChild(link);
276
+ }
277
+ function replaceIconShortcodes(text) {
278
+ // Handle Font Awesome shortcodes
279
+ return text.replace(/:fa-([\w-]+):/g, (_, icon) => `<i class="fa fa-${icon}"></i>`);
280
+ }
281
+ /** --- Basic renderers --- */
282
+ function renderImage(el, asset) {
283
+ if (!asset.src)
284
+ return;
285
+ el.innerHTML = '';
286
+ const img = document.createElement('img');
287
+ img.src = asset.src;
288
+ img.alt = asset.alt || '';
289
+ // Apply custom styling if provided
290
+ if (asset.className) {
291
+ img.className = asset.className;
292
+ }
293
+ if (asset.style) {
294
+ img.setAttribute('style', asset.style);
295
+ }
296
+ // Default styles (can be overridden by asset.style)
297
+ img.style.maxWidth = img.style.maxWidth || '100%';
298
+ img.style.height = img.style.height || 'auto';
299
+ img.style.display = img.style.display || 'block';
300
+ el.appendChild(img);
301
+ }
302
+ function renderText(el, asset) {
303
+ if (asset.content != null) {
304
+ el.textContent = asset.content;
305
+ }
306
+ // Apply custom styling if provided
307
+ if (asset.className) {
308
+ el.className = asset.className;
309
+ }
310
+ if (asset.style) {
311
+ el.setAttribute('style', asset.style);
312
+ }
313
+ }
314
+ function renderHtml(el, asset) {
315
+ if (asset.content != null) {
316
+ el.innerHTML = asset.content;
317
+ }
318
+ // Apply custom styling if provided
319
+ if (asset.className) {
320
+ el.className = asset.className;
321
+ }
322
+ if (asset.style) {
323
+ el.setAttribute('style', asset.style);
324
+ }
325
+ }
326
+ /** --- Unified asset renderer --- */
327
+ function detectAssetType(asset) {
328
+ // If src exists, it's an image
329
+ if (asset.src)
330
+ return 'image';
331
+ // If content contains HTML tags, it's HTML
332
+ if (asset.content && /<[^>]+>/.test(asset.content)) {
333
+ return 'html';
334
+ }
335
+ return 'text';
336
+ }
337
+ function renderAssetInto(el, assetId, assetsManager) {
338
+ const asset = assetsManager.get(assetId);
339
+ if (!asset)
340
+ return;
341
+ const type = asset.type || detectAssetType(asset);
342
+ switch (type) {
343
+ case 'image':
344
+ renderImage(el, asset);
345
+ break;
346
+ case 'text':
347
+ renderText(el, asset);
348
+ break;
349
+ case 'html':
350
+ renderHtml(el, asset);
351
+ break;
352
+ default:
353
+ el.innerHTML = asset.content || String(asset);
354
+ console.warn('[CustomViews] Unknown asset type:', type);
355
+ }
356
+ }
357
+
358
+ // Constants for selectors
359
+ const TABGROUP_SELECTOR = 'cv-tabgroup';
360
+ const TAB_SELECTOR = 'cv-tab';
361
+ const NAV_AUTO_SELECTOR = 'cv-tabgroup[nav="auto"], cv-tabgroup:not([nav])';
362
+ const NAV_CONTAINER_CLASS = 'cv-tabs-nav';
363
+ /**
364
+ * TabManager handles discovery, visibility, and navigation for tab groups and tabs
365
+ */
366
+ class TabManager {
367
+ /**
368
+ * Apply tab selections to all tab groups in the DOM
369
+ */
370
+ static applySelections(rootEl, tabs, cfgGroups) {
371
+ // Find all cv-tabgroup elements
372
+ const tabGroups = rootEl.querySelectorAll(TABGROUP_SELECTOR);
373
+ tabGroups.forEach((groupEl) => {
374
+ const groupId = groupEl.getAttribute('id');
375
+ if (!groupId)
376
+ return;
377
+ // Determine the active tab for this group
378
+ const activeTabId = this.resolveActiveTab(groupId, tabs, cfgGroups, groupEl);
379
+ // Apply visibility to direct child cv-tab elements only (not nested ones)
380
+ const tabElements = Array.from(groupEl.children).filter((child) => child.tagName.toLowerCase() === TAB_SELECTOR);
381
+ tabElements.forEach((tabEl) => {
382
+ const tabId = tabEl.getAttribute('id');
383
+ if (!tabId)
384
+ return;
385
+ const isActive = tabId === activeTabId;
386
+ this.applyTabVisibility(tabEl, isActive);
387
+ });
388
+ });
389
+ }
390
+ /**
391
+ * Resolve the active tab for a group based on state, config, and DOM
392
+ */
393
+ static resolveActiveTab(groupId, tabs, cfgGroups, groupEl) {
394
+ // 1. Check state
395
+ if (tabs[groupId]) {
396
+ return tabs[groupId];
397
+ }
398
+ // 2. Check config default
399
+ if (cfgGroups) {
400
+ const groupCfg = cfgGroups.find(g => g.id === groupId);
401
+ if (groupCfg) {
402
+ if (groupCfg.default) {
403
+ return groupCfg.default;
404
+ }
405
+ // Fallback to first tab in config
406
+ const firstConfigTab = groupCfg.tabs[0];
407
+ if (firstConfigTab) {
408
+ return firstConfigTab.id;
409
+ }
410
+ }
411
+ }
412
+ // 3. Fallback to first direct cv-tab child in DOM
413
+ const firstTab = Array.from(groupEl.children).find((child) => child.tagName.toLowerCase() === TAB_SELECTOR);
414
+ if (firstTab) {
415
+ return firstTab.getAttribute('id');
416
+ }
417
+ return null;
418
+ }
419
+ /**
420
+ * Apply visibility classes to a tab element
421
+ */
422
+ static applyTabVisibility(tabEl, isActive) {
423
+ if (isActive) {
424
+ tabEl.classList.remove('cv-hidden');
425
+ tabEl.classList.add('cv-visible');
426
+ }
427
+ else {
428
+ tabEl.classList.add('cv-hidden');
429
+ tabEl.classList.remove('cv-visible');
430
+ }
431
+ }
432
+ /**
433
+ * Build navigation for tab groups with nav="auto" (one-time setup)
434
+ */
435
+ static buildNavs(rootEl, cfgGroups, onTabClick) {
436
+ // Find all cv-tabgroup elements with nav="auto" or no nav attribute
437
+ const tabGroups = rootEl.querySelectorAll(NAV_AUTO_SELECTOR);
438
+ // Check if any tab headers contain Font Awesome shortcodes
439
+ // Inject Font Awesome CSS only if needed
440
+ let hasFontAwesomeShortcodes = false;
441
+ tabGroups.forEach((groupEl) => {
442
+ const groupId = groupEl.getAttribute('id');
443
+ if (!groupId)
444
+ return;
445
+ const tabElements = Array.from(groupEl.children).filter((child) => child.tagName.toLowerCase() === 'cv-tab');
446
+ tabElements.forEach((tabEl) => {
447
+ const tabId = tabEl.getAttribute('id');
448
+ if (!tabId)
449
+ return;
450
+ const header = tabEl.getAttribute('header') || this.getTabLabel(tabId, groupId, cfgGroups) || tabId;
451
+ if (/:fa-[\w-]+:/.test(header)) {
452
+ hasFontAwesomeShortcodes = true;
453
+ }
454
+ });
455
+ });
456
+ // Inject Font Awesome only if shortcodes are found
457
+ if (hasFontAwesomeShortcodes) {
458
+ ensureFontAwesomeInjected();
459
+ }
460
+ tabGroups.forEach((groupEl) => {
461
+ const groupId = groupEl.getAttribute('id');
462
+ if (!groupId)
463
+ return;
464
+ // Check if nav already exists - if so, skip building
465
+ let navContainer = groupEl.querySelector(`.${NAV_CONTAINER_CLASS}`);
466
+ if (navContainer)
467
+ return; // Already built
468
+ // Get only direct child tabs (not nested ones)
469
+ const tabElements = Array.from(groupEl.children).filter((child) => child.tagName.toLowerCase() === TAB_SELECTOR);
470
+ if (tabElements.length === 0)
471
+ return;
472
+ // Create nav container
473
+ navContainer = document.createElement('ul');
474
+ navContainer.className = `${NAV_CONTAINER_CLASS} nav-tabs`;
475
+ navContainer.setAttribute('role', 'tablist');
476
+ groupEl.insertBefore(navContainer, groupEl.firstChild);
477
+ // Build nav items
478
+ tabElements.forEach((tabEl) => {
479
+ const tabId = tabEl.getAttribute('id');
480
+ if (!tabId)
481
+ return;
482
+ const header = tabEl.getAttribute('header') || this.getTabLabel(tabId, groupId, cfgGroups) || tabId;
483
+ const listItem = document.createElement('li');
484
+ listItem.className = 'nav-item';
485
+ const navLink = document.createElement('a');
486
+ navLink.className = 'nav-link';
487
+ // Replace icon shortcodes in header
488
+ navLink.innerHTML = replaceIconShortcodes(header);
489
+ navLink.href = '#';
490
+ navLink.setAttribute('data-tab-id', tabId);
491
+ navLink.setAttribute('data-group-id', groupId);
492
+ navLink.setAttribute('role', 'tab');
493
+ // Check if this tab is currently active
494
+ const isActive = tabEl.classList.contains('cv-visible');
495
+ if (isActive) {
496
+ navLink.classList.add('active');
497
+ navLink.setAttribute('aria-selected', 'true');
498
+ }
499
+ else {
500
+ navLink.setAttribute('aria-selected', 'false');
501
+ }
502
+ // Add click handler
503
+ if (onTabClick) {
504
+ navLink.addEventListener('click', (e) => {
505
+ e.preventDefault();
506
+ onTabClick(groupId, tabId);
507
+ });
508
+ }
509
+ listItem.appendChild(navLink);
510
+ navContainer.appendChild(listItem);
511
+ });
512
+ // Add bottom border line at the end of the tab group
513
+ const bottomBorder = document.createElement('div');
514
+ bottomBorder.className = 'cv-tabgroup-bottom-border';
515
+ groupEl.appendChild(bottomBorder);
516
+ });
517
+ }
518
+ /**
519
+ * Get tab label from config
520
+ */
521
+ static getTabLabel(tabId, groupId, cfgGroups) {
522
+ if (!cfgGroups)
523
+ return null;
524
+ const groupCfg = cfgGroups.find(g => g.id === groupId);
525
+ if (!groupCfg)
526
+ return null;
527
+ const tabCfg = groupCfg.tabs.find(t => t.id === tabId);
528
+ return tabCfg?.label || null;
529
+ }
530
+ /**
531
+ * Update active state in navs after selection change (single group)
532
+ */
533
+ static updateNavActiveState(rootEl, groupId, activeTabId) {
534
+ const tabGroups = rootEl.querySelectorAll(`${TABGROUP_SELECTOR}[id="${groupId}"]`);
535
+ tabGroups.forEach((groupEl) => {
536
+ const navLinks = groupEl.querySelectorAll('.nav-link');
537
+ navLinks.forEach((link) => {
538
+ const tabId = link.getAttribute('data-tab-id');
539
+ if (tabId === activeTabId) {
540
+ link.classList.add('active');
541
+ link.setAttribute('aria-selected', 'true');
542
+ }
543
+ else {
544
+ link.classList.remove('active');
545
+ link.setAttribute('aria-selected', 'false');
546
+ }
547
+ });
548
+ });
549
+ }
550
+ /**
551
+ * Update active states for all tab groups based on current state
552
+ */
553
+ static updateAllNavActiveStates(rootEl, tabs, cfgGroups) {
554
+ const tabGroups = rootEl.querySelectorAll(TABGROUP_SELECTOR);
555
+ tabGroups.forEach((groupEl) => {
556
+ const groupId = groupEl.getAttribute('id');
557
+ if (!groupId)
558
+ return;
559
+ // Determine the active tab for this group
560
+ const activeTabId = this.resolveActiveTab(groupId, tabs, cfgGroups, groupEl);
561
+ if (!activeTabId)
562
+ return;
563
+ // Update nav links for this group
564
+ const navLinks = groupEl.querySelectorAll('.nav-link');
565
+ navLinks.forEach((link) => {
566
+ const tabId = link.getAttribute('data-tab-id');
567
+ if (tabId === activeTabId) {
568
+ link.classList.add('active');
569
+ link.setAttribute('aria-selected', 'true');
570
+ }
571
+ else {
572
+ link.classList.remove('active');
573
+ link.setAttribute('aria-selected', 'false');
574
+ }
575
+ });
576
+ });
577
+ }
578
+ }
579
+
580
+ class AssetsManager {
581
+ assets;
582
+ baseURL;
583
+ constructor(assets, baseURL = '') {
584
+ this.assets = assets;
585
+ this.baseURL = baseURL;
586
+ if (!this.validate()) {
587
+ console.warn('Invalid assets:', this.assets);
588
+ }
589
+ }
590
+ // Check each asset has content or src
591
+ validate() {
592
+ return Object.values(this.assets).every(a => a.src || a.content);
593
+ }
594
+ get(assetId) {
595
+ const asset = this.assets[assetId];
596
+ if (!asset)
597
+ return undefined;
598
+ // If there's a baseURL and the asset has a src property, prepend the baseURL
599
+ if (this.baseURL && asset.src) {
600
+ // Create a shallow copy to avoid mutating the original asset
601
+ return {
602
+ ...asset,
603
+ src: this.prependBaseURL(asset.src)
604
+ };
605
+ }
606
+ return asset;
607
+ }
608
+ prependBaseURL(path) {
609
+ // Don't prepend if the path is already absolute (starts with http:// or https://)
610
+ if (path.startsWith('http://') || path.startsWith('https://')) {
611
+ return path;
612
+ }
613
+ // Ensure baseURL doesn't end with / and path starts with /
614
+ const cleanBaseURL = this.baseURL.endsWith('/') ? this.baseURL.slice(0, -1) : this.baseURL;
615
+ const cleanPath = path.startsWith('/') ? path : '/' + path;
616
+ return cleanBaseURL + cleanPath;
617
+ }
618
+ loadFromJSON(json) {
619
+ this.assets = json;
620
+ }
621
+ loadAdditionalAssets(additionalAssets) {
622
+ this.assets = { ...this.assets, ...additionalAssets };
623
+ }
624
+ }
625
+
626
+ // Constants for selectors
627
+ const TOGGLE_DATA_SELECTOR = "[data-cv-toggle], [data-customviews-toggle]";
628
+ const TOGGLE_ELEMENT_SELECTOR = "cv-toggle";
629
+ const TOGGLE_SELECTOR = `${TOGGLE_DATA_SELECTOR}, ${TOGGLE_ELEMENT_SELECTOR}`;
630
+ /**
631
+ * ToggleManager handles discovery, visibility, and asset rendering for toggle elements
632
+ */
633
+ class ToggleManager {
634
+ /**
635
+ * Apply toggle visibility to all toggle elements in the DOM
636
+ */
637
+ static applyToggles(rootEl, activeToggles) {
638
+ rootEl.querySelectorAll(TOGGLE_SELECTOR).forEach(el => {
639
+ const categories = this.getToggleCategories(el);
640
+ const shouldShow = categories.some(cat => activeToggles.includes(cat));
641
+ this.applyToggleVisibility(el, shouldShow);
642
+ });
643
+ }
644
+ /**
645
+ * Render assets into toggle elements that are currently visible
646
+ */
647
+ static renderAssets(rootEl, activeToggles, assetsManager) {
648
+ rootEl.querySelectorAll(TOGGLE_SELECTOR).forEach(el => {
649
+ const categories = this.getToggleCategories(el);
650
+ const toggleId = this.getToggleId(el);
651
+ if (toggleId && categories.some(cat => activeToggles.includes(cat))) {
652
+ renderAssetInto(el, toggleId, assetsManager);
653
+ }
654
+ });
655
+ }
656
+ /**
657
+ * Get toggle categories from an element (supports both data attributes and cv-toggle elements)
658
+ */
659
+ static getToggleCategories(el) {
660
+ if (el.tagName.toLowerCase() === 'cv-toggle') {
661
+ const category = el.getAttribute('category');
662
+ return (category || '').split(/\s+/).filter(Boolean);
663
+ }
664
+ else {
665
+ const data = el.dataset.cvToggle || el.dataset.customviewsToggle;
666
+ return (data || '').split(/\s+/).filter(Boolean);
667
+ }
668
+ }
669
+ /**
670
+ * Get toggle ID from an element
671
+ */
672
+ static getToggleId(el) {
673
+ return el.dataset.cvId || el.dataset.customviewsId || el.getAttribute('data-cv-id') || el.getAttribute('data-customviews-id') || undefined;
674
+ }
675
+ /**
676
+ * Apply simple class-based visibility to a toggle element
677
+ */
678
+ static applyToggleVisibility(el, visible) {
679
+ if (visible) {
680
+ el.classList.remove('cv-hidden');
681
+ el.classList.add('cv-visible');
682
+ }
683
+ else {
684
+ el.classList.add('cv-hidden');
685
+ el.classList.remove('cv-visible');
686
+ }
687
+ }
688
+ }
689
+
690
+ /**
691
+ * Styles for toggle visibility and animations
692
+ */
693
+ const TOGGLE_STYLES = `
694
+ /* Core toggle visibility transitions */
695
+ [data-cv-toggle], [data-customviews-toggle], cv-toggle {
696
+ transition: opacity 150ms ease,
697
+ transform 150ms ease,
698
+ max-height 200ms ease,
699
+ margin 150ms ease;
700
+ will-change: opacity, transform, max-height, margin;
701
+ }
702
+
703
+ .cv-visible {
704
+ opacity: 1 !important;
705
+ transform: translateY(0) !important;
706
+ max-height: var(--cv-max-height, 9999px) !important;
707
+ }
708
+
709
+ .cv-hidden {
710
+ opacity: 0 !important;
711
+ transform: translateY(-4px) !important;
712
+ pointer-events: none !important;
713
+ padding-top: 0 !important;
714
+ padding-bottom: 0 !important;
715
+ border-top-width: 0 !important;
716
+ border-bottom-width: 0 !important;
717
+ max-height: 0 !important;
718
+ margin-top: 0 !important;
719
+ margin-bottom: 0 !important;
720
+ overflow: hidden !important;
721
+ }
722
+ `;
723
+
724
+ /**
725
+ * Styles for tab groups and tab navigation
726
+ */
727
+ const TAB_STYLES = `
728
+ /* Tab navigation styles - Bootstrap-style tabs matching MarkBind */
729
+ .cv-tabs-nav {
730
+ display: flex;
731
+ flex-wrap: wrap;
732
+ padding-left: 0;
733
+ margin-top: 0.5rem;
734
+ margin-bottom: 1rem;
735
+ list-style: none;
736
+ border-bottom: 1px solid #dee2e6;
737
+ }
738
+
739
+ .cv-tabs-nav .nav-item {
740
+ margin-bottom: -1px;
741
+ list-style: none;
742
+ display: inline-block;
743
+ }
744
+
745
+ .cv-tabs-nav .nav-link {
746
+ display: block;
747
+ padding: 0.5rem 1rem;
748
+ color: #495057;
749
+ text-decoration: none;
750
+ background-color: transparent;
751
+ border: 1px solid transparent;
752
+ border-top-left-radius: 0.25rem;
753
+ border-top-right-radius: 0.25rem;
754
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out;
755
+ cursor: pointer;
756
+ }
757
+
758
+ .cv-tabs-nav .nav-link:hover,
759
+ .cv-tabs-nav .nav-link:focus {
760
+ border-color: #e9ecef #e9ecef #dee2e6;
761
+ isolation: isolate;
762
+ }
763
+
764
+ .cv-tabs-nav .nav-link.active {
765
+ color: #495057;
766
+ background-color: #fff;
767
+ border-color: #dee2e6 #dee2e6 #fff;
768
+ }
769
+
770
+ .cv-tabs-nav .nav-link:focus {
771
+ outline: 0;
772
+ box-shadow: 0 0 0 0.25rem rgba(13, 110, 253, 0.25);
773
+ }
774
+
775
+ /* Legacy button-based nav (deprecated, kept for compatibility) */
776
+ .cv-tabs-nav-item {
777
+ background: none;
778
+ border: none;
779
+ border-bottom: 2px solid transparent;
780
+ padding: 0.5rem 1rem;
781
+ cursor: pointer;
782
+ font-size: 1rem;
783
+ color: #6c757d;
784
+ transition: color 150ms ease, border-color 150ms ease;
785
+ }
786
+
787
+ .cv-tabs-nav-item:hover {
788
+ color: #495057;
789
+ border-bottom-color: #dee2e6;
790
+ }
791
+
792
+ .cv-tabs-nav-item.active {
793
+ color: #007bff;
794
+ border-bottom-color: #007bff;
795
+ font-weight: 500;
796
+ }
797
+
798
+ .cv-tabs-nav-item:focus {
799
+ outline: 2px solid #007bff;
800
+ outline-offset: 2px;
801
+ }
802
+
803
+ /* Tab panel base styles */
804
+ cv-tab {
805
+ display: block;
806
+ }
807
+
808
+ /* Override visibility for tab panels - use display instead of collapse animation */
809
+ cv-tab.cv-hidden {
810
+ display: none !important;
811
+ }
812
+
813
+ cv-tab.cv-visible {
814
+ display: block !important;
815
+ }
816
+
817
+ cv-tabgroup {
818
+ display: block;
819
+ margin-bottom: 1.5rem;
820
+ }
821
+
822
+ /* Bottom border line for tab groups */
823
+ .cv-tabgroup-bottom-border {
824
+ border-bottom: 1px solid #dee2e6;
825
+ margin-top: 1rem;
826
+ }
827
+
828
+ /* Tab content wrapper */
829
+ .cv-tab-content {
830
+ padding: 1rem 0;
831
+ }
832
+ `;
833
+
834
+ /**
835
+ * Combined core styles for toggles and tabs
836
+ */
837
+ const CORE_STYLES = `
838
+ ${TOGGLE_STYLES}
839
+
840
+ ${TAB_STYLES}
369
841
  `;
370
842
  /**
371
843
  * Add styles for hiding and showing toggles animations and transitions to the document head
@@ -386,36 +858,85 @@ class CustomViewsCore {
386
858
  assetsManager;
387
859
  persistenceManager;
388
860
  visibilityManager;
389
- stateFromUrl = null;
390
- localConfig;
861
+ config;
391
862
  stateChangeListeners = [];
863
+ showUrlEnabled;
864
+ lastAppliedState = null;
392
865
  constructor(opt) {
393
866
  this.assetsManager = opt.assetsManager;
394
- this.localConfig = opt.config;
867
+ this.config = opt.config;
395
868
  this.rootEl = opt.rootEl || document.body;
396
869
  this.persistenceManager = new PersistenceManager();
397
870
  this.visibilityManager = new VisibilityManager();
871
+ this.showUrlEnabled = opt.showUrl ?? true;
872
+ this.lastAppliedState = this.cloneState(this.config?.defaultState);
873
+ }
874
+ getConfig() {
875
+ return this.config;
398
876
  }
399
- getLocalConfig() {
400
- return this.localConfig;
877
+ /**
878
+ * Get tab groups from config
879
+ */
880
+ getTabGroups() {
881
+ return this.config.tabGroups;
882
+ }
883
+ /**
884
+ * Get currently active tabs (from URL > persisted (localStorage) > defaults)
885
+ */
886
+ getCurrentActiveTabs() {
887
+ if (this.lastAppliedState?.tabs) {
888
+ return { ...this.lastAppliedState.tabs };
889
+ }
890
+ const persistedState = this.persistenceManager.getPersistedState();
891
+ if (persistedState?.tabs) {
892
+ return { ...persistedState.tabs };
893
+ }
894
+ return this.config?.defaultState?.tabs ? { ...this.config.defaultState.tabs } : {};
895
+ }
896
+ /**
897
+ * Set active tab for a group and apply state
898
+ */
899
+ setActiveTab(groupId, tabId) {
900
+ // Get current state
901
+ const currentToggles = this.getCurrentActiveToggles();
902
+ const currentTabs = this.getCurrentActiveTabs();
903
+ // Merge new tab selection
904
+ const newTabs = { ...currentTabs, [groupId]: tabId };
905
+ // Create new state
906
+ const newState = {
907
+ toggles: currentToggles,
908
+ tabs: newTabs
909
+ };
910
+ // Apply the state
911
+ this.applyState(newState);
912
+ // Emit custom event
913
+ const event = new CustomEvent('customviews:tab-change', {
914
+ detail: { groupId, tabId },
915
+ bubbles: true
916
+ });
917
+ document.dispatchEvent(event);
401
918
  }
402
919
  // Inject styles, setup listeners and call rendering logic
403
920
  async init() {
404
921
  injectCoreStyles();
922
+ // Build navigation once (with click handlers)
923
+ TabManager.buildNavs(this.rootEl, this.config.tabGroups, (groupId, tabId) => {
924
+ this.setActiveTab(groupId, tabId);
925
+ });
405
926
  // For session history, clicks on back/forward button
406
927
  window.addEventListener("popstate", () => {
407
- this.loadAndRenderState();
928
+ this.loadAndCallApplyState();
408
929
  });
409
- this.loadAndRenderState();
930
+ this.loadAndCallApplyState();
410
931
  }
411
932
  // Priority: URL state > persisted state > default
412
933
  // Also filters using the visibility manager to persist selection
413
934
  // across back/forward button clicks
414
- async loadAndRenderState() {
935
+ async loadAndCallApplyState() {
415
936
  // 1. URL State
416
- this.stateFromUrl = URLStateManager.parseURL();
417
- if (this.stateFromUrl) {
418
- this.applyState(this.stateFromUrl);
937
+ const urlState = URLStateManager.parseURL();
938
+ if (urlState) {
939
+ this.applyState(urlState);
419
940
  return;
420
941
  }
421
942
  // 2. Persisted State
@@ -425,38 +946,35 @@ class CustomViewsCore {
425
946
  return;
426
947
  }
427
948
  // 3. Local Config Fallback
428
- this.renderState(this.localConfig.defaultState);
949
+ this.renderState(this.config.defaultState);
429
950
  }
430
951
  /**
431
- * Apply a custom state, saves to localStorage and updates the URL
432
- */
952
+ * Apply a custom state, saves to localStorage and updates the URL
953
+ */
433
954
  applyState(state) {
434
- this.renderState(state);
435
- this.persistenceManager.persistState(state);
436
- this.stateFromUrl = state;
437
- URLStateManager.updateURL(state);
955
+ const snapshot = this.cloneState(state);
956
+ this.renderState(snapshot);
957
+ this.persistenceManager.persistState(snapshot);
958
+ if (this.showUrlEnabled) {
959
+ URLStateManager.updateURL(snapshot);
960
+ }
961
+ else {
962
+ URLStateManager.clearURL();
963
+ }
438
964
  }
439
965
  /** Render all toggles for the current state */
440
966
  renderState(state) {
967
+ this.lastAppliedState = this.cloneState(state);
441
968
  const toggles = state.toggles || [];
442
969
  const finalToggles = this.visibilityManager.filterVisibleToggles(toggles);
443
- // Toggles hide or show relevant toggles
444
- this.rootEl.querySelectorAll("[data-customviews-toggle]").forEach(el => {
445
- const category = el.dataset.customviewsToggle;
446
- const shouldShow = !!category && finalToggles.includes(category);
447
- this.visibilityManager.applyElementVisibility(el, shouldShow);
448
- });
449
- // Render toggles
450
- for (const category of finalToggles) {
451
- this.rootEl.querySelectorAll(`[data-customviews-toggle="${category}"]`).forEach(el => {
452
- // if it has an id, then we render the asset into it
453
- // if it has no id, then we assume it's a container
454
- const toggleId = el.dataset.customviewsId;
455
- if (toggleId) {
456
- renderAssetInto(el, toggleId, this.assetsManager);
457
- }
458
- });
459
- }
970
+ // Apply toggle visibility
971
+ ToggleManager.applyToggles(this.rootEl, finalToggles);
972
+ // Render assets into toggles
973
+ ToggleManager.renderAssets(this.rootEl, finalToggles, this.assetsManager);
974
+ // Apply tab selections
975
+ TabManager.applySelections(this.rootEl, state.tabs || {}, this.config.tabGroups);
976
+ // Update nav active states (without rebuilding)
977
+ TabManager.updateAllNavActiveStates(this.rootEl, state.tabs || {}, this.config.tabGroups);
460
978
  // Notify state change listeners (like widgets)
461
979
  this.notifyStateChangeListeners();
462
980
  }
@@ -464,10 +982,9 @@ class CustomViewsCore {
464
982
  * Reset to default state
465
983
  */
466
984
  resetToDefault() {
467
- this.stateFromUrl = null;
468
985
  this.persistenceManager.clearAll();
469
- if (this.localConfig) {
470
- this.renderState(this.localConfig.defaultState);
986
+ if (this.config) {
987
+ this.renderState(this.config.defaultState);
471
988
  }
472
989
  else {
473
990
  console.warn("No configuration loaded, cannot reset to default state");
@@ -479,15 +996,12 @@ class CustomViewsCore {
479
996
  * Get the currently active toggles regardless of whether they come from custom state or default configuration
480
997
  */
481
998
  getCurrentActiveToggles() {
482
- // If we have a custom state, return its toggles
483
- if (this.stateFromUrl) {
484
- return this.stateFromUrl.toggles || [];
999
+ if (this.lastAppliedState) {
1000
+ return this.lastAppliedState.toggles || [];
485
1001
  }
486
- // Otherwise, if we have local config, return its default state toggles
487
- if (this.localConfig) {
488
- return this.localConfig.defaultState.toggles || [];
1002
+ if (this.config) {
1003
+ return this.config.defaultState.toggles || [];
489
1004
  }
490
- // No configuration or state
491
1005
  return [];
492
1006
  }
493
1007
  /**
@@ -495,15 +1009,35 @@ class CustomViewsCore {
495
1009
  */
496
1010
  clearPersistence() {
497
1011
  this.persistenceManager.clearAll();
498
- this.stateFromUrl = null;
499
- if (this.localConfig) {
500
- this.renderState(this.localConfig.defaultState);
1012
+ if (this.config) {
1013
+ this.renderState(this.config.defaultState);
501
1014
  }
502
1015
  else {
503
1016
  console.warn("No configuration loaded, cannot reset to default state");
504
1017
  }
505
1018
  URLStateManager.clearURL();
506
1019
  }
1020
+ setOption(flag, value) {
1021
+ switch (flag) {
1022
+ case 'showUrl': {
1023
+ const nextValue = Boolean(value);
1024
+ if (this.showUrlEnabled === nextValue) {
1025
+ return;
1026
+ }
1027
+ this.showUrlEnabled = nextValue;
1028
+ if (nextValue) {
1029
+ const stateForUrl = this.getTrackedStateSnapshot();
1030
+ URLStateManager.updateURL(stateForUrl);
1031
+ }
1032
+ else {
1033
+ URLStateManager.clearURL();
1034
+ }
1035
+ break;
1036
+ }
1037
+ default:
1038
+ console.warn(`[CustomViews] Unknown option '${flag}' passed to setOption`);
1039
+ }
1040
+ }
507
1041
  // === STATE CHANGE LISTENER METHODS ===
508
1042
  /**
509
1043
  * Add a listener that will be called whenever the state changes
@@ -533,51 +1067,136 @@ class CustomViewsCore {
533
1067
  }
534
1068
  });
535
1069
  }
1070
+ cloneState(state) {
1071
+ const toggles = state?.toggles ? [...state.toggles] : [];
1072
+ const tabs = state?.tabs ? { ...state.tabs } : undefined;
1073
+ return tabs ? { toggles, tabs } : { toggles };
1074
+ }
1075
+ getTrackedStateSnapshot() {
1076
+ if (this.lastAppliedState) {
1077
+ return this.cloneState(this.lastAppliedState);
1078
+ }
1079
+ if (this.config) {
1080
+ return this.cloneState(this.config.defaultState);
1081
+ }
1082
+ return { toggles: [] };
1083
+ }
536
1084
  }
537
1085
 
538
- class AssetsManager {
539
- assets;
540
- baseURL;
541
- constructor(assets, baseURL = '') {
542
- this.assets = assets;
543
- this.baseURL = baseURL;
544
- if (!this.validate()) {
545
- console.warn('Invalid assets:', this.assets);
546
- }
1086
+ /**
1087
+ * Helper function to prepend baseUrl to a path
1088
+ * @param path The path to prepend the baseUrl to
1089
+ * @param baseUrl The base URL to prepend
1090
+ * @returns The full URL with baseUrl prepended if applicable
1091
+ */
1092
+ function prependBaseUrl(path, baseUrl) {
1093
+ if (!baseUrl)
1094
+ return path;
1095
+ // Don't prepend if the path is already absolute (starts with http:// or https://)
1096
+ if (path.startsWith('http://') || path.startsWith('https://')) {
1097
+ return path;
1098
+ }
1099
+ // Ensure baseUrl doesn't end with / and path starts with /
1100
+ const cleanbaseUrl = baseUrl.endsWith('/') ? baseUrl.slice(0, -1) : baseUrl;
1101
+ const cleanPath = path.startsWith('/') ? path : '/' + path;
1102
+ return cleanbaseUrl + cleanPath;
1103
+ }
1104
+
1105
+ /**
1106
+ * Custom Elements for Tab Groups and Tabs
1107
+ */
1108
+ /**
1109
+ * <cv-tab> element - represents a single tab panel
1110
+ */
1111
+ class CVTab extends HTMLElement {
1112
+ connectedCallback() {
1113
+ // Element is managed by TabManager
547
1114
  }
548
- // Check each asset has content or src
549
- validate() {
550
- return Object.values(this.assets).every(a => a.src || a.content);
1115
+ }
1116
+ /**
1117
+ * <cv-tabgroup> element - represents a group of tabs
1118
+ */
1119
+ class CVTabgroup extends HTMLElement {
1120
+ connectedCallback() {
1121
+ // Element is managed by TabManager
1122
+ // Emit ready event after a brief delay to ensure children are parsed
1123
+ setTimeout(() => {
1124
+ const event = new CustomEvent('cv:tabgroup-ready', {
1125
+ bubbles: true,
1126
+ detail: { groupId: this.getAttribute('id') }
1127
+ });
1128
+ this.dispatchEvent(event);
1129
+ }, 0);
551
1130
  }
552
- get(assetId) {
553
- const asset = this.assets[assetId];
554
- if (!asset)
555
- return undefined;
556
- // If there's a baseURL and the asset has a src property, prepend the baseURL
557
- if (this.baseURL && asset.src) {
558
- // Create a shallow copy to avoid mutating the original asset
559
- return {
560
- ...asset,
561
- src: this.prependBaseURL(asset.src)
562
- };
563
- }
564
- return asset;
1131
+ }
1132
+ /**
1133
+ * <cv-toggle> element - represents a toggleable content block
1134
+ */
1135
+ class CVToggle extends HTMLElement {
1136
+ connectedCallback() {
1137
+ // Element is managed by Core
565
1138
  }
566
- prependBaseURL(path) {
567
- // Don't prepend if the path is already absolute (starts with http:// or https://)
568
- if (path.startsWith('http://') || path.startsWith('https://')) {
569
- return path;
1139
+ }
1140
+ /**
1141
+ * Register custom elements
1142
+ */
1143
+ function registerCustomElements() {
1144
+ // Only register if not already defined
1145
+ if (!customElements.get('cv-tab')) {
1146
+ customElements.define('cv-tab', CVTab);
1147
+ }
1148
+ if (!customElements.get('cv-tabgroup')) {
1149
+ customElements.define('cv-tabgroup', CVTabgroup);
1150
+ }
1151
+ if (!customElements.get('cv-toggle')) {
1152
+ customElements.define('cv-toggle', CVToggle);
1153
+ }
1154
+ }
1155
+
1156
+ /**
1157
+ * Main CustomViews class for initializing and managing custom views
1158
+ */
1159
+ class CustomViews {
1160
+ /**
1161
+ * Entry Point to use CustomViews
1162
+ * @param opts Initialization options including config object and assets path
1163
+ * @returns Promise resolving to the CustomViewsCore instance or null if initialization fails
1164
+ */
1165
+ static async init(opts) {
1166
+ // Register custom elements
1167
+ registerCustomElements();
1168
+ // Load assets JSON if provided
1169
+ let assetsManager;
1170
+ const baseURL = opts.baseURL || '';
1171
+ if (opts.assetsJsonPath) {
1172
+ const assetsPath = prependBaseUrl(opts.assetsJsonPath, baseURL);
1173
+ const assetsJson = await (await fetch(assetsPath)).json();
1174
+ assetsManager = new AssetsManager(assetsJson, baseURL);
1175
+ }
1176
+ else {
1177
+ assetsManager = new AssetsManager({}, baseURL);
570
1178
  }
571
- // Ensure baseURL doesn't end with / and path starts with /
572
- const cleanBaseURL = this.baseURL.endsWith('/') ? this.baseURL.slice(0, -1) : this.baseURL;
573
- const cleanPath = path.startsWith('/') ? path : '/' + path;
574
- return cleanBaseURL + cleanPath;
575
- }
576
- loadFromJSON(json) {
577
- this.assets = json;
578
- }
579
- loadAdditionalAssets(additionalAssets) {
580
- this.assets = { ...this.assets, ...additionalAssets };
1179
+ // Use provided config or create a minimal default one
1180
+ let config;
1181
+ if (opts.config) {
1182
+ config = opts.config;
1183
+ }
1184
+ else {
1185
+ console.error("No config provided, using minimal default config");
1186
+ // Create a minimal default config
1187
+ config = { allToggles: [], defaultState: { toggles: [] } };
1188
+ }
1189
+ const coreOptions = {
1190
+ assetsManager,
1191
+ config: config,
1192
+ rootEl: opts.rootEl,
1193
+ };
1194
+ if (opts.showUrl !== undefined) {
1195
+ coreOptions.showUrl = opts.showUrl;
1196
+ }
1197
+ const core = new CustomViewsCore(coreOptions);
1198
+ core.init();
1199
+ return core;
581
1200
  }
582
1201
  }
583
1202
 
@@ -588,586 +1207,644 @@ class AssetsManager {
588
1207
  * Note: Styles are kept as a TypeScript string for compatibility with the build system.
589
1208
  * This approach ensures the styles are properly bundled and don't require separate CSS file handling.
590
1209
  */
591
- const WIDGET_STYLES = `
592
- /* Rounded rectangle widget icon styles */
593
- .cv-widget-icon {
594
- position: fixed;
595
- background: white;
596
- color: black;
597
- display: flex;
598
- align-items: center;
599
- justify-content: center;
600
- font-size: 18px;
601
- font-weight: bold;
602
- cursor: pointer;
603
- box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
604
- z-index: 9998;
605
- transition: all 0.3s ease;
606
- font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
607
- }
608
-
609
- .cv-widget-icon:hover {
610
- background: white;
611
- color: black;
612
- }
613
-
614
- /* Top-right: rounded end on left, sticks out leftward on hover */
615
- .cv-widget-top-right {
616
- top: 20px;
617
- right: 0;
618
- border-radius: 18px 0 0 18px;
619
- padding-left: 8px;
620
- justify-content: flex-start;
621
- }
622
-
623
- /* Top-left: rounded end on right, sticks out rightward on hover */
624
- .cv-widget-top-left {
625
- top: 20px;
626
- left: 0;
627
- border-radius: 0 18px 18px 0;
628
- padding-right: 8px;
629
- justify-content: flex-end;
630
- }
631
-
632
- /* Bottom-right: rounded end on left, sticks out leftward on hover */
633
- .cv-widget-bottom-right {
634
- bottom: 20px;
635
- right: 0;
636
- border-radius: 18px 0 0 18px;
637
- padding-left: 8px;
638
- justify-content: flex-start;
639
- }
640
-
641
- /* Bottom-left: rounded end on right, sticks out rightward on hover */
642
- .cv-widget-bottom-left {
643
- bottom: 20px;
644
- left: 0;
645
- border-radius: 0 18px 18px 0;
646
- padding-right: 8px;
647
- justify-content: flex-end;
648
- }
649
-
650
- /* Middle-left: rounded end on right, sticks out rightward on hover */
651
- .cv-widget-middle-left {
652
- top: 50%;
653
- left: 0;
654
- transform: translateY(-50%);
655
- border-radius: 0 18px 18px 0;
656
- padding-right: 8px;
657
- justify-content: flex-end;
658
- }
659
-
660
- /* Middle-right: rounded end on left, sticks out leftward on hover */
661
- .cv-widget-middle-right {
662
- top: 50%;
663
- right: 0;
664
- transform: translateY(-50%);
665
- border-radius: 18px 0 0 18px;
666
- padding-left: 8px;
667
- justify-content: flex-start;
668
- }
669
-
670
- .cv-widget-top-right,
671
- .cv-widget-middle-right,
672
- .cv-widget-bottom-right,
673
- .cv-widget-top-left,
674
- .cv-widget-middle-left,
675
- .cv-widget-bottom-left {
676
- height: 36px;
677
- width: 36px;
678
- }
679
-
680
- .cv-widget-middle-right:hover,
681
- .cv-widget-top-right:hover,
682
- .cv-widget-bottom-right:hover,
683
- .cv-widget-top-left:hover,
684
- .cv-widget-middle-left:hover,
685
- .cv-widget-bottom-left:hover {
686
- width: 55px;
687
- }
688
-
689
- /* Modal content styles */
690
- .cv-widget-section {
691
- margin-bottom: 16px;
692
- }
693
-
694
- .cv-widget-section:last-child {
695
- margin-bottom: 0;
696
- }
697
-
698
- .cv-widget-section label {
699
- display: block;
700
- margin-bottom: 4px;
701
- font-weight: 500;
702
- color: #555;
703
- }
704
-
705
- .cv-widget-profile-select,
706
- .cv-widget-state-select {
707
- width: 100%;
708
- padding: 8px 12px;
709
- border: 1px solid #ddd;
710
- border-radius: 4px;
711
- background: white;
712
- font-size: 14px;
713
- }
714
-
715
- .cv-widget-profile-select:focus,
716
- .cv-widget-state-select:focus {
717
- outline: none;
718
- border-color: #007bff;
719
- box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25);
720
- }
721
-
722
- .cv-widget-profile-select:disabled,
723
- .cv-widget-state-select:disabled {
724
- background: #f8f9fa;
725
- color: #6c757d;
726
- cursor: not-allowed;
727
- }
728
-
729
- .cv-widget-current {
730
- margin: 16px 0;
731
- padding: 12px;
732
- background: #f8f9fa;
733
- border-radius: 4px;
734
- border-left: 4px solid #007bff;
735
- }
736
-
737
- .cv-widget-current label {
738
- font-size: 12px;
739
- text-transform: uppercase;
740
- letter-spacing: 0.5px;
741
- color: #666;
742
- margin-bottom: 4px;
743
- }
744
-
745
- .cv-widget-current-view {
746
- font-weight: 500;
747
- color: #333;
748
- }
749
-
750
- .cv-widget-reset {
751
- width: 100%;
752
- padding: 8px 16px;
753
- background: #dc3545;
754
- color: white;
755
- border: none;
756
- border-radius: 4px;
757
- cursor: pointer;
758
- font-size: 14px;
759
- font-weight: 500;
760
- }
761
-
762
- .cv-widget-reset:hover {
763
- background: #c82333;
764
- }
765
-
766
- .cv-widget-reset:active {
767
- background: #bd2130;
768
- }
769
-
770
- /* Responsive design for mobile */
771
- @media (max-width: 768px) {
772
- .cv-widget-top-right,
773
- .cv-widget-top-left {
774
- top: 10px;
775
- }
776
-
777
- .cv-widget-bottom-right,
778
- .cv-widget-bottom-left {
779
- bottom: 10px;
780
- }
781
-
782
- /* All widgets stay flush with screen edges */
783
- .cv-widget-top-right,
784
- .cv-widget-bottom-right,
785
- .cv-widget-middle-right {
786
- right: 0;
787
- }
788
-
789
- .cv-widget-top-left,
790
- .cv-widget-bottom-left,
791
- .cv-widget-middle-left {
792
- left: 0;
793
- }
794
-
795
- /* Slightly smaller on mobile */
796
- .cv-widget-icon {
797
- width: 60px;
798
- height: 32px;
799
- }
800
-
801
- .cv-widget-icon:hover {
802
- width: 75px;
803
- }
804
- }
805
-
806
- /* Modal styles */
807
- .cv-widget-modal-overlay {
808
- position: fixed;
809
- top: 0;
810
- left: 0;
811
- right: 0;
812
- bottom: 0;
813
- background: rgba(0, 0, 0, 0.5);
814
- display: flex;
815
- align-items: center;
816
- justify-content: center;
817
- z-index: 10002;
818
- animation: fadeIn 0.2s ease;
819
- }
820
-
821
- @keyframes fadeIn {
822
- from { opacity: 0; }
823
- to { opacity: 1; }
824
- }
825
-
826
- .cv-widget-modal {
827
- background: white;
828
- border-radius: 8px;
829
- box-shadow: 0 8px 32px rgba(0, 0, 0, 0.2);
830
- max-width: 400px;
831
- width: 90vw;
832
- max-height: 80vh;
833
- overflow-y: auto;
834
- animation: slideIn 0.2s ease;
835
- }
836
-
837
- @keyframes slideIn {
838
- from {
839
- opacity: 0;
840
- transform: scale(0.9) translateY(-20px);
841
- }
842
- to {
843
- opacity: 1;
844
- transform: scale(1) translateY(0);
845
- }
846
- }
847
-
848
- .cv-widget-modal-header {
849
- display: flex;
850
- justify-content: space-between;
851
- align-items: center;
852
- padding: 16px 20px;
853
- border-bottom: 1px solid #e9ecef;
854
- background: #f8f9fa;
855
- border-radius: 8px 8px 0 0;
856
- }
857
-
858
- .cv-widget-modal-header h3 {
859
- margin: 0;
860
- font-size: 18px;
861
- font-weight: 600;
862
- color: #333;
863
- }
864
-
865
- .cv-widget-modal-close {
866
- background: none;
867
- border: none;
868
- font-size: 24px;
869
- cursor: pointer;
870
- padding: 0;
871
- width: 32px;
872
- height: 32px;
873
- display: flex;
874
- align-items: center;
875
- justify-content: center;
876
- border-radius: 4px;
877
- color: #666;
878
- }
879
-
880
- .cv-widget-modal-close:hover {
881
- background: #e9ecef;
882
- }
883
-
884
- .cv-widget-modal-content {
885
- padding: 20px;
886
- }
887
-
888
- .cv-widget-modal-actions {
889
- margin-top: 20px;
890
- padding-top: 16px;
891
- border-top: 1px solid #e9ecef;
892
- }
893
-
894
- .cv-widget-restore {
895
- width: 100%;
896
- padding: 10px 16px;
897
- background: #28a745;
898
- color: white;
899
- border: none;
900
- border-radius: 4px;
901
- cursor: pointer;
902
- font-size: 14px;
903
- font-weight: 500;
904
- }
905
-
906
- .cv-widget-restore:hover {
907
- background: #218838;
908
- }
909
-
910
- .cv-widget-create-state {
911
- width: 100%;
912
- padding: 10px 16px;
913
- background: #007bff;
914
- color: white;
915
- border: none;
916
- border-radius: 4px;
917
- cursor: pointer;
918
- font-size: 14px;
919
- font-weight: 500;
920
- margin-bottom: 10px;
921
- }
922
-
923
- .cv-widget-create-state:hover {
924
- background: #0056b3;
925
- }
926
-
927
- /* Dark theme modal styles */
928
- .cv-widget-theme-dark .cv-widget-modal {
929
- background: #2d3748;
930
- color: #e2e8f0;
931
- }
932
-
933
- .cv-widget-theme-dark .cv-widget-modal-header {
934
- background: #1a202c;
935
- border-color: #4a5568;
936
- }
937
-
938
- .cv-widget-theme-dark .cv-widget-modal-header h3 {
939
- color: #e2e8f0;
940
- }
941
-
942
- .cv-widget-theme-dark .cv-widget-modal-close {
943
- color: #a0aec0;
944
- }
945
-
946
- .cv-widget-theme-dark .cv-widget-modal-close:hover {
947
- background: #4a5568;
948
- }
949
-
950
- .cv-widget-theme-dark .cv-widget-modal-actions {
951
- border-color: #4a5568;
952
- }
953
-
954
- /* Custom state creator styles */
955
- .cv-custom-state-modal {
956
- max-width: 500px;
957
- }
958
-
959
- .cv-custom-state-form h4 {
960
- margin: 20px 0 10px 0;
961
- font-size: 16px;
962
- font-weight: 600;
963
- color: #333;
964
- border-bottom: 1px solid #e9ecef;
965
- padding-bottom: 5px;
966
- }
967
-
968
- .cv-custom-state-section {
969
- margin-bottom: 16px;
970
- }
971
-
972
- .cv-custom-state-section label {
973
- display: block;
974
- margin-bottom: 4px;
975
- font-weight: 500;
976
- color: #555;
977
- }
978
-
979
- .cv-custom-state-input {
980
- width: 100%;
981
- padding: 8px 12px;
982
- border: 1px solid #ddd;
983
- border-radius: 4px;
984
- background: white;
985
- font-size: 14px;
986
- }
987
-
988
- .cv-custom-state-input:focus {
989
- outline: none;
990
- border-color: #007bff;
991
- box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25);
992
- }
993
-
994
- .cv-custom-toggles {
995
- display: grid;
996
- grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
997
- gap: 10px;
998
- }
999
-
1000
- .cv-custom-state-toggle {
1001
- display: flex;
1002
- align-items: center;
1003
- }
1004
-
1005
- .cv-custom-state-toggle label {
1006
- display: flex;
1007
- align-items: center;
1008
- cursor: pointer;
1009
- font-weight: normal;
1010
- margin: 0;
1011
- }
1012
-
1013
- .cv-custom-toggle-checkbox {
1014
- margin-right: 8px;
1015
- width: auto;
1016
- }
1017
-
1018
- .cv-custom-state-actions {
1019
- display: flex;
1020
- gap: 10px;
1021
- margin-top: 20px;
1022
- padding-top: 16px;
1023
- border-top: 1px solid #e9ecef;
1024
- }
1025
-
1026
- .cv-custom-state-cancel,
1027
- .cv-custom-state-copy-url {
1028
- flex: 1;
1029
- padding: 10px 16px;
1030
- border: none;
1031
- border-radius: 4px;
1032
- cursor: pointer;
1033
- font-size: 14px;
1034
- font-weight: 500;
1035
- }
1036
-
1037
- .cv-custom-state-reset {
1038
- flex: 1;
1039
- padding: 10px 16px;
1040
- border: none;
1041
- border-radius: 4px;
1042
- cursor: pointer;
1043
- font-size: 14px;
1044
- font-weight: 500;
1045
- background: #dc3545;
1046
- color: white;
1047
- }
1048
-
1049
- .cv-custom-state-reset:hover {
1050
- background: #c82333;
1051
- }
1052
-
1053
- .cv-custom-state-cancel {
1054
- background: #6c757d;
1055
- color: white;
1056
- }
1057
-
1058
- .cv-custom-state-cancel:hover {
1059
- background: #5a6268;
1060
- }
1061
-
1062
- .cv-custom-state-copy-url {
1063
- background: #28a745;
1064
- color: white;
1065
- }
1066
-
1067
- .cv-custom-state-copy-url:hover {
1068
- background: #218838;
1069
- }
1070
-
1071
- /* Dark theme custom state styles */
1072
- .cv-widget-theme-dark .cv-custom-state-form h4 {
1073
- color: #e2e8f0;
1074
- border-color: #4a5568;
1075
- }
1076
-
1077
- .cv-widget-theme-dark .cv-custom-state-section label {
1078
- color: #a0aec0;
1079
- }
1080
-
1081
- .cv-widget-theme-dark .cv-custom-state-input {
1082
- background: #1a202c;
1083
- border-color: #4a5568;
1084
- color: #e2e8f0;
1085
- }
1086
-
1087
- .cv-widget-theme-dark .cv-custom-state-actions {
1088
- border-color: #4a5568;
1089
- }
1090
-
1091
- /* Welcome modal styles */
1092
- .cv-welcome-modal {
1093
- max-width: 500px;
1094
- }
1095
-
1096
- .cv-welcome-content {
1097
- text-align: center;
1098
- }
1099
-
1100
- .cv-welcome-content p {
1101
- font-size: 15px;
1102
- line-height: 1.6;
1103
- color: #555;
1104
- margin-bottom: 24px;
1105
- }
1106
-
1107
- .cv-welcome-widget-preview {
1108
- display: flex;
1109
- flex-direction: column;
1110
- align-items: center;
1111
- gap: 12px;
1112
- padding: 20px;
1113
- background: #f8f9fa;
1114
- border-radius: 8px;
1115
- margin-bottom: 24px;
1116
- }
1117
-
1118
- .cv-welcome-widget-icon {
1119
- width: 36px;
1120
- height: 36px;
1121
- background: white;
1122
- color: black;
1123
- border-radius: 0 18px 18px 0;
1124
- display: flex;
1125
- align-items: center;
1126
- justify-content: center;
1127
- font-size: 18px;
1128
- box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
1129
- }
1130
-
1131
- .cv-welcome-widget-label {
1132
- font-size: 14px;
1133
- color: #666;
1134
- margin: 0;
1135
- font-weight: 500;
1136
- }
1137
-
1138
- .cv-welcome-got-it {
1139
- width: 100%;
1140
- padding: 12px 24px;
1141
- background: #007bff;
1142
- color: white;
1143
- border: none;
1144
- border-radius: 4px;
1145
- cursor: pointer;
1146
- font-size: 16px;
1147
- font-weight: 600;
1148
- transition: background 0.2s ease;
1149
- }
1150
-
1151
- .cv-welcome-got-it:hover {
1152
- background: #0056b3;
1153
- }
1154
-
1155
- .cv-welcome-got-it:active {
1156
- background: #004494;
1157
- }
1158
-
1159
- /* Dark theme welcome modal styles */
1160
- .cv-widget-theme-dark .cv-welcome-content p {
1161
- color: #cbd5e0;
1162
- }
1163
-
1164
- .cv-widget-theme-dark .cv-welcome-widget-preview {
1165
- background: #1a202c;
1166
- }
1167
-
1168
- .cv-widget-theme-dark .cv-welcome-widget-label {
1169
- color: #a0aec0;
1170
- }
1210
+ const WIDGET_STYLES = `
1211
+ /* Rounded rectangle widget icon styles */
1212
+ .cv-widget-icon {
1213
+ position: fixed;
1214
+ /* Slightly transparent by default so the widget is subtle at the page edge */
1215
+ background: rgba(255, 255, 255, 0.92);
1216
+ color: rgba(0, 0, 0, 0.9);
1217
+ opacity: 0.6;
1218
+ display: flex;
1219
+ align-items: center;
1220
+ justify-content: center;
1221
+ font-size: 18px;
1222
+ font-weight: bold;
1223
+ cursor: pointer;
1224
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
1225
+ z-index: 9998;
1226
+ transition: all 0.3s ease;
1227
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
1228
+ }
1229
+
1230
+ .cv-widget-icon:hover {
1231
+ /* Become fully opaque on hover to improve readability */
1232
+ background: rgba(255, 255, 255, 1);
1233
+ color: rgba(0, 0, 0, 1);
1234
+ opacity: 1;
1235
+ }
1236
+
1237
+ /* Top-right: rounded end on left, sticks out leftward on hover */
1238
+ .cv-widget-top-right {
1239
+ top: 20px;
1240
+ right: 0;
1241
+ border-radius: 18px 0 0 18px;
1242
+ padding-left: 8px;
1243
+ justify-content: flex-start;
1244
+ }
1245
+
1246
+ /* Top-left: rounded end on right, sticks out rightward on hover */
1247
+ .cv-widget-top-left {
1248
+ top: 20px;
1249
+ left: 0;
1250
+ border-radius: 0 18px 18px 0;
1251
+ padding-right: 8px;
1252
+ justify-content: flex-end;
1253
+ }
1254
+
1255
+ /* Bottom-right: rounded end on left, sticks out leftward on hover */
1256
+ .cv-widget-bottom-right {
1257
+ bottom: 20px;
1258
+ right: 0;
1259
+ border-radius: 18px 0 0 18px;
1260
+ padding-left: 8px;
1261
+ justify-content: flex-start;
1262
+ }
1263
+
1264
+ /* Bottom-left: rounded end on right, sticks out rightward on hover */
1265
+ .cv-widget-bottom-left {
1266
+ bottom: 20px;
1267
+ left: 0;
1268
+ border-radius: 0 18px 18px 0;
1269
+ padding-right: 8px;
1270
+ justify-content: flex-end;
1271
+ }
1272
+
1273
+ /* Middle-left: rounded end on right, sticks out rightward on hover */
1274
+ .cv-widget-middle-left {
1275
+ top: 50%;
1276
+ left: 0;
1277
+ transform: translateY(-50%);
1278
+ border-radius: 0 18px 18px 0;
1279
+ padding-right: 8px;
1280
+ justify-content: flex-end;
1281
+ }
1282
+
1283
+ /* Middle-right: rounded end on left, sticks out leftward on hover */
1284
+ .cv-widget-middle-right {
1285
+ top: 50%;
1286
+ right: 0;
1287
+ transform: translateY(-50%);
1288
+ border-radius: 18px 0 0 18px;
1289
+ padding-left: 8px;
1290
+ justify-content: flex-start;
1291
+ }
1292
+
1293
+ .cv-widget-top-right,
1294
+ .cv-widget-middle-right,
1295
+ .cv-widget-bottom-right,
1296
+ .cv-widget-top-left,
1297
+ .cv-widget-middle-left,
1298
+ .cv-widget-bottom-left {
1299
+ height: 36px;
1300
+ width: 36px;
1301
+ }
1302
+
1303
+ .cv-widget-middle-right:hover,
1304
+ .cv-widget-top-right:hover,
1305
+ .cv-widget-bottom-right:hover,
1306
+ .cv-widget-top-left:hover,
1307
+ .cv-widget-middle-left:hover,
1308
+ .cv-widget-bottom-left:hover {
1309
+ width: 55px;
1310
+ }
1311
+
1312
+ /* Modal content styles */
1313
+ .cv-widget-section {
1314
+ margin-bottom: 16px;
1315
+ }
1316
+
1317
+ .cv-widget-section:last-child {
1318
+ margin-bottom: 0;
1319
+ }
1320
+
1321
+ .cv-widget-section label {
1322
+ display: block;
1323
+ margin-bottom: 4px;
1324
+ font-weight: 500;
1325
+ color: #555;
1326
+ }
1327
+
1328
+ .cv-widget-profile-select,
1329
+ .cv-widget-state-select {
1330
+ width: 100%;
1331
+ padding: 8px 12px;
1332
+ border: 1px solid #ddd;
1333
+ border-radius: 4px;
1334
+ background: white;
1335
+ font-size: 14px;
1336
+ }
1337
+
1338
+ .cv-widget-profile-select:focus,
1339
+ .cv-widget-state-select:focus {
1340
+ outline: none;
1341
+ border-color: #007bff;
1342
+ box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25);
1343
+ }
1344
+
1345
+ .cv-widget-profile-select:disabled,
1346
+ .cv-widget-state-select:disabled {
1347
+ background: #f8f9fa;
1348
+ color: #6c757d;
1349
+ cursor: not-allowed;
1350
+ }
1351
+
1352
+ .cv-widget-current {
1353
+ margin: 16px 0;
1354
+ padding: 12px;
1355
+ background: #f8f9fa;
1356
+ border-radius: 4px;
1357
+ border-left: 4px solid #007bff;
1358
+ }
1359
+
1360
+ .cv-widget-current label {
1361
+ font-size: 12px;
1362
+ text-transform: uppercase;
1363
+ letter-spacing: 0.5px;
1364
+ color: #666;
1365
+ margin-bottom: 4px;
1366
+ }
1367
+
1368
+ .cv-widget-current-view {
1369
+ font-weight: 500;
1370
+ color: #333;
1371
+ }
1372
+
1373
+ .cv-widget-reset {
1374
+ width: 100%;
1375
+ padding: 8px 16px;
1376
+ background: #dc3545;
1377
+ color: white;
1378
+ border: none;
1379
+ border-radius: 4px;
1380
+ cursor: pointer;
1381
+ font-size: 14px;
1382
+ font-weight: 500;
1383
+ }
1384
+
1385
+ .cv-widget-reset:hover {
1386
+ background: #c82333;
1387
+ }
1388
+
1389
+ .cv-widget-reset:active {
1390
+ background: #bd2130;
1391
+ }
1392
+
1393
+ /* Responsive design for mobile */
1394
+ @media (max-width: 768px) {
1395
+ .cv-widget-top-right,
1396
+ .cv-widget-top-left {
1397
+ top: 10px;
1398
+ }
1399
+
1400
+ .cv-widget-bottom-right,
1401
+ .cv-widget-bottom-left {
1402
+ bottom: 10px;
1403
+ }
1404
+
1405
+ /* All widgets stay flush with screen edges */
1406
+ .cv-widget-top-right,
1407
+ .cv-widget-bottom-right,
1408
+ .cv-widget-middle-right {
1409
+ right: 0;
1410
+ }
1411
+
1412
+ .cv-widget-top-left,
1413
+ .cv-widget-bottom-left,
1414
+ .cv-widget-middle-left {
1415
+ left: 0;
1416
+ }
1417
+
1418
+ /* Slightly smaller on mobile */
1419
+ .cv-widget-icon {
1420
+ width: 60px;
1421
+ height: 32px;
1422
+ }
1423
+
1424
+ .cv-widget-icon:hover {
1425
+ width: 75px;
1426
+ }
1427
+ }
1428
+
1429
+ /* Modal styles */
1430
+ .cv-widget-modal-overlay {
1431
+ position: fixed;
1432
+ top: 0;
1433
+ left: 0;
1434
+ right: 0;
1435
+ bottom: 0;
1436
+ background: rgba(0, 0, 0, 0.5);
1437
+ display: flex;
1438
+ align-items: center;
1439
+ justify-content: center;
1440
+ z-index: 10002;
1441
+ animation: fadeIn 0.2s ease;
1442
+ }
1443
+
1444
+ @keyframes fadeIn {
1445
+ from { opacity: 0; }
1446
+ to { opacity: 1; }
1447
+ }
1448
+
1449
+ .cv-widget-modal {
1450
+ background: white;
1451
+ border-radius: 8px;
1452
+ box-shadow: 0 8px 32px rgba(0, 0, 0, 0.2);
1453
+ max-width: 400px;
1454
+ width: 90vw;
1455
+ max-height: 80vh;
1456
+ overflow-y: auto;
1457
+ animation: slideIn 0.2s ease;
1458
+ }
1459
+
1460
+ @keyframes slideIn {
1461
+ from {
1462
+ opacity: 0;
1463
+ transform: scale(0.9) translateY(-20px);
1464
+ }
1465
+ to {
1466
+ opacity: 1;
1467
+ transform: scale(1) translateY(0);
1468
+ }
1469
+ }
1470
+
1471
+ .cv-widget-modal-header {
1472
+ display: flex;
1473
+ justify-content: space-between;
1474
+ align-items: center;
1475
+ padding: 16px 20px;
1476
+ border-bottom: 1px solid #e9ecef;
1477
+ background: #f8f9fa;
1478
+ border-radius: 8px 8px 0 0;
1479
+ }
1480
+
1481
+ .cv-widget-modal-header h3 {
1482
+ margin: 0;
1483
+ font-size: 18px;
1484
+ font-weight: 600;
1485
+ color: #333;
1486
+ }
1487
+
1488
+ .cv-widget-modal-close {
1489
+ background: none;
1490
+ border: none;
1491
+ font-size: 20px;
1492
+ cursor: pointer;
1493
+ padding: 0;
1494
+ width: 32px;
1495
+ height: 32px;
1496
+ display: flex;
1497
+ align-items: center;
1498
+ justify-content: center;
1499
+ border-radius: 4px;
1500
+ color: #666;
1501
+ line-height: 1;
1502
+ transition: all 0.2s ease;
1503
+ }
1504
+
1505
+ .cv-widget-modal-close:hover {
1506
+ background: #e9ecef;
1507
+ color: #333;
1508
+ }
1509
+
1510
+ .cv-widget-modal-content {
1511
+ padding: 20px;
1512
+ }
1513
+
1514
+ .cv-widget-modal-actions {
1515
+ margin-top: 20px;
1516
+ padding-top: 16px;
1517
+ border-top: 1px solid #e9ecef;
1518
+ }
1519
+
1520
+ .cv-widget-restore {
1521
+ width: 100%;
1522
+ padding: 10px 16px;
1523
+ background: #28a745;
1524
+ color: white;
1525
+ border: none;
1526
+ border-radius: 4px;
1527
+ cursor: pointer;
1528
+ font-size: 14px;
1529
+ font-weight: 500;
1530
+ }
1531
+
1532
+ .cv-widget-restore:hover {
1533
+ background: #218838;
1534
+ }
1535
+
1536
+ .cv-widget-create-state {
1537
+ width: 100%;
1538
+ padding: 10px 16px;
1539
+ background: #007bff;
1540
+ color: white;
1541
+ border: none;
1542
+ border-radius: 4px;
1543
+ cursor: pointer;
1544
+ font-size: 14px;
1545
+ font-weight: 500;
1546
+ margin-bottom: 10px;
1547
+ }
1548
+
1549
+ .cv-widget-create-state:hover {
1550
+ background: #0056b3;
1551
+ }
1552
+
1553
+ /* Dark theme modal styles */
1554
+ .cv-widget-theme-dark .cv-widget-modal {
1555
+ background: #2d3748;
1556
+ color: #e2e8f0;
1557
+ }
1558
+
1559
+ .cv-widget-theme-dark .cv-widget-modal-header {
1560
+ background: #1a202c;
1561
+ border-color: #4a5568;
1562
+ }
1563
+
1564
+ .cv-widget-theme-dark .cv-widget-modal-header h3 {
1565
+ color: #e2e8f0;
1566
+ }
1567
+
1568
+ .cv-widget-theme-dark .cv-widget-modal-close {
1569
+ color: #a0aec0;
1570
+ }
1571
+
1572
+ .cv-widget-theme-dark .cv-widget-modal-close:hover {
1573
+ background: #4a5568;
1574
+ color: #e2e8f0;
1575
+ }
1576
+
1577
+ .cv-widget-theme-dark .cv-widget-modal-actions {
1578
+ border-color: #4a5568;
1579
+ }
1580
+
1581
+ /* Custom state creator styles */
1582
+ .cv-custom-state-modal {
1583
+ max-width: 500px;
1584
+ }
1585
+
1586
+ .cv-custom-state-form h4 {
1587
+ margin: 20px 0 10px 0;
1588
+ font-size: 16px;
1589
+ font-weight: 600;
1590
+ color: #333;
1591
+ border-bottom: 1px solid #e9ecef;
1592
+ padding-bottom: 5px;
1593
+ }
1594
+
1595
+ .cv-custom-state-form p {
1596
+ font-size: 15px;
1597
+ line-height: 1.6;
1598
+ color: #555;
1599
+ margin-bottom: 24px;
1600
+ text-align: justify;
1601
+ }
1602
+
1603
+ .cv-custom-state-section {
1604
+ margin-bottom: 16px;
1605
+ }
1606
+
1607
+ .cv-custom-state-section label {
1608
+ display: block;
1609
+ margin-bottom: 4px;
1610
+ font-weight: 500;
1611
+ color: #555;
1612
+ }
1613
+
1614
+ .cv-custom-state-input {
1615
+ width: 100%;
1616
+ padding: 8px 12px;
1617
+ border: 1px solid #ddd;
1618
+ border-radius: 4px;
1619
+ background: white;
1620
+ font-size: 14px;
1621
+ }
1622
+
1623
+ .cv-custom-state-input:focus {
1624
+ outline: none;
1625
+ border-color: #007bff;
1626
+ box-shadow: 0 0 0 2px rgba(0, 123, 255, 0.25);
1627
+ }
1628
+
1629
+ .cv-custom-toggles {
1630
+ display: grid;
1631
+ grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
1632
+ gap: 10px;
1633
+ }
1634
+
1635
+ .cv-custom-state-toggle {
1636
+ display: flex;
1637
+ align-items: center;
1638
+ }
1639
+
1640
+ .cv-custom-state-toggle label {
1641
+ display: flex;
1642
+ align-items: center;
1643
+ cursor: pointer;
1644
+ font-weight: normal;
1645
+ margin: 0;
1646
+ }
1647
+
1648
+ .cv-custom-toggle-checkbox {
1649
+ margin-right: 8px;
1650
+ width: auto;
1651
+ }
1652
+
1653
+ .cv-tab-groups {
1654
+ margin-top: 20px;
1655
+ }
1656
+
1657
+ .cv-tab-group-control {
1658
+ margin-bottom: 15px;
1659
+ }
1660
+
1661
+ .cv-tab-group-control label {
1662
+ display: block;
1663
+ margin-bottom: 5px;
1664
+ font-weight: 500;
1665
+ font-size: 14px;
1666
+ }
1667
+
1668
+ .cv-tab-group-select {
1669
+ width: 100%;
1670
+ padding: 8px 12px;
1671
+ border: 1px solid #ced4da;
1672
+ border-radius: 4px;
1673
+ font-size: 14px;
1674
+ background-color: white;
1675
+ cursor: pointer;
1676
+ }
1677
+
1678
+ .cv-tab-group-select:focus {
1679
+ outline: none;
1680
+ border-color: #007bff;
1681
+ box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);
1682
+ }
1683
+
1684
+ .cv-widget-theme-dark .cv-tab-group-select {
1685
+ background-color: #2d3748;
1686
+ border-color: #4a5568;
1687
+ color: #e2e8f0;
1688
+ }
1689
+
1690
+ .cv-custom-state-actions {
1691
+ display: flex;
1692
+ gap: 10px;
1693
+ margin-top: 20px;
1694
+ padding-top: 16px;
1695
+ border-top: 1px solid #e9ecef;
1696
+ }
1697
+
1698
+ .cv-custom-state-cancel,
1699
+ .cv-custom-state-copy-url {
1700
+ flex: 1;
1701
+ padding: 10px 16px;
1702
+ border: none;
1703
+ border-radius: 4px;
1704
+ cursor: pointer;
1705
+ font-size: 14px;
1706
+ font-weight: 500;
1707
+ }
1708
+
1709
+ .cv-custom-state-reset {
1710
+ flex: 1;
1711
+ padding: 10px 16px;
1712
+ border: none;
1713
+ border-radius: 4px;
1714
+ cursor: pointer;
1715
+ font-size: 14px;
1716
+ font-weight: 500;
1717
+ background: #dc3545;
1718
+ color: white;
1719
+ }
1720
+
1721
+ .cv-custom-state-reset:hover {
1722
+ background: #c82333;
1723
+ }
1724
+
1725
+ .cv-custom-state-cancel {
1726
+ background: #6c757d;
1727
+ color: white;
1728
+ }
1729
+
1730
+ .cv-custom-state-cancel:hover {
1731
+ background: #5a6268;
1732
+ }
1733
+
1734
+ .cv-custom-state-copy-url {
1735
+ background: #28a745;
1736
+ color: white;
1737
+ }
1738
+
1739
+ .cv-custom-state-copy-url:hover {
1740
+ background: #218838;
1741
+ }
1742
+
1743
+ /* Dark theme custom state styles */
1744
+ .cv-widget-theme-dark .cv-custom-state-form h4 {
1745
+ color: #e2e8f0;
1746
+ border-color: #4a5568;
1747
+ }
1748
+
1749
+ .cv-widget-theme-dark .cv-custom-state-form p {
1750
+ color: #cbd5e0;
1751
+ }
1752
+
1753
+ .cv-widget-theme-dark .cv-custom-state-section label {
1754
+ color: #a0aec0;
1755
+ }
1756
+
1757
+ .cv-widget-theme-dark .cv-custom-state-input {
1758
+ background: #1a202c;
1759
+ border-color: #4a5568;
1760
+ color: #e2e8f0;
1761
+ }
1762
+
1763
+ .cv-widget-theme-dark .cv-custom-state-actions {
1764
+ border-color: #4a5568;
1765
+ }
1766
+
1767
+ /* Welcome modal styles */
1768
+ .cv-welcome-modal {
1769
+ max-width: 500px;
1770
+ }
1771
+
1772
+ .cv-welcome-content {
1773
+ text-align: center;
1774
+ }
1775
+
1776
+ .cv-welcome-content p {
1777
+ font-size: 15px;
1778
+ line-height: 1.6;
1779
+ color: #555;
1780
+ margin-bottom: 24px;
1781
+ text-align: justify;
1782
+ }
1783
+
1784
+ .cv-welcome-widget-preview {
1785
+ display: flex;
1786
+ flex-direction: column;
1787
+ align-items: center;
1788
+ gap: 12px;
1789
+ padding: 20px;
1790
+ background: #f8f9fa;
1791
+ border-radius: 8px;
1792
+ margin-bottom: 24px;
1793
+ }
1794
+
1795
+ .cv-welcome-widget-icon {
1796
+ width: 36px;
1797
+ height: 36px;
1798
+ background: white;
1799
+ color: black;
1800
+ border-radius: 0 18px 18px 0;
1801
+ display: flex;
1802
+ align-items: center;
1803
+ justify-content: center;
1804
+ font-size: 18px;
1805
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
1806
+ }
1807
+
1808
+ .cv-welcome-widget-label {
1809
+ font-size: 14px;
1810
+ color: #666;
1811
+ margin: 0;
1812
+ font-weight: 500;
1813
+ }
1814
+
1815
+ .cv-welcome-got-it {
1816
+ width: 100%;
1817
+ padding: 12px 24px;
1818
+ background: #007bff;
1819
+ color: white;
1820
+ border: none;
1821
+ border-radius: 4px;
1822
+ cursor: pointer;
1823
+ font-size: 16px;
1824
+ font-weight: 600;
1825
+ transition: background 0.2s ease;
1826
+ }
1827
+
1828
+ .cv-welcome-got-it:hover {
1829
+ background: #0056b3;
1830
+ }
1831
+
1832
+ .cv-welcome-got-it:active {
1833
+ background: #004494;
1834
+ }
1835
+
1836
+ /* Dark theme welcome modal styles */
1837
+ .cv-widget-theme-dark .cv-welcome-content p {
1838
+ color: #cbd5e0;
1839
+ }
1840
+
1841
+ .cv-widget-theme-dark .cv-welcome-widget-preview {
1842
+ background: #1a202c;
1843
+ }
1844
+
1845
+ .cv-widget-theme-dark .cv-welcome-widget-label {
1846
+ color: #a0aec0;
1847
+ }
1171
1848
  `;
1172
1849
  /**
1173
1850
  * Inject widget styles into the document head
@@ -1202,8 +1879,9 @@ class CustomViewsWidget {
1202
1879
  title: options.title || 'Custom Views',
1203
1880
  description: options.description || 'Toggle different content sections to customize your view. Changes are applied instantly and the URL will be updated for sharing.',
1204
1881
  showWelcome: options.showWelcome ?? false,
1205
- welcomeTitle: options.welcomeTitle || 'Welcome to Custom Views!',
1206
- welcomeMessage: options.welcomeMessage || 'This website uses Custom Views to let you personalize your experience. Use the widget on the side (⚙) to show or hide different content sections based on your preferences. Your selections will be saved and can be shared via URL.'
1882
+ welcomeTitle: options.welcomeTitle || 'This website uses Custom Views',
1883
+ welcomeMessage: options.welcomeMessage || 'This site uses Custom Views to let you personalize your experience. Use the widget on the side (⚙) to show or hide different content sections based on your preferences. Your selections will be saved and can be shared via URL.',
1884
+ showTabGroups: options.showTabGroups ?? true
1207
1885
  };
1208
1886
  // No external state manager to initialize
1209
1887
  }
@@ -1268,8 +1946,8 @@ class CustomViewsWidget {
1268
1946
  */
1269
1947
  openStateModal() {
1270
1948
  // Get toggles from current configuration and open the modal regardless of count
1271
- const localConfig = this.core.getLocalConfig();
1272
- const toggles = localConfig?.allToggles || [];
1949
+ const config = this.core.getConfig();
1950
+ const toggles = config?.allToggles || [];
1273
1951
  this.createCustomStateModal(toggles);
1274
1952
  }
1275
1953
  /**
@@ -1282,37 +1960,61 @@ class CustomViewsWidget {
1282
1960
  this.modal.className = 'cv-widget-modal-overlay';
1283
1961
  this.applyThemeToModal();
1284
1962
  const toggleControls = toggles.length
1285
- ? toggles.map(toggle => `
1286
- <div class="cv-custom-state-toggle">
1287
- <label>
1288
- <input type="checkbox" class="cv-custom-toggle-checkbox" data-toggle="${toggle}" />
1289
- ${this.formatToggleName(toggle)}
1290
- </label>
1291
- </div>
1963
+ ? toggles.map(toggle => `
1964
+ <div class="cv-custom-state-toggle">
1965
+ <label>
1966
+ <input type="checkbox" class="cv-custom-toggle-checkbox" data-toggle="${toggle}" />
1967
+ ${this.formatToggleName(toggle)}
1968
+ </label>
1969
+ </div>
1292
1970
  `).join('')
1293
1971
  : `<p class="cv-no-toggles">No configurable sections available.</p>`;
1294
- this.modal.innerHTML = `
1295
- <div class="cv-widget-modal cv-custom-state-modal">
1296
- <div class="cv-widget-modal-header">
1297
- <h3>Customize View</h3>
1298
- <button class="cv-widget-modal-close" aria-label="Close modal">X</button>
1299
- </div>
1300
- <div class="cv-widget-modal-content">
1301
- <div class="cv-custom-state-form">
1302
- <p>${this.options.description}</p>
1303
-
1304
- <h4>Content Sections</h4>
1305
- <div class="cv-custom-toggles">
1306
- ${toggleControls}
1307
- </div>
1308
-
1309
- <div class="cv-custom-state-actions">
1310
- ${this.options.showReset ? `<button class="cv-custom-state-reset">Reset to Default</button>` : ''}
1311
- <button class="cv-custom-state-copy-url">Copy Shareable URL</button>
1312
- </div>
1313
- </div>
1314
- </div>
1315
- </div>
1972
+ // Get tab groups
1973
+ const tabGroups = this.core.getTabGroups();
1974
+ let tabGroupsHTML = '';
1975
+ if (this.options.showTabGroups && tabGroups && tabGroups.length > 0) {
1976
+ const tabGroupControls = tabGroups.map(group => {
1977
+ const options = group.tabs.map(tab => `<option value="${tab.id}">${tab.label || tab.id}</option>`).join('');
1978
+ return `
1979
+ <div class="cv-tab-group-control">
1980
+ <label for="tab-group-${group.id}">${group.label || group.id}</label>
1981
+ <select id="tab-group-${group.id}" class="cv-tab-group-select" data-group-id="${group.id}">
1982
+ ${options}
1983
+ </select>
1984
+ </div>
1985
+ `;
1986
+ }).join('');
1987
+ tabGroupsHTML = `
1988
+ <h4>Tab Groups</h4>
1989
+ <div class="cv-tab-groups">
1990
+ ${tabGroupControls}
1991
+ </div>
1992
+ `;
1993
+ }
1994
+ this.modal.innerHTML = `
1995
+ <div class="cv-widget-modal cv-custom-state-modal">
1996
+ <div class="cv-widget-modal-header">
1997
+ <h3>Customize View</h3>
1998
+ <button class="cv-widget-modal-close" aria-label="Close modal">×</button>
1999
+ </div>
2000
+ <div class="cv-widget-modal-content">
2001
+ <div class="cv-custom-state-form">
2002
+ <p>${this.options.description}</p>
2003
+
2004
+ <h4>Content Sections</h4>
2005
+ <div class="cv-custom-toggles">
2006
+ ${toggleControls}
2007
+ </div>
2008
+
2009
+ ${tabGroupsHTML}
2010
+
2011
+ <div class="cv-custom-state-actions">
2012
+ ${this.options.showReset ? `<button class="cv-custom-state-reset">Reset to Default</button>` : ''}
2013
+ <button class="cv-custom-state-copy-url">Copy Shareable URL</button>
2014
+ </div>
2015
+ </div>
2016
+ </div>
2017
+ </div>
1316
2018
  `;
1317
2019
  document.body.appendChild(this.modal);
1318
2020
  this.attachStateModalEventListeners();
@@ -1355,6 +2057,17 @@ class CustomViewsWidget {
1355
2057
  this.core.applyState(state);
1356
2058
  });
1357
2059
  });
2060
+ // Listen to tab group selects
2061
+ const tabGroupSelects = this.modal.querySelectorAll('.cv-tab-group-select');
2062
+ tabGroupSelects.forEach(select => {
2063
+ select.addEventListener('change', () => {
2064
+ const groupId = select.dataset.groupId;
2065
+ const tabId = select.value;
2066
+ if (groupId && tabId) {
2067
+ this.core.setActiveTab(groupId, tabId);
2068
+ }
2069
+ });
2070
+ });
1358
2071
  // Overlay click to close
1359
2072
  this.modal.addEventListener('click', (e) => {
1360
2073
  if (e.target === this.modal) {
@@ -1399,7 +2112,16 @@ class CustomViewsWidget {
1399
2112
  toggles.push(toggle);
1400
2113
  }
1401
2114
  });
1402
- return { toggles };
2115
+ // Collect tab selections
2116
+ const tabGroupSelects = this.modal.querySelectorAll('.cv-tab-group-select');
2117
+ const tabs = {};
2118
+ tabGroupSelects.forEach(select => {
2119
+ const groupId = select.dataset.groupId;
2120
+ if (groupId) {
2121
+ tabs[groupId] = select.value;
2122
+ }
2123
+ });
2124
+ return Object.keys(tabs).length > 0 ? { toggles, tabs } : { toggles };
1403
2125
  }
1404
2126
  /**
1405
2127
  * Copy shareable URL to clipboard
@@ -1435,6 +2157,15 @@ class CustomViewsWidget {
1435
2157
  }
1436
2158
  }
1437
2159
  });
2160
+ // Load tab group selections
2161
+ const activeTabs = this.core.getCurrentActiveTabs();
2162
+ const tabGroupSelects = this.modal.querySelectorAll('.cv-tab-group-select');
2163
+ tabGroupSelects.forEach(select => {
2164
+ const groupId = select.dataset.groupId;
2165
+ if (groupId && activeTabs[groupId]) {
2166
+ select.value = activeTabs[groupId];
2167
+ }
2168
+ });
1438
2169
  }
1439
2170
  /**
1440
2171
  * Format toggle name for display
@@ -1468,25 +2199,25 @@ class CustomViewsWidget {
1468
2199
  this.modal = document.createElement('div');
1469
2200
  this.modal.className = 'cv-widget-modal-overlay cv-welcome-modal-overlay';
1470
2201
  this.applyThemeToModal();
1471
- this.modal.innerHTML = `
1472
- <div class="cv-widget-modal cv-welcome-modal">
1473
- <div class="cv-widget-modal-header">
1474
- <h3>${this.options.welcomeTitle}</h3>
1475
- <button class="cv-widget-modal-close" aria-label="Close modal">×</button>
1476
- </div>
1477
- <div class="cv-widget-modal-content">
1478
- <div class="cv-welcome-content">
1479
- <p>${this.options.welcomeMessage}</p>
1480
-
1481
- <div class="cv-welcome-widget-preview">
1482
- <div class="cv-welcome-widget-icon">⚙</div>
1483
- <p class="cv-welcome-widget-label">Look for this widget on the side of the screen</p>
1484
- </div>
1485
-
1486
- <button class="cv-welcome-got-it">Got it!</button>
1487
- </div>
1488
- </div>
1489
- </div>
2202
+ this.modal.innerHTML = `
2203
+ <div class="cv-widget-modal cv-welcome-modal">
2204
+ <div class="cv-widget-modal-header">
2205
+ <h3>${this.options.welcomeTitle}</h3>
2206
+ <button class="cv-widget-modal-close" aria-label="Close modal">×</button>
2207
+ </div>
2208
+ <div class="cv-widget-modal-content">
2209
+ <div class="cv-welcome-content">
2210
+ <p>${this.options.welcomeMessage}</p>
2211
+
2212
+ <div class="cv-welcome-widget-preview">
2213
+ <div class="cv-welcome-widget-icon">⚙</div>
2214
+ <p class="cv-welcome-widget-label">Look for this widget on the side of the screen</p>
2215
+ </div>
2216
+
2217
+ <button class="cv-welcome-got-it">Got it!</button>
2218
+ </div>
2219
+ </div>
2220
+ </div>
1490
2221
  `;
1491
2222
  document.body.appendChild(this.modal);
1492
2223
  this.attachWelcomeModalEventListeners();
@@ -1528,68 +2259,155 @@ class CustomViewsWidget {
1528
2259
  }
1529
2260
  }
1530
2261
 
1531
- class CustomViews {
1532
- // Helper function to prepend baseURL to a path
1533
- static prependBaseURL(path, baseURL) {
1534
- if (!baseURL)
1535
- return path;
1536
- // Don't prepend if the path is already absolute (starts with http:// or https://)
1537
- if (path.startsWith('http://') || path.startsWith('https://')) {
1538
- return path;
1539
- }
1540
- // Ensure baseURL doesn't end with / and path starts with /
1541
- const cleanBaseURL = baseURL.endsWith('/') ? baseURL.slice(0, -1) : baseURL;
1542
- const cleanPath = path.startsWith('/') ? path : '/' + path;
1543
- return cleanBaseURL + cleanPath;
2262
+ /**
2263
+ * Initialize CustomViews from script tag attributes and config file
2264
+ * This function handles the automatic initialization of CustomViews when included via script tag
2265
+ *
2266
+ * Data attributes supported:
2267
+ * - data-base-url: Base URL for the site (e.g., "/customviews" for subdirectory deployments)
2268
+ * - data-config-path: Path to config file (default: "/customviews.config.json")
2269
+ *
2270
+ * The function fetches the config file and uses it directly to initialize CustomViews.
2271
+ * Widget visibility is controlled via the config file (widget.enabled property).
2272
+ */
2273
+ function initializeFromScript() {
2274
+ // Only run in browser environment
2275
+ if (typeof window === 'undefined')
2276
+ return;
2277
+ // Use the typed global `window` (see src/types/global.d.ts)
2278
+ // Idempotency guard: if already initialized, skip setting up listener again.
2279
+ if (window.__customViewsInitialized) {
2280
+ // Informational for developers; harmless in production.
2281
+ console.info('[CustomViews] Auto-init skipped: already initialized.');
2282
+ return;
1544
2283
  }
1545
- // Entry Point to use CustomViews
1546
- static async initFromJson(opts) {
1547
- // Load assets JSON if provided
1548
- let assetsManager;
1549
- const baseURL = opts.baseURL || '';
1550
- if (opts.assetsJsonPath) {
1551
- const assetsPath = this.prependBaseURL(opts.assetsJsonPath, baseURL);
1552
- const assetsJson = await (await fetch(assetsPath)).json();
1553
- assetsManager = new AssetsManager(assetsJson, baseURL);
1554
- }
1555
- else {
1556
- assetsManager = new AssetsManager({}, baseURL);
1557
- }
1558
- // Load config JSON if provided, else just log error and don't load the custom views
1559
- let localConfig;
1560
- if (opts.config) {
1561
- localConfig = opts.config;
2284
+ document.addEventListener('DOMContentLoaded', async function () {
2285
+ // Prevent concurrent initialization runs (race conditions when script is loaded twice)
2286
+ if (window.__customViewsInitInProgress || window.__customViewsInitialized) {
2287
+ return;
1562
2288
  }
1563
- else {
1564
- if (!opts.configPath) {
1565
- console.error("No config path provided, skipping custom views");
1566
- return null;
2289
+ window.__customViewsInitInProgress = true;
2290
+ try {
2291
+ // Find the script tag
2292
+ let scriptTag = document.currentScript;
2293
+ // Fallback if currentScript is not available (executed after page load)
2294
+ if (!scriptTag) {
2295
+ // Try to find the script tag by looking for our script
2296
+ const scripts = document.querySelectorAll('script[src*="custom-views"]');
2297
+ if (scripts.length > 0) {
2298
+ // Find the most specific match (to avoid matching other custom-views scripts)
2299
+ for (let i = 0; i < scripts.length; i++) {
2300
+ const script = scripts[i];
2301
+ const src = script.getAttribute('src') || '';
2302
+ // Look for .min.js or .js at the end
2303
+ if (src.match(/custom-views(\.min)?\.js($|\?)/)) {
2304
+ scriptTag = script;
2305
+ break;
2306
+ }
2307
+ }
2308
+ // If no specific match found, use the first one
2309
+ if (!scriptTag) {
2310
+ scriptTag = scripts[0];
2311
+ }
2312
+ }
1567
2313
  }
2314
+ // Read data attributes from script tag
2315
+ let baseURL = '';
2316
+ let configPath = '/customviews.config.json';
2317
+ if (scriptTag) {
2318
+ baseURL = scriptTag.getAttribute('data-base-url') || '';
2319
+ configPath = scriptTag.getAttribute('data-config-path') || configPath;
2320
+ }
2321
+ // Fetch config file
2322
+ let configFile;
1568
2323
  try {
1569
- const configPath = this.prependBaseURL(opts.configPath, baseURL);
1570
- localConfig = await (await fetch(configPath)).json();
2324
+ const fullConfigPath = prependBaseUrl(configPath, baseURL);
2325
+ console.log(`[CustomViews] Loading config from: ${fullConfigPath}`);
2326
+ const response = await fetch(fullConfigPath);
2327
+ if (!response.ok) {
2328
+ console.warn(`[CustomViews] Config file not found at ${fullConfigPath}. Using defaults.`);
2329
+ // Provide minimal default config structure
2330
+ configFile = {
2331
+ config: {
2332
+ allToggles: [],
2333
+ defaultState: { toggles: [] }
2334
+ },
2335
+ widget: {
2336
+ enabled: true
2337
+ }
2338
+ };
2339
+ }
2340
+ else {
2341
+ configFile = await response.json();
2342
+ console.log('[CustomViews] Config loaded successfully');
2343
+ }
1571
2344
  }
1572
2345
  catch (error) {
1573
- console.error("Error loading config:", error);
1574
- return null;
2346
+ console.error('[CustomViews] Error loading config file:', error);
2347
+ return; // Abort initialization
2348
+ }
2349
+ // Determine effective baseURL (data attribute takes precedence over config)
2350
+ const effectiveBaseURL = baseURL || configFile.baseURL || '';
2351
+ const options = {
2352
+ config: configFile.config,
2353
+ assetsJsonPath: configFile.assetsJsonPath,
2354
+ baseURL: effectiveBaseURL,
2355
+ };
2356
+ if (configFile.showUrl !== undefined) {
2357
+ options.showUrl = configFile.showUrl;
2358
+ }
2359
+ // Initialize CustomViews core
2360
+ const core = await CustomViews.init(options);
2361
+ if (!core) {
2362
+ console.error('[CustomViews] Failed to initialize core.');
2363
+ return; // Abort widget creation
1575
2364
  }
2365
+ // Store instance
2366
+ window.customViewsInstance = { core };
2367
+ // Initialize widget if enabled in config
2368
+ let widget;
2369
+ if (configFile.widget?.enabled !== false) {
2370
+ widget = new CustomViewsWidget({
2371
+ core,
2372
+ ...configFile.widget
2373
+ });
2374
+ widget.render();
2375
+ // Store widget instance
2376
+ window.customViewsInstance.widget = widget;
2377
+ console.log('[CustomViews] Widget initialized and rendered');
2378
+ }
2379
+ else {
2380
+ console.log('[CustomViews] Widget disabled in config - skipping initialization');
2381
+ }
2382
+ // Dispatch ready event
2383
+ const readyEvent = new CustomEvent('customviews:ready', {
2384
+ detail: {
2385
+ core,
2386
+ widget
2387
+ }
2388
+ });
2389
+ document.dispatchEvent(readyEvent);
2390
+ // Mark initialized and clear in-progress flag
2391
+ window.__customViewsInitialized = true;
2392
+ window.__customViewsInitInProgress = false;
1576
2393
  }
1577
- const coreOptions = {
1578
- assetsManager,
1579
- config: localConfig,
1580
- rootEl: opts.rootEl,
1581
- };
1582
- const core = new CustomViewsCore(coreOptions);
1583
- core.init();
1584
- return core;
1585
- }
2394
+ catch (error) {
2395
+ // Clear in-progress flag so a future attempt can retry
2396
+ window.__customViewsInitInProgress = false;
2397
+ console.error('[CustomViews] Auto-initialization error:', error);
2398
+ }
2399
+ });
1586
2400
  }
2401
+
2402
+ // Import from new modules
2403
+ // Set up globals and auto-initialization
1587
2404
  if (typeof window !== "undefined") {
1588
- // @ts-ignore
2405
+ // Expose to window to enable usage (e.g. const app = new window.CustomViews(...))
1589
2406
  window.CustomViews = CustomViews;
1590
- // @ts-ignore
1591
2407
  window.CustomViewsWidget = CustomViewsWidget;
2408
+ // Run auto-initialization
2409
+ initializeFromScript();
1592
2410
  }
1593
2411
 
1594
- export { AssetsManager, CustomViews, CustomViewsCore, CustomViewsWidget, Config as LocalConfig, PersistenceManager, URLStateManager };
2412
+ export { AssetsManager, CustomViews, CustomViewsCore, CustomViewsWidget, PersistenceManager, URLStateManager };
1595
2413
  //# sourceMappingURL=custom-views.esm.js.map