Turns out that was one layer too many; the topmost component was fine for

maintaining the context.
This commit is contained in:
Ken Sternberg 2023-08-11 14:57:48 -07:00
parent 7fae17dac3
commit c0294191ad
3 changed files with 77 additions and 99 deletions

View file

@ -1,85 +0,0 @@
import { CustomListenerElement } from "@goauthentik/elements/utils/eventEmitter";
import { provide } from "@lit-labs/context";
import { customElement, property, state } from "@lit/reactive-element/decorators.js";
import { LitElement, html } from "lit";
import { akWizardCurrentStepContextName } from "./akWizardCurrentStepContextName";
import { akWizardStepsContextName } from "./akWizardStepsContextName";
import type { WizardStep, WizardStepId } from "./types";
/**
* AkWizardContext
*
* @element ak-wizard-context
*
* The WizardContext controls the navigation for the wizard. It listens for navigation events from
* the wizard frame and responds with changes to the view, including handling the close button.
*
*/
@customElement("ak-wizard-context")
export class AkWizardContext extends CustomListenerElement(LitElement) {
@property()
eventName: string = "ak-wizard-nav";
@provide({ context: akWizardStepsContextName })
@property({ attribute: false })
steps: WizardStep[] = [];
@provide({ context: akWizardCurrentStepContextName })
@state()
currentStep!: WizardStep;
constructor() {
super();
this.handleNavigation = this.handleNavigation.bind(this);
}
// This is the only case where currentStep could be anything other than a valid entry. Unless,
// of course, a step itself is so badly messed up it can't point to a real object.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
willUpdate(_changedProperties: Map<string, any>) {
if (this.currentStep === undefined) {
this.currentStep = this.steps[0];
}
}
// Note that we always scan for the valid next step and throw an error if we can't find it.
// There should never be a question that the currentStep is a *valid* step.
//
// TODO: Put a phase in there so that the current step can validate the contents asynchronously
// before setting the currentStep. Especially since setting the currentStep triggers a second
// asynchronous event-- scheduling a re-render of everything interested in the currentStep
// object.
handleNavigation(event: CustomEvent<{ step: WizardStepId }>) {
const requestedStep = event.detail.step;
if (!requestedStep) {
throw new Error("Request for next step when no next step is available");
}
const step = this.steps.find(({ id }) => id === requestedStep);
if (!step) {
throw new Error("Request for next step when no next step is available.");
}
if (step.disabled) {
throw new Error("Request for next step when the next step is disabled.");
}
this.currentStep = step;
return;
}
connectedCallback() {
super.connectedCallback();
this.addCustomListener(this.eventName, this.handleNavigation);
}
disconnectedCallback() {
this.removeCustomListener(this.eventName, this.handleNavigation);
super.disconnectedCallback();
}
render() {
return html`<slot></slot>`;
}
}

View file

