UNPKG

@c8y/ngx-components

Version:

Angular modules for Cumulocity IoT applications

83 lines 12.3 kB
import { Injectable } from '@angular/core'; import { BehaviorSubject, fromEvent, of } from 'rxjs'; import { map, startWith, switchMap } from 'rxjs/operators'; import { OptionsService } from '../common/options.service'; import { gettext } from '@c8y/ngx-components/gettext'; import { getThemePreference, applyTheme, setThemePreference } from '@c8y/bootstrap'; import * as i0 from "@angular/core"; import * as i1 from "../common/options.service"; export class ThemeSwitcherService { constructor(options) { this.options = options; this.darkThemeClass = `c8y-dark-theme`; this.themeOptions = [ { label: gettext('Light'), value: 'light', icon: 'sun' }, { label: gettext('Dark'), value: 'dark', icon: 'moon' }, { label: gettext('System'), value: 'system', icon: 'imac-settings' } ]; this._userSelectedThemePreference$ = new BehaviorSubject(this.getCurrentThemePreference()); this._temporaryThemePreference$ = new BehaviorSubject('none'); this.userSelectedThemePreference$ = this._userSelectedThemePreference$.asObservable(); const userSelectedTheme$ = this.userSelectedThemePreference$.pipe(switchMap(preference => { if (preference === 'system') { return this.getUsersSystemPreferenceForTheme$(); } return of(preference); })); this.disableThemeSelection$ = this._temporaryThemePreference$.pipe(map(preference => preference !== 'none')); this.currentlyAppliedTheme$ = this._temporaryThemePreference$.pipe(switchMap(temporaryPreference => { if (temporaryPreference !== 'none') { return of(temporaryPreference); } return userSelectedTheme$; })); this.darkThemeAvailable$ = this.options.get$('darkThemeAvailable').pipe(map(value => !!value)); } getCurrentThemePreference() { const value = getThemePreference(); if (value === 'system' || value === 'dark') { return value; } return 'light'; } getUsersSystemPreferenceForTheme$() { return fromEvent(window.matchMedia('(prefers-color-scheme: dark)'), 'change').pipe(startWith(window.matchMedia('(prefers-color-scheme: dark)')), map((e) => (e.matches ? 'dark' : 'light'))); } changeUserPreference(preference) { setThemePreference(preference); this._userSelectedThemePreference$.next(preference); this.applyTheme(preference); } temporaryChangeTheme(preference) { this._temporaryThemePreference$.next(preference); this.applyTheme(preference); } resetTemporaryTheme() { this._temporaryThemePreference$.next('none'); this.applyTheme(this.getCurrentThemePreference()); } applyTheme(preference) { applyTheme(preference); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ThemeSwitcherService, deps: [{ token: i1.OptionsService }], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ThemeSwitcherService, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ThemeSwitcherService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: () => [{ type: i1.OptionsService }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"theme-switcher.service.js","sourceRoot":"","sources":["../../../../core/ui-settings/theme-switcher.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable, fromEvent, of } from 'rxjs';\nimport { map, startWith, switchMap } from 'rxjs/operators';\nimport { OptionsService } from '../common/options.service';\nimport { gettext } from '@c8y/ngx-components/gettext';\nimport { SupportedIcons } from '@c8y/ngx-components/icon-selector/icons';\nimport { getThemePreference, applyTheme, setThemePreference } from '@c8y/bootstrap';\n\nexport type ThemeOptions = 'light' | 'dark';\nexport type ThemePreferenceOptions = ThemeOptions | 'system';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class ThemeSwitcherService {\n  darkThemeClass = `c8y-dark-theme`;\n  darkThemeAvailable$: Observable<boolean>;\n  userSelectedThemePreference$: Observable<ThemePreferenceOptions>;\n  currentlyAppliedTheme$: Observable<ThemeOptions>;\n  disableThemeSelection$: Observable<boolean>;\n  themeOptions = [\n    {\n      label: gettext('Light'),\n      value: 'light',\n      icon: 'sun'\n    },\n    {\n      label: gettext('Dark'),\n      value: 'dark',\n      icon: 'moon'\n    },\n    {\n      label: gettext('System'),\n      value: 'system',\n      icon: 'imac-settings'\n    }\n  ] as const satisfies {\n    label: string;\n    value: ThemePreferenceOptions;\n    icon: SupportedIcons;\n  }[];\n  private _userSelectedThemePreference$ = new BehaviorSubject<ThemePreferenceOptions>(\n    this.getCurrentThemePreference()\n  );\n  private _temporaryThemePreference$ = new BehaviorSubject<ThemeOptions | 'none'>('none');\n\n  constructor(private options: OptionsService) {\n    this.userSelectedThemePreference$ = this._userSelectedThemePreference$.asObservable();\n    const userSelectedTheme$ = this.userSelectedThemePreference$.pipe(\n      switchMap(preference => {\n        if (preference === 'system') {\n          return this.getUsersSystemPreferenceForTheme$();\n        }\n        return of(preference);\n      })\n    );\n    this.disableThemeSelection$ = this._temporaryThemePreference$.pipe(\n      map(preference => preference !== 'none')\n    );\n    this.currentlyAppliedTheme$ = this._temporaryThemePreference$.pipe(\n      switchMap(temporaryPreference => {\n        if (temporaryPreference !== 'none') {\n          return of(temporaryPreference);\n        }\n        return userSelectedTheme$;\n      })\n    );\n    this.darkThemeAvailable$ = this.options.get$('darkThemeAvailable').pipe(map(value => !!value));\n  }\n\n  getCurrentThemePreference(): 'light' | 'dark' | 'system' {\n    const value = getThemePreference();\n    if (value === 'system' || value === 'dark') {\n      return value;\n    }\n    return 'light';\n  }\n\n  getUsersSystemPreferenceForTheme$() {\n    return fromEvent(window.matchMedia('(prefers-color-scheme: dark)'), 'change').pipe(\n      startWith(window.matchMedia('(prefers-color-scheme: dark)')),\n      map((e: MediaQueryList) => (e.matches ? 'dark' : 'light'))\n    );\n  }\n\n  changeUserPreference(preference: 'light' | 'dark' | 'system') {\n    setThemePreference(preference);\n    this._userSelectedThemePreference$.next(preference);\n    this.applyTheme(preference);\n  }\n\n  temporaryChangeTheme(preference: 'light' | 'dark') {\n    this._temporaryThemePreference$.next(preference);\n    this.applyTheme(preference);\n  }\n\n  resetTemporaryTheme() {\n    this._temporaryThemePreference$.next('none');\n    this.applyTheme(this.getCurrentThemePreference());\n  }\n\n  applyTheme(preference: 'light' | 'dark' | 'system') {\n    applyTheme(preference);\n  }\n}\n"]}