import { css, CSSResult, customElement, html, LitElement, property, TemplateResult } from "lit-element"; import { unsafeHTML } from "lit-html/directives/unsafe-html"; import PFBase from "@patternfly/patternfly/patternfly-base.css"; import PFButton from "@patternfly/patternfly/components/Button/button.css"; import PFModalBox from "@patternfly/patternfly/components/ModalBox/modal-box.css"; import PFForm from "@patternfly/patternfly/components/Form/form.css"; import PFFormControl from "@patternfly/patternfly/components/FormControl/form-control.css"; import PFBullseye from "@patternfly/patternfly/layouts/Bullseye/bullseye.css"; import PFBackdrop from "@patternfly/patternfly/components/Backdrop/backdrop.css"; import PFPage from "@patternfly/patternfly/components/Page/page.css"; import PFStack from "@patternfly/patternfly/layouts/Stack/stack.css"; import PFCard from "@patternfly/patternfly/components/Card/card.css"; import PFContent from "@patternfly/patternfly/components/Content/content.css"; import AKGlobal from "../../authentik.css"; import CodeMirrorStyle from "codemirror/lib/codemirror.css"; import CodeMirrorTheme from "codemirror/theme/monokai.css"; import { convertToSlug } from "../../utils"; import { SpinnerButton } from "./SpinnerButton"; import { PRIMARY_CLASS, EVENT_REFRESH } from "../../constants"; import { showMessage } from "../messages/MessageContainer"; import { MessageLevel } from "../messages/Message"; @customElement("ak-modal-button") export class ModalButton extends LitElement { @property() href?: string; @property({type: Boolean}) open = false; @property() modal = ""; static get styles(): CSSResult[] { return [PFBase, PFButton, PFModalBox, PFForm, PFFormControl, PFBullseye, PFBackdrop, PFPage, PFStack, PFCard, PFContent, AKGlobal, CodeMirrorStyle, CodeMirrorTheme].concat( css` :host { text-align: left; } .pf-c-modal-box.pf-m-lg { overflow-y: auto; } .pf-c-modal-box > .pf-c-button + * { margin-right: 0; } /* fix multiple selects height */ select[multiple] { height: 15em; } ` ); } constructor() { super(); window.addEventListener("keyup", (e) => { if (e.code === "Escape") { this.open = false; } }); } updateHandlers(): void { // Ensure links close the modal this.shadowRoot?.querySelectorAll("a").forEach((a) => { if (a.target == "_blank") { return; } // Make click on a close the modal a.addEventListener("click", (e) => { e.preventDefault(); this.open = false; }); }); // Make name field update slug field this.shadowRoot?.querySelectorAll("input[name=name]").forEach((input) => { const form = input.closest("form"); if (form === null) { return; } const slugField = form.querySelector("input[name=slug]"); if (!slugField) { return; } // Only attach handler if the slug is already equal to the name // if not, they are probably completely different and shouldn't update // each other if (convertToSlug(input.value) !== slugField.value) { return; } input.addEventListener("input", () => { slugField.value = convertToSlug(input.value); }); }); // Ensure forms sends in AJAX this.shadowRoot?.querySelectorAll("form").forEach((form) => { form.addEventListener("submit", (e) => { e.preventDefault(); const formData = new FormData(form); fetch(this.href ? this.href : form.action, { method: form.method, body: formData, redirect: "manual", }) .then((response) => { return response.text(); }) .then((responseData) => { if (responseData.indexOf("csrfmiddlewaretoken") !== -1) { this.modal = responseData; console.debug("authentik/modalbutton: re-showing form"); } else { this.open = false; console.debug("authentik/modalbutton: successful submit"); this.dispatchEvent( new CustomEvent(EVENT_REFRESH, { bubbles: true, composed: true, }) ); } }) .catch((e) => { showMessage({ level: MessageLevel.error, message: "Unexpected error" }); console.error(e); }); }); }); } onClick(): void { if (!this.href) { this.updateHandlers(); this.open = true; this.querySelectorAll("*").forEach(child => { if ("requestUpdate" in child) { (child as LitElement).requestUpdate(); } }); } else { const request = new Request(this.href); fetch(request, { mode: "same-origin", }) .then((response) => response.text()) .then((responseData) => { this.modal = responseData; this.open = true; this.querySelectorAll("ak-spinner-button").forEach((sb) => { sb.setDone(PRIMARY_CLASS); }); }) .catch((e) => { showMessage({ level: MessageLevel.error, message: "Unexpected error" }); console.error(e); }); } } renderModalInner(): TemplateResult { return html`${unsafeHTML(this.modal)}`; } renderModal(): TemplateResult { return html`
`; } render(): TemplateResult { return html` this.onClick()}> ${this.open ? this.renderModal() : ""}`; } updated(): void { this.updateHandlers(); } }