2022-09-14 22:05:21 +00:00
|
|
|
import { EVENT_LOCALE_CHANGE } from "@goauthentik/common/constants";
|
2023-03-09 22:17:53 +00:00
|
|
|
import { globalAK } from "@goauthentik/common/global";
|
|
|
|
import { uiConfig } from "@goauthentik/common/ui/config";
|
2022-09-14 22:05:21 +00:00
|
|
|
|
|
|
|
import { LitElement } from "lit";
|
|
|
|
|
2023-03-09 22:17:53 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2023-02-27 18:54:19 +00:00
|
|
|
let css: Promise<string[]> | undefined;
|
|
|
|
function fetchCustomCSS(): Promise<string[]> {
|
|
|
|
if (!css) {
|
|
|
|
css = Promise.all(
|
|
|
|
Array.of(...document.head.querySelectorAll<HTMLLinkElement>("link[data-inject]")).map(
|
|
|
|
(link) => {
|
|
|
|
return fetch(link.href)
|
|
|
|
.then((res) => {
|
|
|
|
return res.text();
|
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
return "";
|
|
|
|
});
|
|
|
|
},
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return css;
|
|
|
|
}
|
|
|
|
|
2023-03-09 22:17:53 +00:00
|
|
|
export interface AdoptedStyleSheetsElement {
|
|
|
|
adoptedStyleSheets: readonly CSSStyleSheet[];
|
|
|
|
}
|
|
|
|
|
|
|
|
const QUERY_MEDIA_COLOR_LIGHT = "(prefers-color-scheme: light)";
|
|
|
|
|
2022-09-14 22:05:21 +00:00
|
|
|
export class AKElement extends LitElement {
|
2023-03-09 22:17:53 +00:00
|
|
|
_mediaMatcher?: MediaQueryList;
|
|
|
|
_mediaMatcherHandler?: (ev?: MediaQueryListEvent) => void;
|
|
|
|
_activeTheme?: UiThemeEnum;
|
|
|
|
|
|
|
|
get activeTheme(): UiThemeEnum | undefined {
|
|
|
|
return this._activeTheme;
|
|
|
|
}
|
|
|
|
|
2022-09-14 22:05:21 +00:00
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.addEventListener(EVENT_LOCALE_CHANGE, this._handleLocaleChange);
|
2023-03-09 22:17:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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<void> {
|
|
|
|
// 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<void> {
|
|
|
|
const sheets = await fetchCustomCSS();
|
|
|
|
sheets.map((css) => {
|
|
|
|
if (css === "") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
new CSSStyleSheet().replace(css).then((sheet) => {
|
|
|
|
root.adoptedStyleSheets = [...root.adoptedStyleSheets, sheet];
|
2023-02-27 18:54:19 +00:00
|
|
|
});
|
|
|
|
});
|
2022-09-14 22:05:21 +00:00
|
|
|
}
|
|
|
|
|
2023-03-09 22:17:53 +00:00
|
|
|
_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();
|
|
|
|
}
|
|
|
|
|
2022-09-14 22:05:21 +00:00
|
|
|
disconnectedCallback() {
|
|
|
|
super.disconnectedCallback();
|
|
|
|
this.removeEventListener(EVENT_LOCALE_CHANGE, this._handleLocaleChange);
|
|
|
|
}
|
|
|
|
|
|
|
|
private _handleLocaleChange() {
|
|
|
|
this.requestUpdate();
|
|
|
|
}
|
|
|
|
}
|
2023-03-09 22:17:53 +00:00
|
|
|
|
|
|
|
export class Interface extends AKElement {
|
|
|
|
_activateTheme(root: AdoptedStyleSheetsElement, theme: UiThemeEnum): void {
|
|
|
|
super._activateTheme(root, theme);
|
|
|
|
super._activateTheme(document, theme);
|
|
|
|
}
|
|
|
|
|
|
|
|
async _initTheme(root: AdoptedStyleSheetsElement): Promise<void> {
|
|
|
|
const bootstrapTheme = globalAK()?.tenant.uiTheme || UiThemeEnum.Automatic;
|
|
|
|
this._applyTheme(root, bootstrapTheme);
|
|
|
|
uiConfig().then((config) => {
|
|
|
|
if (config.theme.base) {
|
|
|
|
this._applyTheme(root, config.theme.base);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|