import { css, customElement, html, LitElement, property } from "lit-element"; // @ts-ignore import ModalBoxStyle from "@patternfly/patternfly/components/ModalBox/modal-box.css"; // @ts-ignore import BullseyeStyle from "@patternfly/patternfly/layouts/Bullseye/bullseye.css"; // @ts-ignore import BackdropStyle from "@patternfly/patternfly/components/Backdrop/backdrop.css"; // @ts-ignore import ButtonStyle from "@patternfly/patternfly/components/Button/button.css"; // @ts-ignore import fa from "@fortawesome/fontawesome-free/css/solid.css"; import { convertToSlug } from "../../utils"; import { SpinnerButton } from "./SpinnerButton"; import { PRIMARY_CLASS } from "../../constants"; @customElement("pb-modal-button") export class ModalButton extends LitElement { @property() href?: string; @property() open: boolean = false; static get styles() { return [ css` :host { text-align: left; } ::slotted(*) { overflow-y: auto; } `, ModalBoxStyle, BullseyeStyle, BackdropStyle, ButtonStyle, fa, ]; } constructor() { super(); window.addEventListener("keyup", (e) => { if (e.code === "Escape") { this.open = false; } }); } updateHandlers() { // Ensure links close the modal this.querySelectorAll("[slot=modal] 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.querySelectorAll("input[name=name]").forEach((input) => { input.addEventListener("input", (e) => { const form = input.closest("form"); if (form === null) { return; } const slugField = form.querySelector("input[name=slug]"); if (!slugField) { return; } slugField.value = convertToSlug(input.value); }); }); // Ensure forms sends in AJAX this.querySelectorAll("[slot=modal] form").forEach((form) => { form.addEventListener("submit", (e) => { e.preventDefault(); let formData = new FormData(form); fetch(this.href ? this.href : form.action, { method: form.method, body: formData, redirect: "manual", }) .then((response) => { return response.text(); }) .then((data) => { if (data.indexOf("csrfmiddlewaretoken") !== -1) { this.querySelector("[slot=modal]")!.innerHTML = data; console.debug(`passbook/modalbutton: re-showing form`); this.updateHandlers(); } else { this.open = false; console.debug(`passbook/modalbutton: successful submit`); this.dispatchEvent( new CustomEvent("hashchange", { bubbles: true, }) ); } }) .catch((e) => { console.error(e); }); }); }); } onClick(e: MouseEvent) { if (!this.href) { this.updateHandlers(); this.open = true; } else { const request = new Request(this.href); fetch(request, { mode: "same-origin", }) .then((r) => r.text()) .then((t) => { this.querySelector("[slot=modal]")!.innerHTML = t; this.updateHandlers(); this.open = true; this.querySelectorAll("pb-spinner-button").forEach((sb) => { sb.setDone(PRIMARY_CLASS); }); }) .catch((e) => { console.error(e); }); } } renderModal() { return html`
`; } render() { return html` this.onClick(e)}> ${this.open ? this.renderModal() : ""}`; } }