@tolle_/tolle-ui 0.0.21-beta → 0.0.22-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 +94 -33
- package/fesm2022/tolle-ui.mjs +93 -32
- package/fesm2022/tolle-ui.mjs.map +1 -1
- package/lib/button.component.d.ts +1 -1
- package/lib/theme.service.d.ts +14 -3
- package/package.json +1 -1
|
@@ -21,29 +21,59 @@ export class ThemeService {
|
|
|
21
21
|
initializeTheme() {
|
|
22
22
|
if (!isPlatformBrowser(this.platformId))
|
|
23
23
|
return;
|
|
24
|
-
//
|
|
24
|
+
// LOGIC: User Saved Preferences > Config Defaults > System Preference
|
|
25
25
|
const savedTheme = localStorage.getItem('tolle-theme');
|
|
26
|
+
const savedPrimary = localStorage.getItem('tolle-primary-color');
|
|
27
|
+
const savedRadius = localStorage.getItem('tolle-radius');
|
|
26
28
|
const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
29
|
+
console.log('Theme initialization:', {
|
|
30
|
+
savedTheme,
|
|
31
|
+
savedPrimary,
|
|
32
|
+
savedRadius,
|
|
33
|
+
config: this.config,
|
|
34
|
+
systemPrefersDark
|
|
35
|
+
});
|
|
36
|
+
// 1. Determine Dark/Light Mode
|
|
37
|
+
// Priority: Saved Theme > Config Default > System Preference
|
|
38
|
+
let shouldBeDark = systemPrefersDark; // Start with system
|
|
39
|
+
if (savedTheme) {
|
|
40
|
+
// User has saved preference
|
|
41
|
+
shouldBeDark = savedTheme === 'dark';
|
|
42
|
+
}
|
|
43
|
+
else if (this.config?.darkByDefault !== undefined) {
|
|
44
|
+
// Use config default if no user preference
|
|
45
|
+
shouldBeDark = this.config.darkByDefault;
|
|
46
|
+
}
|
|
47
|
+
// Apply theme mode
|
|
31
48
|
if (shouldBeDark) {
|
|
32
|
-
this.enableDarkMode();
|
|
49
|
+
this.enableDarkMode(false); // Don't save, we'll save after checking all preferences
|
|
33
50
|
}
|
|
34
51
|
else {
|
|
35
|
-
this.disableDarkMode();
|
|
52
|
+
this.disableDarkMode(false);
|
|
36
53
|
}
|
|
37
|
-
|
|
54
|
+
// 2. Apply Primary Color
|
|
55
|
+
// Priority: Saved Color > Config Color
|
|
38
56
|
if (savedPrimary) {
|
|
39
|
-
|
|
57
|
+
// User has saved color preference
|
|
58
|
+
this.setPrimaryColor(savedPrimary, false); // Don't save again
|
|
40
59
|
}
|
|
41
60
|
else if (this.config?.primaryColor) {
|
|
42
|
-
|
|
61
|
+
// Use config default if no user preference
|
|
62
|
+
this.setPrimaryColor(this.config.primaryColor, true); // Save this as user preference
|
|
43
63
|
}
|
|
44
|
-
//
|
|
45
|
-
|
|
46
|
-
|
|
64
|
+
// 3. Apply Radius
|
|
65
|
+
// Priority: Saved Radius > Config Radius
|
|
66
|
+
if (savedRadius) {
|
|
67
|
+
// User has saved radius preference
|
|
68
|
+
this.setRadius(savedRadius, false);
|
|
69
|
+
}
|
|
70
|
+
else if (this.config?.radius) {
|
|
71
|
+
// Use config default if no user preference
|
|
72
|
+
this.setRadius(this.config.radius, true);
|
|
73
|
+
}
|
|
74
|
+
// Save theme mode preference if it came from config or system
|
|
75
|
+
if (!savedTheme) {
|
|
76
|
+
localStorage.setItem('tolle-theme', shouldBeDark ? 'dark' : 'light');
|
|
47
77
|
}
|
|
48
78
|
}
|
|
49
79
|
/**
|
|
@@ -249,14 +279,18 @@ export class ThemeService {
|
|
|
249
279
|
const isCurrentlyDark = this.document.documentElement.classList.contains('dark');
|
|
250
280
|
isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();
|
|
251
281
|
}
|
|
252
|
-
enableDarkMode() {
|
|
282
|
+
enableDarkMode(saveToStorage = true) {
|
|
253
283
|
this.renderer.addClass(this.document.documentElement, 'dark');
|
|
254
|
-
|
|
284
|
+
if (saveToStorage) {
|
|
285
|
+
localStorage.setItem('tolle-theme', 'dark');
|
|
286
|
+
}
|
|
255
287
|
this.isDarkSubject.next(true);
|
|
256
288
|
}
|
|
257
|
-
disableDarkMode() {
|
|
289
|
+
disableDarkMode(saveToStorage = true) {
|
|
258
290
|
this.renderer.removeClass(this.document.documentElement, 'dark');
|
|
259
|
-
|
|
291
|
+
if (saveToStorage) {
|
|
292
|
+
localStorage.setItem('tolle-theme', 'light');
|
|
293
|
+
}
|
|
260
294
|
this.isDarkSubject.next(false);
|
|
261
295
|
}
|
|
262
296
|
setPrimaryColor(color, persist = true) {
|
|
@@ -289,29 +323,56 @@ export class ThemeService {
|
|
|
289
323
|
}
|
|
290
324
|
return localStorage.getItem('tolle-primary-color');
|
|
291
325
|
}
|
|
292
|
-
|
|
326
|
+
/**
|
|
327
|
+
* Reset to config defaults (clears user preferences)
|
|
328
|
+
*/
|
|
329
|
+
resetToConfigDefaults() {
|
|
330
|
+
if (!isPlatformBrowser(this.platformId))
|
|
331
|
+
return;
|
|
332
|
+
// Clear user preferences
|
|
333
|
+
localStorage.removeItem('tolle-theme');
|
|
334
|
+
localStorage.removeItem('tolle-primary-color');
|
|
335
|
+
localStorage.removeItem('tolle-radius');
|
|
336
|
+
// Re-initialize with config defaults
|
|
337
|
+
this.initializeTheme();
|
|
338
|
+
}
|
|
339
|
+
/**
|
|
340
|
+
* Get current user preferences
|
|
341
|
+
*/
|
|
342
|
+
getUserPreferences() {
|
|
293
343
|
if (!isPlatformBrowser(this.platformId))
|
|
294
344
|
return null;
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
}
|
|
300
|
-
return localStorage.getItem('tolle-radius') || '0.5rem';
|
|
345
|
+
return {
|
|
346
|
+
theme: localStorage.getItem('tolle-theme'),
|
|
347
|
+
primaryColor: localStorage.getItem('tolle-primary-color'),
|
|
348
|
+
radius: localStorage.getItem('tolle-radius')
|
|
349
|
+
};
|
|
301
350
|
}
|
|
302
351
|
/**
|
|
303
|
-
*
|
|
352
|
+
* Clear all user preferences
|
|
304
353
|
*/
|
|
305
|
-
|
|
354
|
+
clearUserPreferences() {
|
|
306
355
|
if (!isPlatformBrowser(this.platformId))
|
|
307
356
|
return;
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
357
|
+
localStorage.removeItem('tolle-theme');
|
|
358
|
+
localStorage.removeItem('tolle-primary-color');
|
|
359
|
+
localStorage.removeItem('tolle-radius');
|
|
360
|
+
// Reset to system defaults (not config defaults)
|
|
361
|
+
const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
362
|
+
if (systemPrefersDark) {
|
|
363
|
+
this.enableDarkMode(false);
|
|
311
364
|
}
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
365
|
+
else {
|
|
366
|
+
this.disableDarkMode(false);
|
|
367
|
+
}
|
|
368
|
+
// Remove CSS variables
|
|
369
|
+
const root = this.document.documentElement;
|
|
370
|
+
this.renderer.removeStyle(root, '--primary');
|
|
371
|
+
this.renderer.removeStyle(root, '--radius');
|
|
372
|
+
// Clear dynamic styles
|
|
373
|
+
const existingStyle = this.document.getElementById(this.styleId);
|
|
374
|
+
if (existingStyle) {
|
|
375
|
+
existingStyle.remove();
|
|
315
376
|
}
|
|
316
377
|
}
|
|
317
378
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ThemeService, deps: [{ token: DOCUMENT }, { token: PLATFORM_ID }, { token: TOLLE_CONFIG, optional: true }, { token: i0.RendererFactory2 }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
@@ -332,4 +393,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
332
393
|
type: Inject,
|
|
333
394
|
args: [TOLLE_CONFIG]
|
|
334
395
|
}] }, { type: i0.RendererFactory2 }] });
|
|
335
|
-
//# 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    const savedPrimary = localStorage.getItem('tolle-primary-color');\n\n    if (savedPrimary) {\n      this.setPrimaryColor(savedPrimary, false);\n    } else if (this.config?.primaryColor) {\n      this.setPrimaryColor(this.config.primaryColor, false);\n    }\n\n    // Set radius from config\n    if (this.config?.radius) {\n      this.setRadius(this.config.radius, false);\n    }\n  }\n\n  /**\n   * Sets the border radius for all components\n   */\n  setRadius(radius: string, persist = true) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    const root = this.document.documentElement;\n\n    // Set the CSS variable\n    this.renderer.setStyle(root, '--radius', radius);\n\n    // Also update the dynamic styles to include radius calculations\n    this.updateRadiusInDynamicStyles(radius);\n\n    // Persist if needed\n    if (persist) {\n      localStorage.setItem('tolle-radius', radius);\n    }\n  }\n\n  /**\n   * Updates the radius calculations in dynamic styles\n   */\n  private updateRadiusInDynamicStyles(radius: string) {\n    const existingStyle = this.document.getElementById(this.styleId);\n\n    if (existingStyle) {\n      let css = existingStyle.textContent || '';\n\n      // Update or add radius calculations\n      if (css.includes('--radius:')) {\n        // Replace existing radius declarations\n        css = css.replace(/--radius:[^;]+;/g, `--radius: ${radius};`);\n      } else {\n        // Add radius to the beginning of :root\n        css = css.replace(/:root\\s*{/, `:root {\\n      --radius: ${radius};`);\n      }\n\n      // Update the calculated radius values in the CSS\n      const radiusCalcRegex = /calc\\(var\\(--radius[^)]+\\)/g;\n      css = css.replace(radiusCalcRegex, (match) => {\n        if (match.includes('- 2px')) {\n          return `calc(${radius} - 2px)`;\n        } else if (match.includes('- 4px')) {\n          return `calc(${radius} - 4px)`;\n        }\n        return match;\n      });\n\n      existingStyle.textContent = css;\n    }\n  }\n\n  /**\n   * Generates full primary color palette (50-900) based on base color\n   */\n  private generatePrimaryShades(baseColor: string) {\n    // Convert hex to RGB\n    const rgb = this.hexToRgb(baseColor);\n    const rgbString = rgb ? `${rgb.r} ${rgb.g} ${rgb.b}` : '37 99 235';\n\n    // Create lighter ring colors in RGB\n    const ringLight = this.lightenColor(baseColor, 40);\n    const ringLightRgb = this.hexToRgb(ringLight);\n    const ringLightRgbString = ringLightRgb ? `${ringLightRgb.r} ${ringLightRgb.g} ${ringLightRgb.b}` : '96 165 250';\n\n    const ringDark = this.lightenColor(baseColor, 20);\n    const ringDarkRgb = this.hexToRgb(ringDark);\n    const ringDarkRgbString = ringDarkRgb ? `${ringDarkRgb.r} ${ringDarkRgb.g} ${ringDarkRgb.b}` : '147 197 253';\n\n    // Get current radius or use default\n    const root = this.document.documentElement;\n    const currentRadius = getComputedStyle(root).getPropertyValue('--radius').trim() || '0.5rem';\n\n    const css = `\n      /* Override primary colors - this needs to come AFTER your main CSS */\n      :root {\n        /* Primary in RGB format for Tailwind opacity support */\n        --primary: ${rgbString};\n        --primary-foreground: ${this.getContrastColorRgb(baseColor)};\n\n        /* Radius */\n        --radius: ${currentRadius};\n\n        /* Primary shades for light mode */\n        --primary-50: ${this.hexToRgbString(this.lightenColor(baseColor, 90))};\n        --primary-100: ${this.hexToRgbString(this.lightenColor(baseColor, 80))};\n        --primary-200: ${this.hexToRgbString(this.lightenColor(baseColor, 60))};\n        --primary-300: ${this.hexToRgbString(this.lightenColor(baseColor, 40))};\n        --primary-400: ${this.hexToRgbString(this.lightenColor(baseColor, 20))};\n        --primary-500: ${rgbString};\n        --primary-600: ${this.hexToRgbString(this.darkenColor(baseColor, 20))};\n        --primary-700: ${this.hexToRgbString(this.darkenColor(baseColor, 40))};\n        --primary-800: ${this.hexToRgbString(this.darkenColor(baseColor, 60))};\n        --primary-900: ${this.hexToRgbString(this.darkenColor(baseColor, 80))};\n\n        /* Update ring color to be lighter */\n        --ring: ${ringLightRgbString};\n      }\n\n      .dark {\n        /* For dark mode, we keep the primary color but adjust shades */\n        --primary: ${rgbString};\n        --primary-foreground: ${this.getContrastColorRgb(baseColor)};\n\n        /* Dark mode shades */\n        --primary-50: ${this.hexToRgbString(this.darkenColor(baseColor, 85))};\n        --primary-100: ${this.hexToRgbString(this.darkenColor(baseColor, 75))};\n        --primary-200: ${this.hexToRgbString(this.darkenColor(baseColor, 65))};\n        --primary-300: ${this.hexToRgbString(this.darkenColor(baseColor, 55))};\n        --primary-400: ${this.hexToRgbString(this.darkenColor(baseColor, 45))};\n        --primary-500: ${rgbString};\n        --primary-600: ${this.hexToRgbString(this.lightenColor(baseColor, 20))};\n        --primary-700: ${this.hexToRgbString(this.lightenColor(baseColor, 35))};\n        --primary-800: ${this.hexToRgbString(this.lightenColor(baseColor, 50))};\n        --primary-900: ${this.hexToRgbString(this.lightenColor(baseColor, 65))};\n\n        /* Update ring color for dark mode - lighter variant */\n        --ring: ${ringDarkRgbString};\n      }\n    `;\n\n    this.injectDynamicStyles(css);\n  }\n\n  /**\n   * Convert hex color to RGB object\n   */\n  private hexToRgb(hex: string): { r: number, g: number, b: number } | null {\n    // Remove # if present\n    const cleanedHex = hex.replace('#', '');\n\n    let r = 0, g = 0, b = 0;\n\n    if (cleanedHex.length === 3) {\n      r = parseInt(cleanedHex[0] + cleanedHex[0], 16);\n      g = parseInt(cleanedHex[1] + cleanedHex[1], 16);\n      b = parseInt(cleanedHex[2] + cleanedHex[2], 16);\n      return { r, g, b };\n    }\n\n    if (cleanedHex.length === 6) {\n      r = parseInt(cleanedHex.substring(0, 2), 16);\n      g = parseInt(cleanedHex.substring(2, 4), 16);\n      b = parseInt(cleanedHex.substring(4, 6), 16);\n      return { r, g, b };\n    }\n\n    return null;\n  }\n\n  /**\n   * Convert hex to RGB string format for CSS (space-separated)\n   */\n  private hexToRgbString(hexColor: string): string {\n    const rgb = this.hexToRgb(hexColor);\n    return rgb ? `${rgb.r} ${rgb.g} ${rgb.b}` : '37 99 235';\n  }\n\n  /**\n   * Get contrast color in RGB format\n   */\n  private getContrastColorRgb(hexColor: string): string {\n    const contrast = this.getContrastColor(hexColor);\n    const rgb = this.hexToRgb(contrast);\n    return rgb ? `${rgb.r} ${rgb.g} ${rgb.b}` : '255 255 255';\n  }\n\n  /**\n   * Lighten a hex color by a percentage\n   */\n  private lightenColor(color: string, percent: number): string {\n    const rgb = this.hexToRgb(color);\n    if (!rgb) return color;\n\n    // Lighten by percentage\n    const r = Math.min(255, Math.floor(rgb.r + (255 - rgb.r) * (percent / 100)));\n    const g = Math.min(255, Math.floor(rgb.g + (255 - rgb.g) * (percent / 100)));\n    const b = Math.min(255, Math.floor(rgb.b + (255 - rgb.b) * (percent / 100)));\n\n    // Convert back to hex\n    return `#${((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1)}`;\n  }\n\n  /**\n   * Darken a hex color by a percentage\n   */\n  private darkenColor(color: string, percent: number): string {\n    const rgb = this.hexToRgb(color);\n    if (!rgb) return color;\n\n    const factor = 1 - (percent / 100);\n    const r = Math.max(0, Math.floor(rgb.r * factor));\n    const g = Math.max(0, Math.floor(rgb.g * factor));\n    const b = Math.max(0, Math.floor(rgb.b * factor));\n\n    return `#${r.toString(16).padStart(2, '0')}${g.toString(16).padStart(2, '0')}${b.toString(16).padStart(2, '0')}`;\n  }\n\n  /**\n   * Calculate contrast color (black or white) based on background color\n   */\n  private getContrastColor(hexColor: string): string {\n    const rgb = this.hexToRgb(hexColor);\n    if (!rgb) return '#ffffff';\n\n    // Calculate relative luminance\n    const luminance = (0.299 * rgb.r + 0.587 * rgb.g + 0.114 * rgb.b) / 255;\n\n    // Return black for light colors, white for dark colors\n    return luminance > 0.5 ? '#000000' : '#ffffff';\n  }\n\n  private injectDynamicStyles(css: string) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    const existingStyle = this.document.getElementById(this.styleId);\n    if (existingStyle) {\n      existingStyle.remove();\n    }\n\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  setPrimaryColor(color: string, persist = true) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    this.generatePrimaryShades(color);\n\n    // Also set inline for immediate update\n    const rgb = this.hexToRgb(color);\n    if (rgb) {\n      this.renderer.setStyle(\n        this.document.documentElement,\n        '--primary',\n        `${rgb.r} ${rgb.g} ${rgb.b}`\n      );\n    }\n\n    if (persist) {\n      localStorage.setItem('tolle-primary-color', color);\n    }\n  }\n\n  get currentTheme(): 'dark' | 'light' {\n    return this.isDarkSubject.value ? 'dark' : 'light';\n  }\n\n  get primaryColor(): string | null {\n    if (!isPlatformBrowser(this.platformId)) return null;\n\n    const root = this.document.documentElement;\n    const cssValue = getComputedStyle(root).getPropertyValue('--primary').trim();\n\n    if (cssValue && cssValue !== '') {\n      // Convert RGB string back to hex for external use\n      const rgbParts = cssValue.split(' ').map(Number);\n      if (rgbParts.length === 3) {\n        return `#${rgbParts[0].toString(16).padStart(2, '0')}${rgbParts[1].toString(16).padStart(2, '0')}${rgbParts[2].toString(16).padStart(2, '0')}`;\n      }\n    }\n\n    return localStorage.getItem('tolle-primary-color');\n  }\n\n  get radius(): string | null {\n    if (!isPlatformBrowser(this.platformId)) return null;\n\n    const root = this.document.documentElement;\n    const cssValue = getComputedStyle(root).getPropertyValue('--radius').trim();\n\n    if (cssValue && cssValue !== '') {\n      return cssValue;\n    }\n\n    return localStorage.getItem('tolle-radius') || '0.5rem';\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    // Set primary color if provided\n    if (config.primaryColor) {\n      this.setPrimaryColor(config.primaryColor, false);\n    }\n\n    // Set border radius if provided\n    if (config.radius) {\n      this.setRadius(config.radius, false);\n    }\n  }\n}\n"]}
|
|
396
|
+
//# 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    // LOGIC: User Saved Preferences > Config Defaults > System Preference\n    const savedTheme = localStorage.getItem('tolle-theme');\n    const savedPrimary = localStorage.getItem('tolle-primary-color');\n    const savedRadius = localStorage.getItem('tolle-radius');\n    const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;\n\n    console.log('Theme initialization:', {\n      savedTheme,\n      savedPrimary,\n      savedRadius,\n      config: this.config,\n      systemPrefersDark\n    });\n\n    // 1. Determine Dark/Light Mode\n    // Priority: Saved Theme > Config Default > System Preference\n    let shouldBeDark = systemPrefersDark; // Start with system\n\n    if (savedTheme) {\n      // User has saved preference\n      shouldBeDark = savedTheme === 'dark';\n    } else if (this.config?.darkByDefault !== undefined) {\n      // Use config default if no user preference\n      shouldBeDark = this.config.darkByDefault;\n    }\n\n    // Apply theme mode\n    if (shouldBeDark) {\n      this.enableDarkMode(false); // Don't save, we'll save after checking all preferences\n    } else {\n      this.disableDarkMode(false);\n    }\n\n    // 2. Apply Primary Color\n    // Priority: Saved Color > Config Color\n    if (savedPrimary) {\n      // User has saved color preference\n      this.setPrimaryColor(savedPrimary, false); // Don't save again\n    } else if (this.config?.primaryColor) {\n      // Use config default if no user preference\n      this.setPrimaryColor(this.config.primaryColor, true); // Save this as user preference\n    }\n\n    // 3. Apply Radius\n    // Priority: Saved Radius > Config Radius\n    if (savedRadius) {\n      // User has saved radius preference\n      this.setRadius(savedRadius, false);\n    } else if (this.config?.radius) {\n      // Use config default if no user preference\n      this.setRadius(this.config.radius, true);\n    }\n\n    // Save theme mode preference if it came from config or system\n    if (!savedTheme) {\n      localStorage.setItem('tolle-theme', shouldBeDark ? 'dark' : 'light');\n    }\n  }\n  /**\n   * Sets the border radius for all components\n   */\n  setRadius(radius: string, persist = true) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    const root = this.document.documentElement;\n\n    // Set the CSS variable\n    this.renderer.setStyle(root, '--radius', radius);\n\n    // Also update the dynamic styles to include radius calculations\n    this.updateRadiusInDynamicStyles(radius);\n\n    // Persist if needed\n    if (persist) {\n      localStorage.setItem('tolle-radius', radius);\n    }\n  }\n\n  /**\n   * Updates the radius calculations in dynamic styles\n   */\n  private updateRadiusInDynamicStyles(radius: string) {\n    const existingStyle = this.document.getElementById(this.styleId);\n\n    if (existingStyle) {\n      let css = existingStyle.textContent || '';\n\n      // Update or add radius calculations\n      if (css.includes('--radius:')) {\n        // Replace existing radius declarations\n        css = css.replace(/--radius:[^;]+;/g, `--radius: ${radius};`);\n      } else {\n        // Add radius to the beginning of :root\n        css = css.replace(/:root\\s*{/, `:root {\\n      --radius: ${radius};`);\n      }\n\n      // Update the calculated radius values in the CSS\n      const radiusCalcRegex = /calc\\(var\\(--radius[^)]+\\)/g;\n      css = css.replace(radiusCalcRegex, (match) => {\n        if (match.includes('- 2px')) {\n          return `calc(${radius} - 2px)`;\n        } else if (match.includes('- 4px')) {\n          return `calc(${radius} - 4px)`;\n        }\n        return match;\n      });\n\n      existingStyle.textContent = css;\n    }\n  }\n\n  /**\n   * Generates full primary color palette (50-900) based on base color\n   */\n  private generatePrimaryShades(baseColor: string) {\n    // Convert hex to RGB\n    const rgb = this.hexToRgb(baseColor);\n    const rgbString = rgb ? `${rgb.r} ${rgb.g} ${rgb.b}` : '37 99 235';\n\n    // Create lighter ring colors in RGB\n    const ringLight = this.lightenColor(baseColor, 40);\n    const ringLightRgb = this.hexToRgb(ringLight);\n    const ringLightRgbString = ringLightRgb ? `${ringLightRgb.r} ${ringLightRgb.g} ${ringLightRgb.b}` : '96 165 250';\n\n    const ringDark = this.lightenColor(baseColor, 20);\n    const ringDarkRgb = this.hexToRgb(ringDark);\n    const ringDarkRgbString = ringDarkRgb ? `${ringDarkRgb.r} ${ringDarkRgb.g} ${ringDarkRgb.b}` : '147 197 253';\n\n    // Get current radius or use default\n    const root = this.document.documentElement;\n    const currentRadius = getComputedStyle(root).getPropertyValue('--radius').trim() || '0.5rem';\n\n    const css = `\n      /* Override primary colors - this needs to come AFTER your main CSS */\n      :root {\n        /* Primary in RGB format for Tailwind opacity support */\n        --primary: ${rgbString};\n        --primary-foreground: ${this.getContrastColorRgb(baseColor)};\n\n        /* Radius */\n        --radius: ${currentRadius};\n\n        /* Primary shades for light mode */\n        --primary-50: ${this.hexToRgbString(this.lightenColor(baseColor, 90))};\n        --primary-100: ${this.hexToRgbString(this.lightenColor(baseColor, 80))};\n        --primary-200: ${this.hexToRgbString(this.lightenColor(baseColor, 60))};\n        --primary-300: ${this.hexToRgbString(this.lightenColor(baseColor, 40))};\n        --primary-400: ${this.hexToRgbString(this.lightenColor(baseColor, 20))};\n        --primary-500: ${rgbString};\n        --primary-600: ${this.hexToRgbString(this.darkenColor(baseColor, 20))};\n        --primary-700: ${this.hexToRgbString(this.darkenColor(baseColor, 40))};\n        --primary-800: ${this.hexToRgbString(this.darkenColor(baseColor, 60))};\n        --primary-900: ${this.hexToRgbString(this.darkenColor(baseColor, 80))};\n\n        /* Update ring color to be lighter */\n        --ring: ${ringLightRgbString};\n      }\n\n      .dark {\n        /* For dark mode, we keep the primary color but adjust shades */\n        --primary: ${rgbString};\n        --primary-foreground: ${this.getContrastColorRgb(baseColor)};\n\n        /* Dark mode shades */\n        --primary-50: ${this.hexToRgbString(this.darkenColor(baseColor, 85))};\n        --primary-100: ${this.hexToRgbString(this.darkenColor(baseColor, 75))};\n        --primary-200: ${this.hexToRgbString(this.darkenColor(baseColor, 65))};\n        --primary-300: ${this.hexToRgbString(this.darkenColor(baseColor, 55))};\n        --primary-400: ${this.hexToRgbString(this.darkenColor(baseColor, 45))};\n        --primary-500: ${rgbString};\n        --primary-600: ${this.hexToRgbString(this.lightenColor(baseColor, 20))};\n        --primary-700: ${this.hexToRgbString(this.lightenColor(baseColor, 35))};\n        --primary-800: ${this.hexToRgbString(this.lightenColor(baseColor, 50))};\n        --primary-900: ${this.hexToRgbString(this.lightenColor(baseColor, 65))};\n\n        /* Update ring color for dark mode - lighter variant */\n        --ring: ${ringDarkRgbString};\n      }\n    `;\n\n    this.injectDynamicStyles(css);\n  }\n\n  /**\n   * Convert hex color to RGB object\n   */\n  private hexToRgb(hex: string): { r: number, g: number, b: number } | null {\n    // Remove # if present\n    const cleanedHex = hex.replace('#', '');\n\n    let r = 0, g = 0, b = 0;\n\n    if (cleanedHex.length === 3) {\n      r = parseInt(cleanedHex[0] + cleanedHex[0], 16);\n      g = parseInt(cleanedHex[1] + cleanedHex[1], 16);\n      b = parseInt(cleanedHex[2] + cleanedHex[2], 16);\n      return { r, g, b };\n    }\n\n    if (cleanedHex.length === 6) {\n      r = parseInt(cleanedHex.substring(0, 2), 16);\n      g = parseInt(cleanedHex.substring(2, 4), 16);\n      b = parseInt(cleanedHex.substring(4, 6), 16);\n      return { r, g, b };\n    }\n\n    return null;\n  }\n\n  /**\n   * Convert hex to RGB string format for CSS (space-separated)\n   */\n  private hexToRgbString(hexColor: string): string {\n    const rgb = this.hexToRgb(hexColor);\n    return rgb ? `${rgb.r} ${rgb.g} ${rgb.b}` : '37 99 235';\n  }\n\n  /**\n   * Get contrast color in RGB format\n   */\n  private getContrastColorRgb(hexColor: string): string {\n    const contrast = this.getContrastColor(hexColor);\n    const rgb = this.hexToRgb(contrast);\n    return rgb ? `${rgb.r} ${rgb.g} ${rgb.b}` : '255 255 255';\n  }\n\n  /**\n   * Lighten a hex color by a percentage\n   */\n  private lightenColor(color: string, percent: number): string {\n    const rgb = this.hexToRgb(color);\n    if (!rgb) return color;\n\n    // Lighten by percentage\n    const r = Math.min(255, Math.floor(rgb.r + (255 - rgb.r) * (percent / 100)));\n    const g = Math.min(255, Math.floor(rgb.g + (255 - rgb.g) * (percent / 100)));\n    const b = Math.min(255, Math.floor(rgb.b + (255 - rgb.b) * (percent / 100)));\n\n    // Convert back to hex\n    return `#${((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1)}`;\n  }\n\n  /**\n   * Darken a hex color by a percentage\n   */\n  private darkenColor(color: string, percent: number): string {\n    const rgb = this.hexToRgb(color);\n    if (!rgb) return color;\n\n    const factor = 1 - (percent / 100);\n    const r = Math.max(0, Math.floor(rgb.r * factor));\n    const g = Math.max(0, Math.floor(rgb.g * factor));\n    const b = Math.max(0, Math.floor(rgb.b * factor));\n\n    return `#${r.toString(16).padStart(2, '0')}${g.toString(16).padStart(2, '0')}${b.toString(16).padStart(2, '0')}`;\n  }\n\n  /**\n   * Calculate contrast color (black or white) based on background color\n   */\n  private getContrastColor(hexColor: string): string {\n    const rgb = this.hexToRgb(hexColor);\n    if (!rgb) return '#ffffff';\n\n    // Calculate relative luminance\n    const luminance = (0.299 * rgb.r + 0.587 * rgb.g + 0.114 * rgb.b) / 255;\n\n    // Return black for light colors, white for dark colors\n    return luminance > 0.5 ? '#000000' : '#ffffff';\n  }\n\n  private injectDynamicStyles(css: string) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    const existingStyle = this.document.getElementById(this.styleId);\n    if (existingStyle) {\n      existingStyle.remove();\n    }\n\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(saveToStorage = true) {\n    this.renderer.addClass(this.document.documentElement, 'dark');\n    if (saveToStorage) {\n      localStorage.setItem('tolle-theme', 'dark');\n    }\n    this.isDarkSubject.next(true);\n  }\n\n  private disableDarkMode(saveToStorage = true) {\n    this.renderer.removeClass(this.document.documentElement, 'dark');\n    if (saveToStorage) {\n      localStorage.setItem('tolle-theme', 'light');\n    }\n    this.isDarkSubject.next(false);\n  }\n  setPrimaryColor(color: string, persist = true) {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    this.generatePrimaryShades(color);\n\n    // Also set inline for immediate update\n    const rgb = this.hexToRgb(color);\n    if (rgb) {\n      this.renderer.setStyle(\n        this.document.documentElement,\n        '--primary',\n        `${rgb.r} ${rgb.g} ${rgb.b}`\n      );\n    }\n\n    if (persist) {\n      localStorage.setItem('tolle-primary-color', color);\n    }\n  }\n\n  get currentTheme(): 'dark' | 'light' {\n    return this.isDarkSubject.value ? 'dark' : 'light';\n  }\n\n  get primaryColor(): string | null {\n    if (!isPlatformBrowser(this.platformId)) return null;\n\n    const root = this.document.documentElement;\n    const cssValue = getComputedStyle(root).getPropertyValue('--primary').trim();\n\n    if (cssValue && cssValue !== '') {\n      // Convert RGB string back to hex for external use\n      const rgbParts = cssValue.split(' ').map(Number);\n      if (rgbParts.length === 3) {\n        return `#${rgbParts[0].toString(16).padStart(2, '0')}${rgbParts[1].toString(16).padStart(2, '0')}${rgbParts[2].toString(16).padStart(2, '0')}`;\n      }\n    }\n\n    return localStorage.getItem('tolle-primary-color');\n  }\n\n  /**\n   * Reset to config defaults (clears user preferences)\n   */\n  resetToConfigDefaults() {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    // Clear user preferences\n    localStorage.removeItem('tolle-theme');\n    localStorage.removeItem('tolle-primary-color');\n    localStorage.removeItem('tolle-radius');\n\n    // Re-initialize with config defaults\n    this.initializeTheme();\n  }\n\n  /**\n   * Get current user preferences\n   */\n  getUserPreferences() {\n    if (!isPlatformBrowser(this.platformId)) return null;\n\n    return {\n      theme: localStorage.getItem('tolle-theme'),\n      primaryColor: localStorage.getItem('tolle-primary-color'),\n      radius: localStorage.getItem('tolle-radius')\n    };\n  }\n\n  /**\n   * Clear all user preferences\n   */\n  clearUserPreferences() {\n    if (!isPlatformBrowser(this.platformId)) return;\n\n    localStorage.removeItem('tolle-theme');\n    localStorage.removeItem('tolle-primary-color');\n    localStorage.removeItem('tolle-radius');\n\n    // Reset to system defaults (not config defaults)\n    const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;\n    if (systemPrefersDark) {\n      this.enableDarkMode(false);\n    } else {\n      this.disableDarkMode(false);\n    }\n\n    // Remove CSS variables\n    const root = this.document.documentElement;\n    this.renderer.removeStyle(root, '--primary');\n    this.renderer.removeStyle(root, '--radius');\n\n    // Clear dynamic styles\n    const existingStyle = this.document.getElementById(this.styleId);\n    if (existingStyle) {\n      existingStyle.remove();\n    }\n  }\n}\n"]}
|
package/fesm2022/tolle-ui.mjs
CHANGED
|
@@ -1470,29 +1470,59 @@ class ThemeService {
|
|
|
1470
1470
|
initializeTheme() {
|
|
1471
1471
|
if (!isPlatformBrowser(this.platformId))
|
|
1472
1472
|
return;
|
|
1473
|
-
//
|
|
1473
|
+
// LOGIC: User Saved Preferences > Config Defaults > System Preference
|
|
1474
1474
|
const savedTheme = localStorage.getItem('tolle-theme');
|
|
1475
|
+
const savedPrimary = localStorage.getItem('tolle-primary-color');
|
|
1476
|
+
const savedRadius = localStorage.getItem('tolle-radius');
|
|
1475
1477
|
const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
1476
|
-
|
|
1477
|
-
|
|
1478
|
-
|
|
1479
|
-
|
|
1478
|
+
console.log('Theme initialization:', {
|
|
1479
|
+
savedTheme,
|
|
1480
|
+
savedPrimary,
|
|
1481
|
+
savedRadius,
|
|
1482
|
+
config: this.config,
|
|
1483
|
+
systemPrefersDark
|
|
1484
|
+
});
|
|
1485
|
+
// 1. Determine Dark/Light Mode
|
|
1486
|
+
// Priority: Saved Theme > Config Default > System Preference
|
|
1487
|
+
let shouldBeDark = systemPrefersDark; // Start with system
|
|
1488
|
+
if (savedTheme) {
|
|
1489
|
+
// User has saved preference
|
|
1490
|
+
shouldBeDark = savedTheme === 'dark';
|
|
1491
|
+
}
|
|
1492
|
+
else if (this.config?.darkByDefault !== undefined) {
|
|
1493
|
+
// Use config default if no user preference
|
|
1494
|
+
shouldBeDark = this.config.darkByDefault;
|
|
1495
|
+
}
|
|
1496
|
+
// Apply theme mode
|
|
1480
1497
|
if (shouldBeDark) {
|
|
1481
|
-
this.enableDarkMode();
|
|
1498
|
+
this.enableDarkMode(false); // Don't save, we'll save after checking all preferences
|
|
1482
1499
|
}
|
|
1483
1500
|
else {
|
|
1484
|
-
this.disableDarkMode();
|
|
1501
|
+
this.disableDarkMode(false);
|
|
1485
1502
|
}
|
|
1486
|
-
|
|
1503
|
+
// 2. Apply Primary Color
|
|
1504
|
+
// Priority: Saved Color > Config Color
|
|
1487
1505
|
if (savedPrimary) {
|
|
1488
|
-
|
|
1506
|
+
// User has saved color preference
|
|
1507
|
+
this.setPrimaryColor(savedPrimary, false); // Don't save again
|
|
1489
1508
|
}
|
|
1490
1509
|
else if (this.config?.primaryColor) {
|
|
1491
|
-
|
|
1510
|
+
// Use config default if no user preference
|
|
1511
|
+
this.setPrimaryColor(this.config.primaryColor, true); // Save this as user preference
|
|
1512
|
+
}
|
|
1513
|
+
// 3. Apply Radius
|
|
1514
|
+
// Priority: Saved Radius > Config Radius
|
|
1515
|
+
if (savedRadius) {
|
|
1516
|
+
// User has saved radius preference
|
|
1517
|
+
this.setRadius(savedRadius, false);
|
|
1518
|
+
}
|
|
1519
|
+
else if (this.config?.radius) {
|
|
1520
|
+
// Use config default if no user preference
|
|
1521
|
+
this.setRadius(this.config.radius, true);
|
|
1492
1522
|
}
|
|
1493
|
-
//
|
|
1494
|
-
if (
|
|
1495
|
-
|
|
1523
|
+
// Save theme mode preference if it came from config or system
|
|
1524
|
+
if (!savedTheme) {
|
|
1525
|
+
localStorage.setItem('tolle-theme', shouldBeDark ? 'dark' : 'light');
|
|
1496
1526
|
}
|
|
1497
1527
|
}
|
|
1498
1528
|
/**
|
|
@@ -1698,14 +1728,18 @@ class ThemeService {
|
|
|
1698
1728
|
const isCurrentlyDark = this.document.documentElement.classList.contains('dark');
|
|
1699
1729
|
isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();
|
|
1700
1730
|
}
|
|
1701
|
-
enableDarkMode() {
|
|
1731
|
+
enableDarkMode(saveToStorage = true) {
|
|
1702
1732
|
this.renderer.addClass(this.document.documentElement, 'dark');
|
|
1703
|
-
|
|
1733
|
+
if (saveToStorage) {
|
|
1734
|
+
localStorage.setItem('tolle-theme', 'dark');
|
|
1735
|
+
}
|
|
1704
1736
|
this.isDarkSubject.next(true);
|
|
1705
1737
|
}
|
|
1706
|
-
disableDarkMode() {
|
|
1738
|
+
disableDarkMode(saveToStorage = true) {
|
|
1707
1739
|
this.renderer.removeClass(this.document.documentElement, 'dark');
|
|
1708
|
-
|
|
1740
|
+
if (saveToStorage) {
|
|
1741
|
+
localStorage.setItem('tolle-theme', 'light');
|
|
1742
|
+
}
|
|
1709
1743
|
this.isDarkSubject.next(false);
|
|
1710
1744
|
}
|
|
1711
1745
|
setPrimaryColor(color, persist = true) {
|
|
@@ -1738,29 +1772,56 @@ class ThemeService {
|
|
|
1738
1772
|
}
|
|
1739
1773
|
return localStorage.getItem('tolle-primary-color');
|
|
1740
1774
|
}
|
|
1741
|
-
|
|
1775
|
+
/**
|
|
1776
|
+
* Reset to config defaults (clears user preferences)
|
|
1777
|
+
*/
|
|
1778
|
+
resetToConfigDefaults() {
|
|
1779
|
+
if (!isPlatformBrowser(this.platformId))
|
|
1780
|
+
return;
|
|
1781
|
+
// Clear user preferences
|
|
1782
|
+
localStorage.removeItem('tolle-theme');
|
|
1783
|
+
localStorage.removeItem('tolle-primary-color');
|
|
1784
|
+
localStorage.removeItem('tolle-radius');
|
|
1785
|
+
// Re-initialize with config defaults
|
|
1786
|
+
this.initializeTheme();
|
|
1787
|
+
}
|
|
1788
|
+
/**
|
|
1789
|
+
* Get current user preferences
|
|
1790
|
+
*/
|
|
1791
|
+
getUserPreferences() {
|
|
1742
1792
|
if (!isPlatformBrowser(this.platformId))
|
|
1743
1793
|
return null;
|
|
1744
|
-
|
|
1745
|
-
|
|
1746
|
-
|
|
1747
|
-
|
|
1748
|
-
}
|
|
1749
|
-
return localStorage.getItem('tolle-radius') || '0.5rem';
|
|
1794
|
+
return {
|
|
1795
|
+
theme: localStorage.getItem('tolle-theme'),
|
|
1796
|
+
primaryColor: localStorage.getItem('tolle-primary-color'),
|
|
1797
|
+
radius: localStorage.getItem('tolle-radius')
|
|
1798
|
+
};
|
|
1750
1799
|
}
|
|
1751
1800
|
/**
|
|
1752
|
-
*
|
|
1801
|
+
* Clear all user preferences
|
|
1753
1802
|
*/
|
|
1754
|
-
|
|
1803
|
+
clearUserPreferences() {
|
|
1755
1804
|
if (!isPlatformBrowser(this.platformId))
|
|
1756
1805
|
return;
|
|
1757
|
-
|
|
1758
|
-
|
|
1759
|
-
|
|
1806
|
+
localStorage.removeItem('tolle-theme');
|
|
1807
|
+
localStorage.removeItem('tolle-primary-color');
|
|
1808
|
+
localStorage.removeItem('tolle-radius');
|
|
1809
|
+
// Reset to system defaults (not config defaults)
|
|
1810
|
+
const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
1811
|
+
if (systemPrefersDark) {
|
|
1812
|
+
this.enableDarkMode(false);
|
|
1813
|
+
}
|
|
1814
|
+
else {
|
|
1815
|
+
this.disableDarkMode(false);
|
|
1760
1816
|
}
|
|
1761
|
-
//
|
|
1762
|
-
|
|
1763
|
-
|
|
1817
|
+
// Remove CSS variables
|
|
1818
|
+
const root = this.document.documentElement;
|
|
1819
|
+
this.renderer.removeStyle(root, '--primary');
|
|
1820
|
+
this.renderer.removeStyle(root, '--radius');
|
|
1821
|
+
// Clear dynamic styles
|
|
1822
|
+
const existingStyle = this.document.getElementById(this.styleId);
|
|
1823
|
+
if (existingStyle) {
|
|
1824
|
+
existingStyle.remove();
|
|
1764
1825
|
}
|
|
1765
1826
|
}
|
|
1766
1827
|
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ThemeService, deps: [{ token: DOCUMENT }, { token: PLATFORM_ID }, { token: TOLLE_CONFIG, optional: true }, { token: i0.RendererFactory2 }], target: i0.ɵɵFactoryTarget.Injectable });
|