@tolle_/tolle-ui 0.0.21-beta → 0.0.23-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 +87 -34
- package/fesm2022/tolle-ui.mjs +86 -33
- 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,51 @@ 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
|
-
|
|
42
|
-
|
|
60
|
+
// 3. Apply Radius
|
|
61
|
+
// Priority: Saved Radius > Config Radius
|
|
62
|
+
if (savedRadius) {
|
|
63
|
+
// User has saved radius preference
|
|
64
|
+
this.setRadius(savedRadius, false);
|
|
43
65
|
}
|
|
44
|
-
//
|
|
45
|
-
if (
|
|
46
|
-
|
|
66
|
+
// Save theme mode preference if it came from config or system
|
|
67
|
+
if (!savedTheme) {
|
|
68
|
+
localStorage.setItem('tolle-theme', shouldBeDark ? 'dark' : 'light');
|
|
47
69
|
}
|
|
48
70
|
}
|
|
49
71
|
/**
|
|
@@ -249,14 +271,18 @@ export class ThemeService {
|
|
|
249
271
|
const isCurrentlyDark = this.document.documentElement.classList.contains('dark');
|
|
250
272
|
isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();
|
|
251
273
|
}
|
|
252
|
-
enableDarkMode() {
|
|
274
|
+
enableDarkMode(saveToStorage = true) {
|
|
253
275
|
this.renderer.addClass(this.document.documentElement, 'dark');
|
|
254
|
-
|
|
276
|
+
if (saveToStorage) {
|
|
277
|
+
localStorage.setItem('tolle-theme', 'dark');
|
|
278
|
+
}
|
|
255
279
|
this.isDarkSubject.next(true);
|
|
256
280
|
}
|
|
257
|
-
disableDarkMode() {
|
|
281
|
+
disableDarkMode(saveToStorage = true) {
|
|
258
282
|
this.renderer.removeClass(this.document.documentElement, 'dark');
|
|
259
|
-
|
|
283
|
+
if (saveToStorage) {
|
|
284
|
+
localStorage.setItem('tolle-theme', 'light');
|
|
285
|
+
}
|
|
260
286
|
this.isDarkSubject.next(false);
|
|
261
287
|
}
|
|
262
288
|
setPrimaryColor(color, persist = true) {
|
|
@@ -289,29 +315,56 @@ export class ThemeService {
|
|
|
289
315
|
}
|
|
290
316
|
return localStorage.getItem('tolle-primary-color');
|
|
291
317
|
}
|
|
292
|
-
|
|
318
|
+
/**
|
|
319
|
+
* Reset to config defaults (clears user preferences)
|
|
320
|
+
*/
|
|
321
|
+
resetToConfigDefaults() {
|
|
322
|
+
if (!isPlatformBrowser(this.platformId))
|
|
323
|
+
return;
|
|
324
|
+
// Clear user preferences
|
|
325
|
+
localStorage.removeItem('tolle-theme');
|
|
326
|
+
localStorage.removeItem('tolle-primary-color');
|
|
327
|
+
localStorage.removeItem('tolle-radius');
|
|
328
|
+
// Re-initialize with config defaults
|
|
329
|
+
this.initializeTheme();
|
|
330
|
+
}
|
|
331
|
+
/**
|
|
332
|
+
* Get current user preferences
|
|
333
|
+
*/
|
|
334
|
+
getUserPreferences() {
|
|
293
335
|
if (!isPlatformBrowser(this.platformId))
|
|
294
336
|
return null;
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
}
|
|
300
|
-
return localStorage.getItem('tolle-radius') || '0.5rem';
|
|
337
|
+
return {
|
|
338
|
+
theme: localStorage.getItem('tolle-theme'),
|
|
339
|
+
primaryColor: localStorage.getItem('tolle-primary-color'),
|
|
340
|
+
radius: localStorage.getItem('tolle-radius')
|
|
341
|
+
};
|
|
301
342
|
}
|
|
302
343
|
/**
|
|
303
|
-
*
|
|
344
|
+
* Clear all user preferences
|
|
304
345
|
*/
|
|
305
|
-
|
|
346
|
+
clearUserPreferences() {
|
|
306
347
|
if (!isPlatformBrowser(this.platformId))
|
|
307
348
|
return;
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
349
|
+
localStorage.removeItem('tolle-theme');
|
|
350
|
+
localStorage.removeItem('tolle-primary-color');
|
|
351
|
+
localStorage.removeItem('tolle-radius');
|
|
352
|
+
// Reset to system defaults (not config defaults)
|
|
353
|
+
const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
354
|
+
if (systemPrefersDark) {
|
|
355
|
+
this.enableDarkMode(false);
|
|
311
356
|
}
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
357
|
+
else {
|
|
358
|
+
this.disableDarkMode(false);
|
|
359
|
+
}
|
|
360
|
+
// Remove CSS variables
|
|
361
|
+
const root = this.document.documentElement;
|
|
362
|
+
this.renderer.removeStyle(root, '--primary');
|
|
363
|
+
this.renderer.removeStyle(root, '--radius');
|
|
364
|
+
// Clear dynamic styles
|
|
365
|
+
const existingStyle = this.document.getElementById(this.styleId);
|
|
366
|
+
if (existingStyle) {
|
|
367
|
+
existingStyle.remove();
|
|
315
368
|
}
|
|
316
369
|
}
|
|
317
370
|
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 +385,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
332
385
|
type: Inject,
|
|
333
386
|
args: [TOLLE_CONFIG]
|
|
334
387
|
}] }, { 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"]}
|
|
388
|
+
//# 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    }\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    }\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,51 @@ 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
|
-
|
|
1491
|
-
|
|
1509
|
+
// 3. Apply Radius
|
|
1510
|
+
// Priority: Saved Radius > Config Radius
|
|
1511
|
+
if (savedRadius) {
|
|
1512
|
+
// User has saved radius preference
|
|
1513
|
+
this.setRadius(savedRadius, false);
|
|
1492
1514
|
}
|
|
1493
|
-
//
|
|
1494
|
-
if (
|
|
1495
|
-
|
|
1515
|
+
// Save theme mode preference if it came from config or system
|
|
1516
|
+
if (!savedTheme) {
|
|
1517
|
+
localStorage.setItem('tolle-theme', shouldBeDark ? 'dark' : 'light');
|
|
1496
1518
|
}
|
|
1497
1519
|
}
|
|
1498
1520
|
/**
|
|
@@ -1698,14 +1720,18 @@ class ThemeService {
|
|
|
1698
1720
|
const isCurrentlyDark = this.document.documentElement.classList.contains('dark');
|
|
1699
1721
|
isCurrentlyDark ? this.disableDarkMode() : this.enableDarkMode();
|
|
1700
1722
|
}
|
|
1701
|
-
enableDarkMode() {
|
|
1723
|
+
enableDarkMode(saveToStorage = true) {
|
|
1702
1724
|
this.renderer.addClass(this.document.documentElement, 'dark');
|
|
1703
|
-
|
|
1725
|
+
if (saveToStorage) {
|
|
1726
|
+
localStorage.setItem('tolle-theme', 'dark');
|
|
1727
|
+
}
|
|
1704
1728
|
this.isDarkSubject.next(true);
|
|
1705
1729
|
}
|
|
1706
|
-
disableDarkMode() {
|
|
1730
|
+
disableDarkMode(saveToStorage = true) {
|
|
1707
1731
|
this.renderer.removeClass(this.document.documentElement, 'dark');
|
|
1708
|
-
|
|
1732
|
+
if (saveToStorage) {
|
|
1733
|
+
localStorage.setItem('tolle-theme', 'light');
|
|
1734
|
+
}
|
|
1709
1735
|
this.isDarkSubject.next(false);
|
|
1710
1736
|
}
|
|
1711
1737
|
setPrimaryColor(color, persist = true) {
|
|
@@ -1738,29 +1764,56 @@ class ThemeService {
|
|
|
1738
1764
|
}
|
|
1739
1765
|
return localStorage.getItem('tolle-primary-color');
|
|
1740
1766
|
}
|
|
1741
|
-
|
|
1767
|
+
/**
|
|
1768
|
+
* Reset to config defaults (clears user preferences)
|
|
1769
|
+
*/
|
|
1770
|
+
resetToConfigDefaults() {
|
|
1771
|
+
if (!isPlatformBrowser(this.platformId))
|
|
1772
|
+
return;
|
|
1773
|
+
// Clear user preferences
|
|
1774
|
+
localStorage.removeItem('tolle-theme');
|
|
1775
|
+
localStorage.removeItem('tolle-primary-color');
|
|
1776
|
+
localStorage.removeItem('tolle-radius');
|
|
1777
|
+
// Re-initialize with config defaults
|
|
1778
|
+
this.initializeTheme();
|
|
1779
|
+
}
|
|
1780
|
+
/**
|
|
1781
|
+
* Get current user preferences
|
|
1782
|
+
*/
|
|
1783
|
+
getUserPreferences() {
|
|
1742
1784
|
if (!isPlatformBrowser(this.platformId))
|
|
1743
1785
|
return null;
|
|
1744
|
-
|
|
1745
|
-
|
|
1746
|
-
|
|
1747
|
-
|
|
1748
|
-
}
|
|
1749
|
-
return localStorage.getItem('tolle-radius') || '0.5rem';
|
|
1786
|
+
return {
|
|
1787
|
+
theme: localStorage.getItem('tolle-theme'),
|
|
1788
|
+
primaryColor: localStorage.getItem('tolle-primary-color'),
|
|
1789
|
+
radius: localStorage.getItem('tolle-radius')
|
|
1790
|
+
};
|
|
1750
1791
|
}
|
|
1751
1792
|
/**
|
|
1752
|
-
*
|
|
1793
|
+
* Clear all user preferences
|
|
1753
1794
|
*/
|
|
1754
|
-
|
|
1795
|
+
clearUserPreferences() {
|
|
1755
1796
|
if (!isPlatformBrowser(this.platformId))
|
|
1756
1797
|
return;
|
|
1757
|
-
|
|
1758
|
-
|
|
1759
|
-
|
|
1798
|
+
localStorage.removeItem('tolle-theme');
|
|
1799
|
+
localStorage.removeItem('tolle-primary-color');
|
|
1800
|
+
localStorage.removeItem('tolle-radius');
|
|
1801
|
+
// Reset to system defaults (not config defaults)
|
|
1802
|
+
const systemPrefersDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
|
|
1803
|
+
if (systemPrefersDark) {
|
|
1804
|
+
this.enableDarkMode(false);
|
|
1760
1805
|
}
|
|
1761
|
-
|
|
1762
|
-
|
|
1763
|
-
|
|
1806
|
+
else {
|
|
1807
|
+
this.disableDarkMode(false);
|
|
1808
|
+
}
|
|
1809
|
+
// Remove CSS variables
|
|
1810
|
+
const root = this.document.documentElement;
|
|
1811
|
+
this.renderer.removeStyle(root, '--primary');
|
|
1812
|
+
this.renderer.removeStyle(root, '--radius');
|
|
1813
|
+
// Clear dynamic styles
|
|
1814
|
+
const existingStyle = this.document.getElementById(this.styleId);
|
|
1815
|
+
if (existingStyle) {
|
|
1816
|
+
existingStyle.remove();
|
|
1764
1817
|
}
|
|
1765
1818
|
}
|
|
1766
1819
|
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 });
|