2023-03-23 13:05:14 +00:00
|
|
|
import { config, tenant } from "@goauthentik/common/api/config";
|
2023-07-07 14:23:10 +00:00
|
|
|
import { EVENT_THEME_CHANGE } from "@goauthentik/common/constants";
|
2023-03-23 13:05:14 +00:00
|
|
|
import { UIConfig, uiConfig } from "@goauthentik/common/ui/config";
|
web: Storybook css import fix (#5964)
* web: fix storybook `build` css import issue
This is an incredibly frustrating issue, because Storybook works
in `dev` mode but not in `build` mode, and that's not at all what
you'd expecte from a mature piece of software. Lit uses the native
CSS adoptedStylesheets field, which takes only a constructedStylesheet.
Lit provides a way of generating those, but the imports from
Patternfly (or any `.css` file) are text, and converting those to
stylesheets required a bit of magic.
What this means going forward is that any Storied components will
have to have their CSS wrapped in a way that ensures it is managed
correctly by Lit (well, to be pedantic, by the
shadowDOM.adoptedStylesheets). That wrapper is provided and the
components that need it have been wrapped.
This problem deserves further investigation, but for the time
being this actually does solve it with a minimum amount of surgical
pain.
* web: fix storybook build issue
This commit further fixes the typing issues around strings, CSSResults,
and CSSStyleSheets by providing overloaded functions that assist
consumers in knowing that if they send an array to expect an array
in return, and if they send a scalar expect a scalar in return.
* replace any with unknown
Signed-off-by: Jens Langhammer <jens@goauthentik.io>
---------
Signed-off-by: Jens Langhammer <jens@goauthentik.io>
Co-authored-by: Jens Langhammer <jens@goauthentik.io>
2023-06-16 11:36:04 +00:00
|
|
|
import { adaptCSS } from "@goauthentik/common/utils";
|
2022-09-14 22:05:21 +00:00
|
|
|
|
2023-07-07 14:23:10 +00:00
|
|
|
import { localized } from "@lit/localize";
|
2023-08-03 15:27:58 +00:00
|
|
|
import { CSSResult, LitElement } from "lit";
|
2023-03-17 22:10:19 +00:00
|
|
|
import { state } from "lit/decorators.js";
|
2022-09-14 22:05:21 +00:00
|
|
|
|
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";
|
2023-03-17 22:10:19 +00:00
|
|
|
import PFBase from "@patternfly/patternfly/patternfly-base.css";
|
2023-03-09 22:17:53 +00:00
|
|
|
|
2023-03-23 13:05:14 +00:00
|
|
|
import { Config, CurrentTenant, UiThemeEnum } from "@goauthentik/api";
|
2023-03-09 22:17:53 +00:00
|
|
|
|
2023-10-20 21:26:57 +00:00
|
|
|
type AkInterface = HTMLElement & {
|
|
|
|
getTheme: () => Promise<UiThemeEnum>;
|
|
|
|
tenant?: CurrentTenant;
|
|
|
|
uiConfig?: UIConfig;
|
|
|
|
config?: Config;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const rootInterface = <T extends AkInterface>(): T | undefined =>
|
|
|
|
(document.body.querySelector("[data-ak-interface-root]") as T) ?? undefined;
|
2023-03-09 22:17:53 +00:00
|
|
|
|
2023-08-03 15:27:58 +00:00
|
|
|
export function ensureCSSStyleSheet(css: CSSStyleSheet | CSSResult): CSSStyleSheet {
|
|
|
|
if (css instanceof CSSResult) {
|
|
|
|
return css.styleSheet!;
|
|
|
|
}
|
|
|
|
return css;
|
|
|
|
}
|
|
|
|
|
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)";
|
|
|
|
|
2023-07-07 14:23:10 +00:00
|
|
|
@localized()
|
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();
|
2023-03-09 22:17:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
protected createRenderRoot(): ShadowRoot | Element {
|
|
|
|
const root = super.createRenderRoot() as ShadowRoot;
|
2023-03-12 21:19:03 +00:00
|
|
|
let styleRoot: AdoptedStyleSheetsElement = root;
|
|
|
|
if ("ShadyDOM" in window) {
|
|
|
|
styleRoot = document;
|
|
|
|
}
|
2023-08-03 15:27:58 +00:00
|
|
|
styleRoot.adoptedStyleSheets = adaptCSS([
|
|
|
|
...styleRoot.adoptedStyleSheets,
|
|
|
|
ensureCSSStyleSheet(AKGlobal),
|
|
|
|
]);
|
2023-03-12 21:19:03 +00:00
|
|
|
this._initTheme(styleRoot);
|
|
|
|
this._initCustomCSS(styleRoot);
|
2023-03-09 22:17:53 +00:00
|
|
|
return root;
|
|
|
|
}
|
|
|
|
|
2023-03-10 16:33:03 +00:00
|
|
|
async getTheme(): Promise<UiThemeEnum> {
|
|
|
|
return rootInterface()?.getTheme() || UiThemeEnum.Automatic;
|
|
|
|
}
|
|
|
|
|
2023-03-09 22:17:53 +00:00
|
|
|
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,
|
|
|
|
);
|
2023-03-10 16:33:03 +00:00
|
|
|
this._applyTheme(root, await this.getTheme());
|
2023-03-09 22:17:53 +00:00
|
|
|
}
|
|
|
|
|
2023-03-12 21:19:03 +00:00
|
|
|
private async _initCustomCSS(root: AdoptedStyleSheetsElement): Promise<void> {
|
2023-03-09 22:17:53 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2023-03-10 16:33:03 +00:00
|
|
|
static themeToStylesheet(theme?: UiThemeEnum): CSSStyleSheet | undefined {
|
|
|
|
if (theme === UiThemeEnum.Dark) {
|
|
|
|
return ThemeDark;
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2023-03-09 22:17:53 +00:00
|
|
|
_activateTheme(root: AdoptedStyleSheetsElement, theme: UiThemeEnum) {
|
2023-03-10 16:33:03 +00:00
|
|
|
if (theme === this._activeTheme) {
|
2023-03-09 22:17:53 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Make sure we only get to this callback once we've picked a concise theme choice
|
|
|
|
this.dispatchEvent(
|
2023-03-10 16:33:03 +00:00
|
|
|
new CustomEvent(EVENT_THEME_CHANGE, {
|
2023-03-09 22:17:53 +00:00
|
|
|
bubbles: true,
|
|
|
|
composed: true,
|
|
|
|
detail: theme,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
this.setAttribute("theme", theme);
|
2023-03-10 16:33:03 +00:00
|
|
|
const stylesheet = AKElement.themeToStylesheet(theme);
|
|
|
|
const oldStylesheet = AKElement.themeToStylesheet(this._activeTheme);
|
|
|
|
if (stylesheet) {
|
2023-08-03 15:27:58 +00:00
|
|
|
root.adoptedStyleSheets = [...root.adoptedStyleSheets, ensureCSSStyleSheet(stylesheet)];
|
2023-03-09 22:17:53 +00:00
|
|
|
}
|
2023-03-10 16:33:03 +00:00
|
|
|
if (oldStylesheet) {
|
|
|
|
root.adoptedStyleSheets = root.adoptedStyleSheets.filter((v) => v !== oldStylesheet);
|
|
|
|
}
|
|
|
|
this._activeTheme = theme;
|
2023-06-13 13:41:48 +00:00
|
|
|
this.requestUpdate();
|
2023-03-09 22:17:53 +00:00
|
|
|
}
|
2022-09-14 22:05:21 +00:00
|
|
|
}
|
2023-03-09 22:17:53 +00:00
|
|
|
|
2023-10-20 21:26:57 +00:00
|
|
|
export class Interface extends AKElement implements AkInterface {
|
2023-03-17 22:10:19 +00:00
|
|
|
@state()
|
|
|
|
tenant?: CurrentTenant;
|
|
|
|
|
2023-03-23 13:05:14 +00:00
|
|
|
@state()
|
|
|
|
uiConfig?: UIConfig;
|
|
|
|
|
|
|
|
@state()
|
|
|
|
config?: Config;
|
|
|
|
|
2023-03-17 22:10:19 +00:00
|
|
|
constructor() {
|
|
|
|
super();
|
2023-08-03 15:27:58 +00:00
|
|
|
document.adoptedStyleSheets = [...document.adoptedStyleSheets, ensureCSSStyleSheet(PFBase)];
|
2023-03-17 22:10:19 +00:00
|
|
|
tenant().then((tenant) => (this.tenant = tenant));
|
2023-03-23 13:05:14 +00:00
|
|
|
config().then((config) => (this.config = config));
|
2023-10-20 21:26:57 +00:00
|
|
|
this.dataset.akInterfaceRoot = "true";
|
2023-03-17 22:10:19 +00:00
|
|
|
}
|
|
|
|
|
2023-03-09 22:17:53 +00:00
|
|
|
_activateTheme(root: AdoptedStyleSheetsElement, theme: UiThemeEnum): void {
|
|
|
|
super._activateTheme(root, theme);
|
|
|
|
super._activateTheme(document, theme);
|
|
|
|
}
|
|
|
|
|
2023-03-10 16:33:03 +00:00
|
|
|
async getTheme(): Promise<UiThemeEnum> {
|
2023-03-23 13:05:14 +00:00
|
|
|
if (!this.uiConfig) {
|
|
|
|
this.uiConfig = await uiConfig();
|
|
|
|
}
|
|
|
|
return this.uiConfig.theme?.base || UiThemeEnum.Automatic;
|
2023-03-09 22:17:53 +00:00
|
|
|
}
|
|
|
|
}
|