@ -123,6 +123,8 @@ export class AkWizardFrame extends CustomEmitterElement(ModalButton) {
`; `;
} }
// This is where the panel is shown. We expect the panel to get its information from an
// independent context.
renderMainSection() { renderMainSection() {
return html`<main class="pf-c-wizard__main"> return html`<main class="pf-c-wizard__main">
<div class="pf-c-wizard__main-body">${this.currentStep.renderer()}</div> <div class="pf-c-wizard__main-body">${this.currentStep.renderer()}</div>

View file

@ -1,16 +1,18 @@
import { AKElement } from "@goauthentik/elements/Base"; import { AKElement } from "@goauthentik/elements/Base";
import { CustomListenerElement } from "@goauthentik/elements/utils/eventEmitter";
import { customElement } from "@lit/reactive-element/decorators/custom-element.js"; import { provide } from "@lit-labs/context";
import { html } from "lit"; import { html } from "lit";
import { property } from "lit/decorators.js"; import { customElement, property, state } from "lit/decorators.js";
import { ifDefined } from "lit/directives/if-defined.js"; import { ifDefined } from "lit/directives/if-defined.js";
import PFButton from "@patternfly/patternfly/components/Button/button.css"; import PFButton from "@patternfly/patternfly/components/Button/button.css";
import PFRadio from "@patternfly/patternfly/components/Radio/radio.css"; import PFRadio from "@patternfly/patternfly/components/Radio/radio.css";
import PFBase from "@patternfly/patternfly/patternfly-base.css"; import PFBase from "@patternfly/patternfly/patternfly-base.css";
import "./ak-wizard-context";
import "./ak-wizard-frame"; import "./ak-wizard-frame";
import { akWizardCurrentStepContextName } from "./akWizardCurrentStepContextName";
import { akWizardStepsContextName } from "./akWizardStepsContextName";
import type { WizardStep } from "./types"; import type { WizardStep } from "./types";
/** /**
@ -23,21 +25,39 @@ import type { WizardStep } from "./types";
*/ */
@customElement("ak-wizard-main") @customElement("ak-wizard-main")
export class AkWizardMain extends AKElement { export class AkWizardMain extends CustomListenerElement(AKElement) {
static get styles() { static get styles() {
return [PFBase, PFButton, PFRadio]; return [PFBase, PFButton, PFRadio];
} }
@property()
eventName: string = "ak-wizard-nav";
/** /**
* The steps of the Wizard. * The steps of the Wizard.
* *
* @attribute * @attribute
*/ */
@provide({ context: akWizardStepsContextName })
@property({ attribute: false }) @property({ attribute: false })
steps: WizardStep[] = []; steps: WizardStep[] = [];
/** /**
* The text of the button * The current step of the wizard.
*
* @attribute
*/
@provide({ context: akWizardCurrentStepContextName })
@state()
currentStep!: WizardStep;
constructor() {
super();
this.handleNavigation = this.handleNavigation.bind(this);
}
/**
* The text of the modal button
* *
* @attribute * @attribute
*/ */
@ -68,17 +88,58 @@ export class AkWizardMain extends AKElement {
@property() @property()
description?: string; description?: string;
// Guarantee that if the current step was not passed in by the client, that we know
// and set to the first step.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
willUpdate(_changedProperties: Map<string, any>) {
if (this.currentStep === undefined) {
this.currentStep = this.steps[0];
}
}
connectedCallback() {
super.connectedCallback();
this.addCustomListener(this.eventName, this.handleNavigation);
}
disconnectedCallback() {
this.removeCustomListener(this.eventName, this.handleNavigation);
super.disconnectedCallback();
}
// Note that we always scan for the valid next step and throw an error if we can't find it.
// There should never be a question that the currentStep is a *valid* step.
//
// TODO: Put a phase in there so that the current step can validate the contents asynchronously
// before setting the currentStep. Especially since setting the currentStep triggers a second
// asynchronous event-- scheduling a re-render of everything interested in the currentStep
// object.
handleNavigation(event: CustomEvent<{ step: string }>) {
const requestedStep = event.detail.step;
if (!requestedStep) {
throw new Error("Request for next step when no next step is available");
}
const step = this.steps.find(({ id }) => id === requestedStep);
if (!step) {
throw new Error("Request for next step when no next step is available.");
}
if (step.disabled) {
throw new Error("Request for next step when the next step is disabled.");
}
this.currentStep = step;
return;
}
render() { render() {
return html` return html`
<ak-wizard-context .steps=${this.steps}> <ak-wizard-frame
<ak-wizard-frame ?open=${this.open}
?open=${this.open} header=${this.header}
header=${this.header} description=${ifDefined(this.description)}
description=${ifDefined(this.description)} eventName=${this.eventName}
> >
<button slot="trigger" class="pf-c-button pf-m-primary">${this.prompt}</button> <button slot="trigger" class="pf-c-button pf-m-primary">${this.prompt}</button>
</ak-wizard-frame> </ak-wizard-frame>
</ak-wizard-context>
`; `;
} }
} }