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";
|
web: refactor sidebar capabilities for categorical subsections (#7482)
* web: break circular dependency between AKElement & Interface.
This commit changes the way the root node of the web application shell is
discovered by child components, such that the base class shared by both
no longer results in a circular dependency between the two models.
I've run this in isolation and have seen no failures of discovery; the identity
token exists as soon as the Interface is constructed and is found by every item
on the page.
* web: fix broken typescript references
This built... and then it didn't? Anyway, the current fix is to
provide type information the AkInterface for the data that consumers
require.
* web: rollback dependabot's upgrade of context
The most frustrating part of this is that I RAN THIS, dammit, with the updated
context and the current Wizard, and it finished the End-to-End tests without
complaint.
* Due for amendment
* Revert "Due for amendment"
This reverts commit 829ad5d3f214fa163958593636b28300d010da42.
* web: refactor sidebar capabilities for categorical subsections
The project "Change Admin UI lists to have sublists per type" requires some initial changes to the
UI to facilitate this request. The AdminSidebar is the principle target of this project, and it is
embedded in the AdminInterface. To facilitate editing the AdminSidebar as an independent entity,
AdminInterface has been moved into its own folder and the AdminSidebar extracted as a standalone Web
Component. This removes, oh, about half the code from AdminInterface. A little cleanup with
`classMap` was also committed.
The rollup config was adjusted to find the new AdminInterface location.
The Sidebar uses the global `config: Config` object to check for Enterprise capabilities. Rather
than plumb all the way down through the Interface => AdminInterface -> AdminSidebar, I chose to make
provide an alternative way of reaching the `config` object, as a *context*. Other configuration
objects (Me, UiConfig, Tenant) interfaces will be contextualized as demand warrants.
Demand will warrant. Just not yet. <sup>1</sup>
The Sidebar has been refactored only slightly; the renderers are entirely the same as they were
prior to extraction. What has been changed is the source of information: when we retrieve the
current version we story *only* the information, and use type information to ensure that the version
we store is the version we care about. The same is true of `impersonation`; we care only about the
name of the person being impersonated being present, so we don't store anything else.
Fetches have been moved from `firstUpdated` to the constructor. No reason to have the sidebar
render twice if the network returns before the render is scheduled.
Because the path used to identify the user being impersonated has changed, the `str()` references in
the XLIFF files had to be adjusted. **This change is to a variable only and does not require
translation.**
---
<sup>1</sup> The code is littered with checks to `me()?`, `uiConfig?`, `config?`, etc. In the
*context* of being logged in as an administrator those should never be in doubt. I intend to make
our interfaces not have any doubt.
* Function to help generate sizing solutions across Javascript and CSS.
* web: refactor sidebar capabilities for categorical subsections
Move open/close logic into the ak-admin-sidebar itself.
This commit removes the responsibility for opening/closing the sidebar from the interface parent
code and places it inside the sidebar entirely. Since the Django invocation passes none of the
properties ak-interface-admin is capable of receiving, this seems like a safe operation.
The sidebar now assumes the responsibility for hooking up the window event listeners for open/close
and resize.
On connection to the DOM, and on resize, the sidebar checks to see if the viewport width meets the
criteria for a behavioral change (slide-overlay vs slide-push), and on slide-push automatically
opens the sidebar on the assumption that there's plenty of room. In order to support more dynamic
styling going forward, I've substituted the 1280px with 80rem, which is the same, but allows for
some better styling if someone with older eyes needs to "zoom in" on the whole thing with a larger
font size.
The hide/show code involves "reaching up" to touch the host's classList. There's a comment
indicating that this is a slightly fragile thing to do, but in a well-known way.
2023-11-20 18:24:59 +00:00
|
|
|
import { authentikConfigContext } from "@goauthentik/elements/AuthentikContexts";
|
2022-09-14 22:05:21 +00:00
|
|
|
|
web: refactor sidebar capabilities for categorical subsections (#7482)
* web: break circular dependency between AKElement & Interface.
This commit changes the way the root node of the web application shell is
discovered by child components, such that the base class shared by both
no longer results in a circular dependency between the two models.
I've run this in isolation and have seen no failures of discovery; the identity
token exists as soon as the Interface is constructed and is found by every item
on the page.
* web: fix broken typescript references
This built... and then it didn't? Anyway, the current fix is to
provide type information the AkInterface for the data that consumers
require.
* web: rollback dependabot's upgrade of context
The most frustrating part of this is that I RAN THIS, dammit, with the updated
context and the current Wizard, and it finished the End-to-End tests without
complaint.
* Due for amendment
* Revert "Due for amendment"
This reverts commit 829ad5d3f214fa163958593636b28300d010da42.
* web: refactor sidebar capabilities for categorical subsections
The project "Change Admin UI lists to have sublists per type" requires some initial changes to the
UI to facilitate this request. The AdminSidebar is the principle target of this project, and it is
embedded in the AdminInterface. To facilitate editing the AdminSidebar as an independent entity,
AdminInterface has been moved into its own folder and the AdminSidebar extracted as a standalone Web
Component. This removes, oh, about half the code from AdminInterface. A little cleanup with
`classMap` was also committed.
The rollup config was adjusted to find the new AdminInterface location.
The Sidebar uses the global `config: Config` object to check for Enterprise capabilities. Rather
than plumb all the way down through the Interface => AdminInterface -> AdminSidebar, I chose to make
provide an alternative way of reaching the `config` object, as a *context*. Other configuration
objects (Me, UiConfig, Tenant) interfaces will be contextualized as demand warrants.
Demand will warrant. Just not yet. <sup>1</sup>
The Sidebar has been refactored only slightly; the renderers are entirely the same as they were
prior to extraction. What has been changed is the source of information: when we retrieve the
current version we story *only* the information, and use type information to ensure that the version
we store is the version we care about. The same is true of `impersonation`; we care only about the
name of the person being impersonated being present, so we don't store anything else.
Fetches have been moved from `firstUpdated` to the constructor. No reason to have the sidebar
render twice if the network returns before the render is scheduled.
Because the path used to identify the user being impersonated has changed, the `str()` references in
the XLIFF files had to be adjusted. **This change is to a variable only and does not require
translation.**
---
<sup>1</sup> The code is littered with checks to `me()?`, `uiConfig?`, `config?`, etc. In the
*context* of being logged in as an administrator those should never be in doubt. I intend to make
our interfaces not have any doubt.
* Function to help generate sizing solutions across Javascript and CSS.
* web: refactor sidebar capabilities for categorical subsections
Move open/close logic into the ak-admin-sidebar itself.
This commit removes the responsibility for opening/closing the sidebar from the interface parent
code and places it inside the sidebar entirely. Since the Django invocation passes none of the
properties ak-interface-admin is capable of receiving, this seems like a safe operation.
The sidebar now assumes the responsibility for hooking up the window event listeners for open/close
and resize.
On connection to the DOM, and on resize, the sidebar checks to see if the viewport width meets the
criteria for a behavioral change (slide-overlay vs slide-push), and on slide-push automatically
opens the sidebar on the assumption that there's plenty of room. In order to support more dynamic
styling going forward, I've substituted the 1280px with 80rem, which is the same, but allows for
some better styling if someone with older eyes needs to "zoom in" on the whole thing with a larger
font size.
The hide/show code involves "reaching up" to touch the host's classList. There's a comment
indicating that this is a slightly fragile thing to do, but in a well-known way.
2023-11-20 18:24:59 +00:00
|
|
|
import { ContextProvider } from "@lit-labs/context";
|
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;
|
|
|
|
|
web: refactor sidebar capabilities for categorical subsections (#7482)
* web: break circular dependency between AKElement & Interface.
This commit changes the way the root node of the web application shell is
discovered by child components, such that the base class shared by both
no longer results in a circular dependency between the two models.
I've run this in isolation and have seen no failures of discovery; the identity
token exists as soon as the Interface is constructed and is found by every item
on the page.
* web: fix broken typescript references
This built... and then it didn't? Anyway, the current fix is to
provide type information the AkInterface for the data that consumers
require.
* web: rollback dependabot's upgrade of context
The most frustrating part of this is that I RAN THIS, dammit, with the updated
context and the current Wizard, and it finished the End-to-End tests without
complaint.
* Due for amendment
* Revert "Due for amendment"
This reverts commit 829ad5d3f214fa163958593636b28300d010da42.
* web: refactor sidebar capabilities for categorical subsections
The project "Change Admin UI lists to have sublists per type" requires some initial changes to the
UI to facilitate this request. The AdminSidebar is the principle target of this project, and it is
embedded in the AdminInterface. To facilitate editing the AdminSidebar as an independent entity,
AdminInterface has been moved into its own folder and the AdminSidebar extracted as a standalone Web
Component. This removes, oh, about half the code from AdminInterface. A little cleanup with
`classMap` was also committed.
The rollup config was adjusted to find the new AdminInterface location.
The Sidebar uses the global `config: Config` object to check for Enterprise capabilities. Rather
than plumb all the way down through the Interface => AdminInterface -> AdminSidebar, I chose to make
provide an alternative way of reaching the `config` object, as a *context*. Other configuration
objects (Me, UiConfig, Tenant) interfaces will be contextualized as demand warrants.
Demand will warrant. Just not yet. <sup>1</sup>
The Sidebar has been refactored only slightly; the renderers are entirely the same as they were
prior to extraction. What has been changed is the source of information: when we retrieve the
current version we story *only* the information, and use type information to ensure that the version
we store is the version we care about. The same is true of `impersonation`; we care only about the
name of the person being impersonated being present, so we don't store anything else.
Fetches have been moved from `firstUpdated` to the constructor. No reason to have the sidebar
render twice if the network returns before the render is scheduled.
Because the path used to identify the user being impersonated has changed, the `str()` references in
the XLIFF files had to be adjusted. **This change is to a variable only and does not require
translation.**
---
<sup>1</sup> The code is littered with checks to `me()?`, `uiConfig?`, `config?`, etc. In the
*context* of being logged in as an administrator those should never be in doubt. I intend to make
our interfaces not have any doubt.
* Function to help generate sizing solutions across Javascript and CSS.
* web: refactor sidebar capabilities for categorical subsections
Move open/close logic into the ak-admin-sidebar itself.
This commit removes the responsibility for opening/closing the sidebar from the interface parent
code and places it inside the sidebar entirely. Since the Django invocation passes none of the
properties ak-interface-admin is capable of receiving, this seems like a safe operation.
The sidebar now assumes the responsibility for hooking up the window event listeners for open/close
and resize.
On connection to the DOM, and on resize, the sidebar checks to see if the viewport width meets the
criteria for a behavioral change (slide-overlay vs slide-push), and on slide-push automatically
opens the sidebar on the assumption that there's plenty of room. In order to support more dynamic
styling going forward, I've substituted the 1280px with 80rem, which is the same, but allows for
some better styling if someone with older eyes needs to "zoom in" on the whole thing with a larger
font size.
The hide/show code involves "reaching up" to touch the host's classList. There's a comment
indicating that this is a slightly fragile thing to do, but in a well-known way.
2023-11-20 18:24:59 +00:00
|
|
|
_configContext = new ContextProvider(this, {
|
|
|
|
context: authentikConfigContext,
|
|
|
|
initialValue: undefined,
|
|
|
|
});
|
|
|
|
|
|
|
|
_config?: Config;
|
|
|
|
|
2023-03-23 13:05:14 +00:00
|
|
|
@state()
|
web: refactor sidebar capabilities for categorical subsections (#7482)
* web: break circular dependency between AKElement & Interface.
This commit changes the way the root node of the web application shell is
discovered by child components, such that the base class shared by both
no longer results in a circular dependency between the two models.
I've run this in isolation and have seen no failures of discovery; the identity
token exists as soon as the Interface is constructed and is found by every item
on the page.
* web: fix broken typescript references
This built... and then it didn't? Anyway, the current fix is to
provide type information the AkInterface for the data that consumers
require.
* web: rollback dependabot's upgrade of context
The most frustrating part of this is that I RAN THIS, dammit, with the updated
context and the current Wizard, and it finished the End-to-End tests without
complaint.
* Due for amendment
* Revert "Due for amendment"
This reverts commit 829ad5d3f214fa163958593636b28300d010da42.
* web: refactor sidebar capabilities for categorical subsections
The project "Change Admin UI lists to have sublists per type" requires some initial changes to the
UI to facilitate this request. The AdminSidebar is the principle target of this project, and it is
embedded in the AdminInterface. To facilitate editing the AdminSidebar as an independent entity,
AdminInterface has been moved into its own folder and the AdminSidebar extracted as a standalone Web
Component. This removes, oh, about half the code from AdminInterface. A little cleanup with
`classMap` was also committed.
The rollup config was adjusted to find the new AdminInterface location.
The Sidebar uses the global `config: Config` object to check for Enterprise capabilities. Rather
than plumb all the way down through the Interface => AdminInterface -> AdminSidebar, I chose to make
provide an alternative way of reaching the `config` object, as a *context*. Other configuration
objects (Me, UiConfig, Tenant) interfaces will be contextualized as demand warrants.
Demand will warrant. Just not yet. <sup>1</sup>
The Sidebar has been refactored only slightly; the renderers are entirely the same as they were
prior to extraction. What has been changed is the source of information: when we retrieve the
current version we story *only* the information, and use type information to ensure that the version
we store is the version we care about. The same is true of `impersonation`; we care only about the
name of the person being impersonated being present, so we don't store anything else.
Fetches have been moved from `firstUpdated` to the constructor. No reason to have the sidebar
render twice if the network returns before the render is scheduled.
Because the path used to identify the user being impersonated has changed, the `str()` references in
the XLIFF files had to be adjusted. **This change is to a variable only and does not require
translation.**
---
<sup>1</sup> The code is littered with checks to `me()?`, `uiConfig?`, `config?`, etc. In the
*context* of being logged in as an administrator those should never be in doubt. I intend to make
our interfaces not have any doubt.
* Function to help generate sizing solutions across Javascript and CSS.
* web: refactor sidebar capabilities for categorical subsections
Move open/close logic into the ak-admin-sidebar itself.
This commit removes the responsibility for opening/closing the sidebar from the interface parent
code and places it inside the sidebar entirely. Since the Django invocation passes none of the
properties ak-interface-admin is capable of receiving, this seems like a safe operation.
The sidebar now assumes the responsibility for hooking up the window event listeners for open/close
and resize.
On connection to the DOM, and on resize, the sidebar checks to see if the viewport width meets the
criteria for a behavioral change (slide-overlay vs slide-push), and on slide-push automatically
opens the sidebar on the assumption that there's plenty of room. In order to support more dynamic
styling going forward, I've substituted the 1280px with 80rem, which is the same, but allows for
some better styling if someone with older eyes needs to "zoom in" on the whole thing with a larger
font size.
The hide/show code involves "reaching up" to touch the host's classList. There's a comment
indicating that this is a slightly fragile thing to do, but in a well-known way.
2023-11-20 18:24:59 +00:00
|
|
|
set config(c: Config) {
|
|
|
|
this._config = c;
|
|
|
|
this._configContext.setValue(c);
|
|
|
|
this.requestUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
get config(): Config | undefined {
|
|
|
|
return this._config;
|
|
|
|
}
|
2023-03-23 13:05:14 +00:00
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|