import { EVENT_LOCALE_CHANGE } from "@goauthentik/common/constants"; import { globalAK } from "@goauthentik/common/global"; import { uiConfig } from "@goauthentik/common/ui/config"; import { LitElement } from "lit"; import AKGlobal from "@goauthentik/common/styles/authentik.css"; import ThemeDark from "@goauthentik/common/styles/theme-dark.css"; import { UiThemeEnum } from "@goauthentik/api"; export function rootInterface(): Interface | undefined { const el = Array.from(document.body.querySelectorAll("*")).filter( (el) => el instanceof Interface, ); return el[0] as Interface; } let css: Promise | undefined; function fetchCustomCSS(): Promise { if (!css) { css = Promise.all( Array.of(...document.head.querySelectorAll("link[data-inject]")).map( (link) => { return fetch(link.href) .then((res) => { return res.text(); }) .finally(() => { return ""; }); }, ), ); } return css; } export interface AdoptedStyleSheetsElement { adoptedStyleSheets: readonly CSSStyleSheet[]; } const QUERY_MEDIA_COLOR_LIGHT = "(prefers-color-scheme: light)"; export class AKElement extends LitElement { _mediaMatcher?: MediaQueryList; _mediaMatcherHandler?: (ev?: MediaQueryListEvent) => void; _activeTheme?: UiThemeEnum; get activeTheme(): UiThemeEnum | undefined { return this._activeTheme; } private _handleLocaleChange: () => void; constructor() { super(); this._handleLocaleChange = (() => { this.requestUpdate(); }).bind(this); window.addEventListener(EVENT_LOCALE_CHANGE, this._handleLocaleChange); } protected createRenderRoot(): ShadowRoot | Element { const root = super.createRenderRoot() as ShadowRoot; root.adoptedStyleSheets = [...root.adoptedStyleSheets, AKGlobal]; this._initTheme(root); this._initCustomCSS(root); return root; } async _initTheme(root: AdoptedStyleSheetsElement): Promise { // Early activate theme based on media query to prevent light flash // when dark is preferred this._activateTheme( root, window.matchMedia(QUERY_MEDIA_COLOR_LIGHT).matches ? UiThemeEnum.Light : UiThemeEnum.Dark, ); rootInterface()?._initTheme(root); } private async _initCustomCSS(root: ShadowRoot): Promise { const sheets = await fetchCustomCSS(); sheets.map((css) => { if (css === "") { return; } new CSSStyleSheet().replace(css).then((sheet) => { root.adoptedStyleSheets = [...root.adoptedStyleSheets, sheet]; }); }); } _applyTheme(root: AdoptedStyleSheetsElement, theme?: UiThemeEnum): void { if (!theme) { theme = UiThemeEnum.Automatic; } if (theme === UiThemeEnum.Automatic) { // Create a media matcher to automatically switch the theme depending on // prefers-color-scheme if (!this._mediaMatcher) { this._mediaMatcher = window.matchMedia(QUERY_MEDIA_COLOR_LIGHT); this._mediaMatcherHandler = (ev?: MediaQueryListEvent) => { const theme = ev?.matches || this._mediaMatcher?.matches ? UiThemeEnum.Light : UiThemeEnum.Dark; this._activateTheme(root, theme); }; this._mediaMatcher.addEventListener("change", this._mediaMatcherHandler); } return; } else if (this._mediaMatcher && this._mediaMatcherHandler) { // Theme isn't automatic and we have a matcher configured, remove the matcher // to prevent changes this._mediaMatcher.removeEventListener("change", this._mediaMatcherHandler); this._mediaMatcher = undefined; } this._activateTheme(root, theme); } _activateTheme(root: AdoptedStyleSheetsElement, theme: UiThemeEnum) { if (this._activeTheme === theme) { return; } // Make sure we only get to this callback once we've picked a concise theme choice this.dispatchEvent( new CustomEvent("themeChange", { bubbles: true, composed: true, detail: theme, }), ); this._activeTheme = theme; this.setAttribute("theme", theme); let stylesheet: CSSStyleSheet | undefined; if (theme === UiThemeEnum.Dark) { stylesheet = ThemeDark; } if (!stylesheet) { return; } if (root.adoptedStyleSheets.indexOf(stylesheet) === -1) { root.adoptedStyleSheets = [...root.adoptedStyleSheets, stylesheet]; } else { root.adoptedStyleSheets = root.adoptedStyleSheets.filter((v) => v !== stylesheet); } this.requestUpdate(); } disconnectedCallback() { super.disconnectedCallback(); window.removeEventListener(EVENT_LOCALE_CHANGE, this._handleLocaleChange); } } export class Interface extends AKElement { _activateTheme(root: AdoptedStyleSheetsElement, theme: UiThemeEnum): void { super._activateTheme(root, theme); super._activateTheme(document, theme); } async _initTheme(root: AdoptedStyleSheetsElement): Promise { const bootstrapTheme = globalAK()?.tenant.uiTheme || UiThemeEnum.Automatic; this._applyTheme(root, bootstrapTheme); uiConfig().then((config) => { if (config.theme.base) { this._applyTheme(root, config.theme.base); } }); } }