@tolle_/tolle-ui 0.0.6-beta → 0.0.8-beta
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.
- package/esm2022/lib/theme.service.mjs +77 -7
- package/esm2022/lib/tolle-config.mjs +3 -4
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/tolle_-tolle-ui.mjs +79 -10
- package/fesm2022/tolle_-tolle-ui.mjs.map +1 -1
- package/lib/theme.service.d.ts +8 -5
- package/package.json +1 -1
- package/preset.js +17 -6
- package/public-api.d.ts +1 -0
- package/theme.css +39 -14
|
@@ -10,6 +10,7 @@ export class ThemeService {
|
|
|
10
10
|
renderer;
|
|
11
11
|
isDarkSubject = new BehaviorSubject(false);
|
|
12
12
|
isDark$ = this.isDarkSubject.asObservable();
|
|
13
|
+
styleId = 'tolle-dynamic-theme';
|
|
13
14
|
constructor(document, platformId, config, rendererFactory) {
|
|
14
15
|
this.document = document;
|
|
15
16
|
this.platformId = platformId;
|
|
@@ -33,26 +34,98 @@ export class ThemeService {
|
|
|
33
34
|
else {
|
|
34
35
|
this.disableDarkMode();
|
|
35
36
|
}
|
|
36
|
-
// 2. Apply Brand Config
|
|
37
|
+
// 2. Apply Brand Config - This will generate full palette
|
|
37
38
|
if (this.config) {
|
|
38
39
|
this.applyBrandConfig(this.config);
|
|
39
40
|
}
|
|
40
41
|
}
|
|
41
42
|
/**
|
|
42
|
-
* Applies the brand identity variables
|
|
43
|
-
* These are applied to :root (html) so they ripple through all color-mix functions.
|
|
43
|
+
* Applies the brand identity variables with full shade palette
|
|
44
44
|
*/
|
|
45
45
|
applyBrandConfig(config) {
|
|
46
46
|
if (!isPlatformBrowser(this.platformId))
|
|
47
47
|
return;
|
|
48
48
|
const root = this.document.documentElement;
|
|
49
|
+
// Set primary color if provided
|
|
49
50
|
if (config.primaryColor) {
|
|
50
51
|
this.renderer.setStyle(root, '--primary', config.primaryColor);
|
|
52
|
+
this.generatePrimaryShades(config.primaryColor);
|
|
51
53
|
}
|
|
54
|
+
// Set border radius if provided
|
|
52
55
|
if (config.radius) {
|
|
53
56
|
this.renderer.setStyle(root, '--radius', config.radius);
|
|
54
57
|
}
|
|
55
58
|
}
|
|
59
|
+
/**
|
|
60
|
+
* Generates full primary color palette (50-900) based on base color
|
|
61
|
+
* Uses color-mix() for consistency with your existing approach
|
|
62
|
+
*/
|
|
63
|
+
generatePrimaryShades(baseColor) {
|
|
64
|
+
const css = `
|
|
65
|
+
:root {
|
|
66
|
+
/* Primary Shades */
|
|
67
|
+
--primary-50: color-mix(in srgb, ${baseColor}, white 90%);
|
|
68
|
+
--primary-100: color-mix(in srgb, ${baseColor}, white 80%);
|
|
69
|
+
--primary-200: color-mix(in srgb, ${baseColor}, white 60%);
|
|
70
|
+
--primary-300: color-mix(in srgb, ${baseColor}, white 40%);
|
|
71
|
+
--primary-400: color-mix(in srgb, ${baseColor}, white 20%);
|
|
72
|
+
--primary-500: ${baseColor};
|
|
73
|
+
--primary-600: color-mix(in srgb, ${baseColor}, black 20%);
|
|
74
|
+
--primary-700: color-mix(in srgb, ${baseColor}, black 40%);
|
|
75
|
+
--primary-800: color-mix(in srgb, ${baseColor}, black 60%);
|
|
76
|
+
--primary-900: color-mix(in srgb, ${baseColor}, black 80%);
|
|
77
|
+
|
|
78
|
+
/* Your existing derived colors - updated to use new shades */
|
|
79
|
+
--primary-foreground: white;
|
|
80
|
+
--secondary: color-mix(in srgb, var(--primary-200), transparent 40%);
|
|
81
|
+
--secondary-foreground: var(--primary-900);
|
|
82
|
+
--muted: color-mix(in srgb, var(--primary-50), #f3f4f6 50%);
|
|
83
|
+
--muted-foreground: color-mix(in srgb, var(--primary-400), #4b5563 60%);
|
|
84
|
+
--accent: color-mix(in srgb, var(--primary-100), transparent 70%);
|
|
85
|
+
--accent-foreground: var(--primary-700);
|
|
86
|
+
--ring: color-mix(in srgb, var(--primary-300), transparent 50%);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
.dark {
|
|
90
|
+
/* Dark mode variants */
|
|
91
|
+
--primary-50: color-mix(in srgb, ${baseColor}, black 85%);
|
|
92
|
+
--primary-100: color-mix(in srgb, ${baseColor}, black 75%);
|
|
93
|
+
--primary-200: color-mix(in srgb, ${baseColor}, black 65%);
|
|
94
|
+
--primary-300: color-mix(in srgb, ${baseColor}, black 55%);
|
|
95
|
+
--primary-400: color-mix(in srgb, ${baseColor}, black 45%);
|
|
96
|
+
--primary-500: ${baseColor};
|
|
97
|
+
--primary-600: color-mix(in srgb, ${baseColor}, white 20%);
|
|
98
|
+
--primary-700: color-mix(in srgb, ${baseColor}, white 35%);
|
|
99
|
+
--primary-800: color-mix(in srgb, ${baseColor}, white 50%);
|
|
100
|
+
--primary-900: color-mix(in srgb, ${baseColor}, white 65%);
|
|
101
|
+
|
|
102
|
+
/* Dark mode derived colors */
|
|
103
|
+
--primary-foreground: color-mix(in srgb, ${baseColor}, white 90%);
|
|
104
|
+
--secondary: color-mix(in srgb, var(--primary-900), transparent 70%);
|
|
105
|
+
--secondary-foreground: var(--primary-100);
|
|
106
|
+
--muted: color-mix(in srgb, var(--primary-950), #1f2937 50%);
|
|
107
|
+
--muted-foreground: color-mix(in srgb, var(--primary-300), #9ca3af 40%);
|
|
108
|
+
--accent: color-mix(in srgb, var(--primary-800), transparent 80%);
|
|
109
|
+
--accent-foreground: var(--primary-200);
|
|
110
|
+
--ring: color-mix(in srgb, var(--primary-600), transparent 60%);
|
|
111
|
+
}
|
|
112
|
+
`;
|
|
113
|
+
this.injectDynamicStyles(css);
|
|
114
|
+
}
|
|
115
|
+
injectDynamicStyles(css) {
|
|
116
|
+
if (!isPlatformBrowser(this.platformId))
|
|
117
|
+
return;
|
|
118
|
+
// Remove existing dynamic styles
|
|
119
|
+
const existingStyle = this.document.getElementById(this.styleId);
|
|
120
|
+
if (existingStyle) {
|
|
121
|
+
existingStyle.remove();
|
|
122
|
+
}
|
|
123
|
+
// Create and inject new styles
|
|
124
|
+
const styleElement = this.document.createElement('style');
|
|
125
|
+
styleElement.id = this.styleId;
|
|
126
|
+
styleElement.textContent = css;
|
|
127
|
+
this.renderer.appendChild(this.document.head, styleElement);
|
|
128
|
+
}
|
|
56
129
|
toggleTheme() {
|
|
57
130
|
const isCurrentlyDark = this.document.documentElement.classList.contains('dark');
|
|
58
131
|
isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();
|
|
@@ -67,9 +140,6 @@ export class ThemeService {
|
|
|
67
140
|
localStorage.setItem('tolle-theme', 'light');
|
|
68
141
|
this.isDarkSubject.next(false);
|
|
69
142
|
}
|
|
70
|
-
/**
|
|
71
|
-
* Helper to get current state for components that might need JS-based logic
|
|
72
|
-
*/
|
|
73
143
|
get currentTheme() {
|
|
74
144
|
return this.isDarkSubject.value ? 'dark' : 'light';
|
|
75
145
|
}
|
|
@@ -91,4 +161,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
91
161
|
type: Inject,
|
|
92
162
|
args: [TOLLE_CONFIG]
|
|
93
163
|
}] }, { type: i0.RendererFactory2 }] });
|
|
94
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGhlbWUuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3RvbGxlL3NyYy9saWIvdGhlbWUuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSxXQUFXLEVBQUUsUUFBUSxFQUErQixNQUFNLGVBQWUsQ0FBQztBQUN2RyxPQUFPLEVBQUUsUUFBUSxFQUFFLGlCQUFpQixFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDOUQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUN2QyxPQUFPLEVBQUUsWUFBWSxFQUFlLE1BQU0sZ0JBQWdCLENBQUM7O0FBRzNELE1BQU0sT0FBTyxZQUFZO0lBTUs7SUFDRztJQUNhO0lBUHBDLFFBQVEsQ0FBWTtJQUNwQixhQUFhLEdBQUcsSUFBSSxlQUFlLENBQVUsS0FBSyxDQUFDLENBQUM7SUFDNUQsT0FBTyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsWUFBWSxFQUFFLENBQUM7SUFFNUMsWUFDNEIsUUFBa0IsRUFDZixVQUFrQixFQUNMLE1BQW1CLEVBQzdELGVBQWlDO1FBSFAsYUFBUSxHQUFSLFFBQVEsQ0FBVTtRQUNmLGVBQVUsR0FBVixVQUFVLENBQVE7UUFDTCxXQUFNLEdBQU4sTUFBTSxDQUFhO1FBRzdELElBQUksQ0FBQyxRQUFRLEdBQUcsZUFBZSxDQUFDLGNBQWMsQ0FBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFDM0QsSUFBSSxDQUFDLGVBQWUsRUFBRSxDQUFDO0lBQ3pCLENBQUM7SUFFTyxlQUFlO1FBQ3JCLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDO1lBQUUsT0FBTztRQUVoRCx5Q0FBeUM7UUFDekMsTUFBTSxVQUFVLEdBQUcsWUFBWSxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQztRQUN2RCxNQUFNLGlCQUFpQixH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IEluamVjdCwgUExBVEZPUk1fSUQsIE9wdGlvbmFsLCBSZW5kZXJlcjIsIFJlbmRlcmVyRmFjdG9yeTIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IERPQ1VNRU5ULCBpc1BsYXRmb3JtQnJvd3NlciB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IFRPTExFX0NPTkZJRywgVG9sbGVDb25maWcgfSBmcm9tICcuL3RvbGxlLWNvbmZpZyc7XG5cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgVGhlbWVTZXJ2aWNlIHtcbiAgcHJpdmF0ZSByZW5kZXJlcjogUmVuZGVyZXIyO1xuICBwcml2YXRlIGlzRGFya1N1YmplY3QgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PGJvb2xlYW4+KGZhbHNlKTtcbiAgaXNEYXJrJCA9IHRoaXMuaXNEYXJrU3ViamVjdC5hc09ic2VydmFibGUoKTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBASW5qZWN0KERPQ1VNRU5UKSBwcml2YXRlIGRvY3VtZW50OiBEb2N1bWVudCxcbiAgICBASW5qZWN0KFBMQVRGT1JNX0lEKSBwcml2YXRlIHBsYXRmb3JtSWQ6IE9iamVjdCxcbiAgICBAT3B0aW9uYWwoKSBASW5qZWN0KFRPTExFX0NPTkZJRykgcHJpdmF0ZSBjb25maWc6IFRvbGxlQ29uZmlnLFxuICAgIHJlbmRlcmVyRmFjdG9yeTogUmVuZGVyZXJGYWN0b3J5MlxuICApIHtcbiAgICB0aGlzLnJlbmRlcmVyID0gcmVuZGVyZXJGYWN0b3J5LmNyZWF0ZVJlbmRlcmVyKG51bGwsIG51bGwpO1xuICAgIHRoaXMuaW5pdGlhbGl6ZVRoZW1lKCk7XG4gIH1cblxuICBwcml2YXRlIGluaXRpYWxpemVUaGVtZSgpIHtcbiAgICBpZiAoIWlzUGxhdGZvcm1Ccm93c2VyKHRoaXMucGxhdGZvcm1JZCkpIHJldHVybjtcblxuICAgIC8vIDEuIERldGVybWluZSBJbml0aWFsIE1vZGUgKERhcmsvTGlnaHQpXG4gICAgY29uc3Qgc2F2ZWRUaGVtZSA9IGxvY2FsU3RvcmFnZS5nZXRJdGVtKCd0b2xsZS10aGVtZScpO1xuICAgIGNvbnN0IHN5c3RlbVByZWZlcnNEYXJrID0gd2luZG93Lm1hdGNoTWVkaWEoJyhwcmVmZXJzLWNvbG9yLXNjaGVtZTogZGFyayknKS5tYXRjaGVzO1xuXG4gICAgLy8gTG9naWM6IFNhdmVkIFByZWZlcmVuY2UgPiBDb25maWcgRGVmYXVsdCA+IFN5c3RlbSBQcmVmZXJlbmNlXG4gICAgY29uc3Qgc2hvdWxkQmVEYXJrID0gc2F2ZWRUaGVtZVxuICAgICAgPyBzYXZlZFRoZW1lID09PSAnZGFyaydcbiAgICAgIDogKHRoaXMuY29uZmlnPy5kYXJrQnlEZWZhdWx0ID8/IHN5c3RlbVByZWZlcnNEYXJrKTtcblxuICAgIGlmIChzaG91bGRCZURhcmspIHtcbiAgICAgIHRoaXMuZW5hYmxlRGFya01vZGUoKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5kaXNhYmxlRGFya01vZGUoKTtcbiAgICB9XG5cbiAgICAvLyAyLiBBcHBseSBCcmFuZCBDb25maWdcbiAgICBpZiAodGhpcy5jb25maWcpIHtcbiAgICAgIHRoaXMuYXBwbHlCcmFuZENvbmZpZyh0aGlzLmNvbmZpZyk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIEFwcGxpZXMgdGhlIGJyYW5kIGlkZW50aXR5IHZhcmlhYmxlcy5cbiAgICogVGhlc2UgYXJlIGFwcGxpZWQgdG8gOnJvb3QgKGh0bWwpIHNvIHRoZXkgcmlwcGxlIHRocm91Z2ggYWxsIGNvbG9yLW1peCBmdW5jdGlvbnMuXG4gICAqL1xuICBhcHBseUJyYW5kQ29uZmlnKGNvbmZpZzogVG9sbGVDb25maWcpIHtcbiAgICBpZiAoIWlzUGxhdGZvcm1Ccm93c2VyKHRoaXMucGxhdGZvcm1JZCkpIHJldHVybjtcblxuICAgIGNvbnN0IHJvb3QgPSB0aGlzLmRvY3VtZW50LmRvY3VtZW50RWxlbWVudDtcblxuICAgIGlmIChjb25maWcucHJpbWFyeUNvbG9yKSB7XG4gICAgICB0aGlzLnJlbmRlcmVyLnNldFN0eWxlKHJvb3QsICctLXByaW1hcnknLCBjb25maWcucHJpbWFyeUNvbG9yKTtcbiAgICB9XG5cbiAgICBpZiAoY29uZmlnLnJhZGl1cykge1xuICAgICAgdGhpcy5yZW5kZXJlci5zZXRTdHlsZShyb290LCAnLS1yYWRpdXMnLCBjb25maWcucmFkaXVzKTtcbiAgICB9XG4gIH1cblxuICB0b2dnbGVUaGVtZSgpIHtcbiAgICBjb25zdCBpc0N1cnJlbnRseURhcmsgPSB0aGlzLmRvY3VtZW50LmRvY3VtZW50RWxlbWVudC5jbGFzc0xpc3QuY29udGFpbnMoJ2RhcmsnKTtcbiAgICBpc0N1cnJlbnRseURhcmsgPyB0aGlzLmRpc2FibGVEYXJrTW9kZSgpIDogdGhpcy5lbmFibGVEYXJrTW9kZSgpO1xuICB9XG5cbiAgcHJpdmF0ZSBlbmFibGVEYXJrTW9kZSgpIHtcbiAgICB0aGlzLnJlbmRlcmVyLmFkZENsYXNzKHRoaXMuZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50LCAnZGFyaycpO1xuICAgIGxvY2FsU3RvcmFnZS5zZXRJdGVtKCd0b2xsZS10aGVtZScsICdkYXJrJyk7XG4gICAgdGhpcy5pc0RhcmtTdWJqZWN0Lm5leHQodHJ1ZSk7XG4gIH1cblxuICBwcml2YXRlIGRpc2FibGVEYXJrTW9kZSgpIHtcbiAgICB0aGlzLnJlbmRlcmVyLnJlbW92ZUNsYXNzKHRoaXMuZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50LCAnZGFyaycpO1xuICAgIGxvY2FsU3RvcmFnZS5zZXRJdGVtKCd0b2xsZS10aGVtZScsICdsaWdodCcpO1xuICAgIHRoaXMuaXNEYXJrU3ViamVjdC5uZXh0KGZhbHNlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBIZWxwZXIgdG8gZ2V0IGN1cnJlbnQgc3RhdGUgZm9yIGNvbXBvbmVudHMgdGhhdCBtaWdodCBuZWVkIEpTLWJhc2VkIGxvZ2ljXG4gICAqL1xuICBnZXQgY3VycmVudFRoZW1lKCk6ICdkYXJrJyB8ICdsaWdodCcge1xuICAgIHJldHVybiB0aGlzLmlzRGFya1N1YmplY3QudmFsdWUgPyAnZGFyaycgOiAnbGlnaHQnO1xuICB9XG59XG4iXX0=
|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"theme.service.js","sourceRoot":"","sources":["../../../../projects/tolle/src/lib/theme.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, Inject, PLATFORM_ID, Optional, Renderer2, RendererFactory2 } from '@angular/core';\nimport { DOCUMENT, isPlatformBrowser } from '@angular/common';\nimport { BehaviorSubject } from 'rxjs';\nimport { TOLLE_CONFIG, TolleConfig } from './tolle-config';\n\n@Injectable({ providedIn: 'root' })\nexport class ThemeService {\n  private renderer: Renderer2;\n  private isDarkSubject = new BehaviorSubject<boolean>(false);\n  isDark$ = this.isDarkSubject.asObservable();\n  private styleId = 'tolle-dynamic-theme';\n\n  constructor(\n    @Inject(DOCUMENT) private document: Document,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    @Optional() @Inject(TOLLE_CONFIG) private config: TolleConfig,\n    rendererFactory: RendererFactory2\n  ) {\n    this.renderer = rendererFactory.createRenderer(null, null);\n    this.initializeTheme();\n  }\n\n  private initializeTheme() {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    // 1. Determine Initial Mode (Dark/Light)\n    const savedTheme = localStorage.getItem('tolle-theme');\n    const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;\n\n    // Logic: Saved Preference > Config Default > System Preference\n    const shouldBeDark = savedTheme\n      ? savedTheme === 'dark'\n      : (this.config?.darkByDefault ?? systemPrefersDark);\n\n    if (shouldBeDark) {\n      this.enableDarkMode();\n    } else {\n      this.disableDarkMode();\n    }\n\n    // 2. Apply Brand Config - This will generate full palette\n    if (this.config) {\n      this.applyBrandConfig(this.config);\n    }\n  }\n\n  /**\n   * Applies the brand identity variables with full shade palette\n   */\n  applyBrandConfig(config: TolleConfig) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    const root = this.document.documentElement;\n\n    // Set primary color if provided\n    if (config.primaryColor) {\n      this.renderer.setStyle(root, '--primary', config.primaryColor);\n      this.generatePrimaryShades(config.primaryColor);\n    }\n\n    // Set border radius if provided\n    if (config.radius) {\n      this.renderer.setStyle(root, '--radius', config.radius);\n    }\n  }\n\n  /**\n   * Generates full primary color palette (50-900) based on base color\n   * Uses color-mix() for consistency with your existing approach\n   */\n  private generatePrimaryShades(baseColor: string) {\n    const css = `\n      :root {\n        /* Primary Shades */\n        --primary-50: color-mix(in srgb, ${baseColor}, white 90%);\n        --primary-100: color-mix(in srgb, ${baseColor}, white 80%);\n        --primary-200: color-mix(in srgb, ${baseColor}, white 60%);\n        --primary-300: color-mix(in srgb, ${baseColor}, white 40%);\n        --primary-400: color-mix(in srgb, ${baseColor}, white 20%);\n        --primary-500: ${baseColor};\n        --primary-600: color-mix(in srgb, ${baseColor}, black 20%);\n        --primary-700: color-mix(in srgb, ${baseColor}, black 40%);\n        --primary-800: color-mix(in srgb, ${baseColor}, black 60%);\n        --primary-900: color-mix(in srgb, ${baseColor}, black 80%);\n\n        /* Your existing derived colors - updated to use new shades */\n        --primary-foreground: white;\n        --secondary: color-mix(in srgb, var(--primary-200), transparent 40%);\n        --secondary-foreground: var(--primary-900);\n        --muted: color-mix(in srgb, var(--primary-50), #f3f4f6 50%);\n        --muted-foreground: color-mix(in srgb, var(--primary-400), #4b5563 60%);\n        --accent: color-mix(in srgb, var(--primary-100), transparent 70%);\n        --accent-foreground: var(--primary-700);\n        --ring: color-mix(in srgb, var(--primary-300), transparent 50%);\n      }\n\n      .dark {\n        /* Dark mode variants */\n        --primary-50: color-mix(in srgb, ${baseColor}, black 85%);\n        --primary-100: color-mix(in srgb, ${baseColor}, black 75%);\n        --primary-200: color-mix(in srgb, ${baseColor}, black 65%);\n        --primary-300: color-mix(in srgb, ${baseColor}, black 55%);\n        --primary-400: color-mix(in srgb, ${baseColor}, black 45%);\n        --primary-500: ${baseColor};\n        --primary-600: color-mix(in srgb, ${baseColor}, white 20%);\n        --primary-700: color-mix(in srgb, ${baseColor}, white 35%);\n        --primary-800: color-mix(in srgb, ${baseColor}, white 50%);\n        --primary-900: color-mix(in srgb, ${baseColor}, white 65%);\n\n        /* Dark mode derived colors */\n        --primary-foreground: color-mix(in srgb, ${baseColor}, white 90%);\n        --secondary: color-mix(in srgb, var(--primary-900), transparent 70%);\n        --secondary-foreground: var(--primary-100);\n        --muted: color-mix(in srgb, var(--primary-950), #1f2937 50%);\n        --muted-foreground: color-mix(in srgb, var(--primary-300), #9ca3af 40%);\n        --accent: color-mix(in srgb, var(--primary-800), transparent 80%);\n        --accent-foreground: var(--primary-200);\n        --ring: color-mix(in srgb, var(--primary-600), transparent 60%);\n      }\n    `;\n\n    this.injectDynamicStyles(css);\n  }\n\n  private injectDynamicStyles(css: string) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    // Remove existing dynamic styles\n    const existingStyle = this.document.getElementById(this.styleId);\n    if (existingStyle) {\n      existingStyle.remove();\n    }\n\n    // Create and inject new styles\n    const styleElement = this.document.createElement('style');\n    styleElement.id = this.styleId;\n    styleElement.textContent = css;\n    this.renderer.appendChild(this.document.head, styleElement);\n  }\n\n  toggleTheme() {\n    const isCurrentlyDark = this.document.documentElement.classList.contains('dark');\n    isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();\n  }\n\n  private enableDarkMode() {\n    this.renderer.addClass(this.document.documentElement, 'dark');\n    localStorage.setItem('tolle-theme', 'dark');\n    this.isDarkSubject.next(true);\n  }\n\n  private disableDarkMode() {\n    this.renderer.removeClass(this.document.documentElement, 'dark');\n    localStorage.setItem('tolle-theme', 'light');\n    this.isDarkSubject.next(false);\n  }\n\n  get currentTheme(): 'dark' | 'light' {\n    return this.isDarkSubject.value ? 'dark' : 'light';\n  }\n}\n"]}
|
|
@@ -1,19 +1,18 @@
|
|
|
1
1
|
import { APP_INITIALIZER, InjectionToken } from '@angular/core';
|
|
2
2
|
import { ThemeService } from './theme.service';
|
|
3
3
|
export const TOLLE_CONFIG = new InjectionToken('TolleConfig');
|
|
4
|
-
// projects/tolle/src/lib/tolle-config.ts
|
|
5
4
|
export function provideTolleConfig(config) {
|
|
6
5
|
return [
|
|
7
6
|
{ provide: TOLLE_CONFIG, useValue: config },
|
|
8
7
|
{
|
|
9
8
|
provide: APP_INITIALIZER,
|
|
10
9
|
useFactory: (themeService) => () => {
|
|
11
|
-
//
|
|
12
|
-
//
|
|
10
|
+
// Theme is already initialized in constructor
|
|
11
|
+
// This factory ensures themeService is instantiated during bootstrap
|
|
13
12
|
},
|
|
14
13
|
deps: [ThemeService],
|
|
15
14
|
multi: true,
|
|
16
15
|
},
|
|
17
16
|
];
|
|
18
17
|
}
|
|
19
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9sbGUtY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvdG9sbGUvc3JjL2xpYi90b2xsZS1jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGVBQWUsRUFBRSxjQUFjLEVBQVksTUFBTSxlQUFlLENBQUM7QUFDMUUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBUy9DLE1BQU0sQ0FBQyxNQUFNLFlBQVksR0FBRyxJQUFJLGNBQWMsQ0FBYyxhQUFhLENBQUMsQ0FBQztBQUUzRSxNQUFNLFVBQVUsa0JBQWtCLENBQUMsTUFBbUI7SUFDcEQsT0FBTztRQUNMLEVBQUUsT0FBTyxFQUFFLFlBQVksRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFO1FBQzNDO1lBQ0UsT0FBTyxFQUFFLGVBQWU7WUFDeEIsVUFBVSxFQUFFLENBQUMsWUFBMEIsRUFBRSxFQUFFLENBQUMsR0FBRyxFQUFFO2dCQUMvQyw4Q0FBOEM7Z0JBQzlDLHFFQUFxRTtZQUN2RSxDQUFDO1lBQ0QsSUFBSSxFQUFFLENBQUMsWUFBWSxDQUFDO1lBQ3BCLEtBQUssRUFBRSxJQUFJO1NBQ1o7S0FDRixDQUFDO0FBQ0osQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFQUF9JTklUSUFMSVpFUiwgSW5qZWN0aW9uVG9rZW4sIFByb3ZpZGVyIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBUaGVtZVNlcnZpY2UgfSBmcm9tICcuL3RoZW1lLnNlcnZpY2UnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFRvbGxlQ29uZmlnIHtcbiAgcHJpbWFyeUNvbG9yPzogc3RyaW5nO1xuICByYWRpdXM/OiBzdHJpbmc7XG4gIGRlZmF1bHRUb2FzdER1cmF0aW9uPzogbnVtYmVyO1xuICBkYXJrQnlEZWZhdWx0PzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGNvbnN0IFRPTExFX0NPTkZJRyA9IG5ldyBJbmplY3Rpb25Ub2tlbjxUb2xsZUNvbmZpZz4oJ1RvbGxlQ29uZmlnJyk7XG5cbmV4cG9ydCBmdW5jdGlvbiBwcm92aWRlVG9sbGVDb25maWcoY29uZmlnOiBUb2xsZUNvbmZpZyk6IFByb3ZpZGVyW10ge1xuICByZXR1cm4gW1xuICAgIHsgcHJvdmlkZTogVE9MTEVfQ09ORklHLCB1c2VWYWx1ZTogY29uZmlnIH0sXG4gICAge1xuICAgICAgcHJvdmlkZTogQVBQX0lOSVRJQUxJWkVSLFxuICAgICAgdXNlRmFjdG9yeTogKHRoZW1lU2VydmljZTogVGhlbWVTZXJ2aWNlKSA9PiAoKSA9PiB7XG4gICAgICAgIC8vIFRoZW1lIGlzIGFscmVhZHkgaW5pdGlhbGl6ZWQgaW4gY29uc3RydWN0b3JcbiAgICAgICAgLy8gVGhpcyBmYWN0b3J5IGVuc3VyZXMgdGhlbWVTZXJ2aWNlIGlzIGluc3RhbnRpYXRlZCBkdXJpbmcgYm9vdHN0cmFwXG4gICAgICB9LFxuICAgICAgZGVwczogW1RoZW1lU2VydmljZV0sXG4gICAgICBtdWx0aTogdHJ1ZSxcbiAgICB9LFxuICBdO1xufVxuIl19
|
package/esm2022/public-api.mjs
CHANGED
|
@@ -33,4 +33,5 @@ export * from './lib/modal';
|
|
|
33
33
|
export * from './lib/button-group.component';
|
|
34
34
|
export * from './lib/date-range-picker.component';
|
|
35
35
|
export * from './lib/range-calendar.component';
|
|
36
|
-
|
|
36
|
+
export * from './lib/theme.service';
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL3RvbGxlL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLHdCQUF3QixDQUFDO0FBQ3ZDLGNBQWMsdUJBQXVCLENBQUM7QUFDdEMsY0FBYyxzQkFBc0IsQ0FBQztBQUNyQyxjQUFjLHdCQUF3QixDQUFDO0FBQ3ZDLGNBQWMsNkJBQTZCLENBQUM7QUFDNUMsY0FBYyw4QkFBOEIsQ0FBQztBQUM3QyxjQUFjLGtDQUFrQyxDQUFDO0FBQ2pELGNBQWMsd0JBQXdCLENBQUM7QUFDdkMsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsMEJBQTBCLENBQUM7QUFDekMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLHFCQUFxQixDQUFBO0FBQ25DLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYyw4QkFBOEIsQ0FBQTtBQUM1QyxjQUFjLDBCQUEwQixDQUFBO0FBQ3hDLGNBQWMsOEJBQThCLENBQUE7QUFDNUMsY0FBYyw2QkFBNkIsQ0FBQTtBQUMzQyxjQUFjLDRCQUE0QixDQUFBO0FBQzFDLGNBQWMsNEJBQTRCLENBQUE7QUFDMUMsY0FBYyw0QkFBNEIsQ0FBQTtBQUMxQyxjQUFjLGdDQUFnQyxDQUFBO0FBQzlDLGNBQWMsMkJBQTJCLENBQUE7QUFDekMsY0FBYyx1QkFBdUIsQ0FBQTtBQUNyQyxjQUFjLGlCQUFpQixDQUFBO0FBQy9CLGNBQWMscUJBQXFCLENBQUE7QUFDbkMsY0FBYywyQkFBMkIsQ0FBQTtBQUN6QyxjQUFjLGFBQWEsQ0FBQTtBQUMzQixjQUFjLDhCQUE4QixDQUFBO0FBQzVDLGNBQWMsbUNBQW1DLENBQUE7QUFDakQsY0FBYyxnQ0FBZ0MsQ0FBQTtBQUM5QyxjQUFjLHFCQUFxQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIFB1YmxpYyBBUEkgU3VyZmFjZSBvZiB0b2xsZVxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vbGliL3V0aWxzL2NuJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2J1dHRvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvaW5wdXQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NhcmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3NlbGVjdC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvc2VsZWN0LWl0ZW0uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3NlbGVjdC1ncm91cC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvc2VsZWN0LXNlcGFyYXRvci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvc3dpdGNoLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9iYWRnZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvc2tlbGV0b24uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2NoZWNrYm94LmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi90b29sdGlwLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi90b2FzdC5zZXJ2aWNlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvdG9sbGUtY29uZmlnJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL211bHRpLXNlbGVjdC5jb21wb25lbnQnXG5leHBvcnQgKiBmcm9tICcuL2xpYi9jYWxlbmRhci5jb21wb25lbnQnXG5leHBvcnQgKiBmcm9tICcuL2xpYi9tYXNrZWQtaW5wdXQuY29tcG9uZW50J1xuZXhwb3J0ICogZnJvbSAnLi9saWIvZGF0ZS1waWNrZXIuY29tcG9uZW50J1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcGFnaW5hdGlvbi5jb21wb25lbnQnXG5leHBvcnQgKiBmcm9tICcuL2xpYi9kYXRhLXRhYmxlLmNvbXBvbmVudCdcbmV4cG9ydCAqIGZyb20gJy4vbGliL3RvbGxlLWNlbGwuZGlyZWN0aXZlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYWNjb3JkaW9uLWl0ZW0uY29tcG9uZW50J1xuZXhwb3J0ICogZnJvbSAnLi9saWIvYWNjb3JkaW9uLmNvbXBvbmVudCdcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGFsLmNvbXBvbmVudCdcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGFsLXJlZidcbmV4cG9ydCAqIGZyb20gJy4vbGliL21vZGFsLnNlcnZpY2UnXG5leHBvcnQgKiBmcm9tICcuL2xpYi9tb2RhbC1zdGFjay5zZXJ2aWNlJ1xuZXhwb3J0ICogZnJvbSAnLi9saWIvbW9kYWwnXG5leHBvcnQgKiBmcm9tICcuL2xpYi9idXR0b24tZ3JvdXAuY29tcG9uZW50J1xuZXhwb3J0ICogZnJvbSAnLi9saWIvZGF0ZS1yYW5nZS1waWNrZXIuY29tcG9uZW50J1xuZXhwb3J0ICogZnJvbSAnLi9saWIvcmFuZ2UtY2FsZW5kYXIuY29tcG9uZW50J1xuZXhwb3J0ICogZnJvbSAnLi9saWIvdGhlbWUuc2VydmljZSdcbiJdfQ==
|
|
@@ -1144,6 +1144,7 @@ class ThemeService {
|
|
|
1144
1144
|
renderer;
|
|
1145
1145
|
isDarkSubject = new BehaviorSubject(false);
|
|
1146
1146
|
isDark$ = this.isDarkSubject.asObservable();
|
|
1147
|
+
styleId = 'tolle-dynamic-theme';
|
|
1147
1148
|
constructor(document, platformId, config, rendererFactory) {
|
|
1148
1149
|
this.document = document;
|
|
1149
1150
|
this.platformId = platformId;
|
|
@@ -1167,26 +1168,98 @@ class ThemeService {
|
|
|
1167
1168
|
else {
|
|
1168
1169
|
this.disableDarkMode();
|
|
1169
1170
|
}
|
|
1170
|
-
// 2. Apply Brand Config
|
|
1171
|
+
// 2. Apply Brand Config - This will generate full palette
|
|
1171
1172
|
if (this.config) {
|
|
1172
1173
|
this.applyBrandConfig(this.config);
|
|
1173
1174
|
}
|
|
1174
1175
|
}
|
|
1175
1176
|
/**
|
|
1176
|
-
* Applies the brand identity variables
|
|
1177
|
-
* These are applied to :root (html) so they ripple through all color-mix functions.
|
|
1177
|
+
* Applies the brand identity variables with full shade palette
|
|
1178
1178
|
*/
|
|
1179
1179
|
applyBrandConfig(config) {
|
|
1180
1180
|
if (!isPlatformBrowser(this.platformId))
|
|
1181
1181
|
return;
|
|
1182
1182
|
const root = this.document.documentElement;
|
|
1183
|
+
// Set primary color if provided
|
|
1183
1184
|
if (config.primaryColor) {
|
|
1184
1185
|
this.renderer.setStyle(root, '--primary', config.primaryColor);
|
|
1186
|
+
this.generatePrimaryShades(config.primaryColor);
|
|
1185
1187
|
}
|
|
1188
|
+
// Set border radius if provided
|
|
1186
1189
|
if (config.radius) {
|
|
1187
1190
|
this.renderer.setStyle(root, '--radius', config.radius);
|
|
1188
1191
|
}
|
|
1189
1192
|
}
|
|
1193
|
+
/**
|
|
1194
|
+
* Generates full primary color palette (50-900) based on base color
|
|
1195
|
+
* Uses color-mix() for consistency with your existing approach
|
|
1196
|
+
*/
|
|
1197
|
+
generatePrimaryShades(baseColor) {
|
|
1198
|
+
const css = `
|
|
1199
|
+
:root {
|
|
1200
|
+
/* Primary Shades */
|
|
1201
|
+
--primary-50: color-mix(in srgb, ${baseColor}, white 90%);
|
|
1202
|
+
--primary-100: color-mix(in srgb, ${baseColor}, white 80%);
|
|
1203
|
+
--primary-200: color-mix(in srgb, ${baseColor}, white 60%);
|
|
1204
|
+
--primary-300: color-mix(in srgb, ${baseColor}, white 40%);
|
|
1205
|
+
--primary-400: color-mix(in srgb, ${baseColor}, white 20%);
|
|
1206
|
+
--primary-500: ${baseColor};
|
|
1207
|
+
--primary-600: color-mix(in srgb, ${baseColor}, black 20%);
|
|
1208
|
+
--primary-700: color-mix(in srgb, ${baseColor}, black 40%);
|
|
1209
|
+
--primary-800: color-mix(in srgb, ${baseColor}, black 60%);
|
|
1210
|
+
--primary-900: color-mix(in srgb, ${baseColor}, black 80%);
|
|
1211
|
+
|
|
1212
|
+
/* Your existing derived colors - updated to use new shades */
|
|
1213
|
+
--primary-foreground: white;
|
|
1214
|
+
--secondary: color-mix(in srgb, var(--primary-200), transparent 40%);
|
|
1215
|
+
--secondary-foreground: var(--primary-900);
|
|
1216
|
+
--muted: color-mix(in srgb, var(--primary-50), #f3f4f6 50%);
|
|
1217
|
+
--muted-foreground: color-mix(in srgb, var(--primary-400), #4b5563 60%);
|
|
1218
|
+
--accent: color-mix(in srgb, var(--primary-100), transparent 70%);
|
|
1219
|
+
--accent-foreground: var(--primary-700);
|
|
1220
|
+
--ring: color-mix(in srgb, var(--primary-300), transparent 50%);
|
|
1221
|
+
}
|
|
1222
|
+
|
|
1223
|
+
.dark {
|
|
1224
|
+
/* Dark mode variants */
|
|
1225
|
+
--primary-50: color-mix(in srgb, ${baseColor}, black 85%);
|
|
1226
|
+
--primary-100: color-mix(in srgb, ${baseColor}, black 75%);
|
|
1227
|
+
--primary-200: color-mix(in srgb, ${baseColor}, black 65%);
|
|
1228
|
+
--primary-300: color-mix(in srgb, ${baseColor}, black 55%);
|
|
1229
|
+
--primary-400: color-mix(in srgb, ${baseColor}, black 45%);
|
|
1230
|
+
--primary-500: ${baseColor};
|
|
1231
|
+
--primary-600: color-mix(in srgb, ${baseColor}, white 20%);
|
|
1232
|
+
--primary-700: color-mix(in srgb, ${baseColor}, white 35%);
|
|
1233
|
+
--primary-800: color-mix(in srgb, ${baseColor}, white 50%);
|
|
1234
|
+
--primary-900: color-mix(in srgb, ${baseColor}, white 65%);
|
|
1235
|
+
|
|
1236
|
+
/* Dark mode derived colors */
|
|
1237
|
+
--primary-foreground: color-mix(in srgb, ${baseColor}, white 90%);
|
|
1238
|
+
--secondary: color-mix(in srgb, var(--primary-900), transparent 70%);
|
|
1239
|
+
--secondary-foreground: var(--primary-100);
|
|
1240
|
+
--muted: color-mix(in srgb, var(--primary-950), #1f2937 50%);
|
|
1241
|
+
--muted-foreground: color-mix(in srgb, var(--primary-300), #9ca3af 40%);
|
|
1242
|
+
--accent: color-mix(in srgb, var(--primary-800), transparent 80%);
|
|
1243
|
+
--accent-foreground: var(--primary-200);
|
|
1244
|
+
--ring: color-mix(in srgb, var(--primary-600), transparent 60%);
|
|
1245
|
+
}
|
|
1246
|
+
`;
|
|
1247
|
+
this.injectDynamicStyles(css);
|
|
1248
|
+
}
|
|
1249
|
+
injectDynamicStyles(css) {
|
|
1250
|
+
if (!isPlatformBrowser(this.platformId))
|
|
1251
|
+
return;
|
|
1252
|
+
// Remove existing dynamic styles
|
|
1253
|
+
const existingStyle = this.document.getElementById(this.styleId);
|
|
1254
|
+
if (existingStyle) {
|
|
1255
|
+
existingStyle.remove();
|
|
1256
|
+
}
|
|
1257
|
+
// Create and inject new styles
|
|
1258
|
+
const styleElement = this.document.createElement('style');
|
|
1259
|
+
styleElement.id = this.styleId;
|
|
1260
|
+
styleElement.textContent = css;
|
|
1261
|
+
this.renderer.appendChild(this.document.head, styleElement);
|
|
1262
|
+
}
|
|
1190
1263
|
toggleTheme() {
|
|
1191
1264
|
const isCurrentlyDark = this.document.documentElement.classList.contains('dark');
|
|
1192
1265
|
isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();
|
|
@@ -1201,9 +1274,6 @@ class ThemeService {
|
|
|
1201
1274
|
localStorage.setItem('tolle-theme', 'light');
|
|
1202
1275
|
this.isDarkSubject.next(false);
|
|
1203
1276
|
}
|
|
1204
|
-
/**
|
|
1205
|
-
* Helper to get current state for components that might need JS-based logic
|
|
1206
|
-
*/
|
|
1207
1277
|
get currentTheme() {
|
|
1208
1278
|
return this.isDarkSubject.value ? 'dark' : 'light';
|
|
1209
1279
|
}
|
|
@@ -1227,15 +1297,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
1227
1297
|
}] }, { type: i0.RendererFactory2 }] });
|
|
1228
1298
|
|
|
1229
1299
|
const TOLLE_CONFIG = new InjectionToken('TolleConfig');
|
|
1230
|
-
// projects/tolle/src/lib/tolle-config.ts
|
|
1231
1300
|
function provideTolleConfig(config) {
|
|
1232
1301
|
return [
|
|
1233
1302
|
{ provide: TOLLE_CONFIG, useValue: config },
|
|
1234
1303
|
{
|
|
1235
1304
|
provide: APP_INITIALIZER,
|
|
1236
1305
|
useFactory: (themeService) => () => {
|
|
1237
|
-
//
|
|
1238
|
-
//
|
|
1306
|
+
// Theme is already initialized in constructor
|
|
1307
|
+
// This factory ensures themeService is instantiated during bootstrap
|
|
1239
1308
|
},
|
|
1240
1309
|
deps: [ThemeService],
|
|
1241
1310
|
multi: true,
|
|
@@ -3639,5 +3708,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
3639
3708
|
* Generated bundle index. Do not edit.
|
|
3640
3709
|
*/
|
|
3641
3710
|
|
|
3642
|
-
export { AccordionComponent, AccordionItemComponent, BadgeComponent, ButtonComponent, ButtonGroupComponent, CalendarComponent, CardComponent, CardContentComponent, CardFooterComponent, CardHeaderComponent, CardTitleComponent, CheckboxComponent, DataTableComponent, DatePickerComponent, DateRangePickerComponent, InputComponent, MaskedInputComponent, Modal, ModalComponent, ModalRef, ModalService, ModalStackService, MultiSelectComponent, PaginationComponent, RangeCalendarComponent, SelectComponent, SelectGroupComponent, SelectItemComponent, SelectSeparatorComponent, SkeletonComponent, SwitchComponent, TOLLE_CONFIG, ToastService, TolleCellDirective, TooltipDirective, cn, provideTolleConfig };
|
|
3711
|
+
export { AccordionComponent, AccordionItemComponent, BadgeComponent, ButtonComponent, ButtonGroupComponent, CalendarComponent, CardComponent, CardContentComponent, CardFooterComponent, CardHeaderComponent, CardTitleComponent, CheckboxComponent, DataTableComponent, DatePickerComponent, DateRangePickerComponent, InputComponent, MaskedInputComponent, Modal, ModalComponent, ModalRef, ModalService, ModalStackService, MultiSelectComponent, PaginationComponent, RangeCalendarComponent, SelectComponent, SelectGroupComponent, SelectItemComponent, SelectSeparatorComponent, SkeletonComponent, SwitchComponent, TOLLE_CONFIG, ThemeService, ToastService, TolleCellDirective, TooltipDirective, cn, provideTolleConfig };
|
|
3643
3712
|
//# sourceMappingURL=tolle_-tolle-ui.mjs.map
|