163 lines
5.5 KiB
TypeScript
163 lines
5.5 KiB
TypeScript
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<HTMLAnchorElement>("[slot=modal] a").forEach((a) => {
|
|
// Make click on a close the modal
|
|
a.addEventListener("click", (e) => {
|
|
e.preventDefault();
|
|
this.open = false;
|
|
});
|
|
});
|
|
// Make name field update slug field
|
|
this.querySelectorAll<HTMLInputElement>("input[name=name]").forEach((input) => {
|
|
input.addEventListener("input", (e) => {
|
|
const form = input.closest("form");
|
|
if (form === null) {
|
|
return;
|
|
}
|
|
const slugField = form.querySelector<HTMLInputElement>("input[name=slug]");
|
|
if (!slugField) {
|
|
return;
|
|
}
|
|
slugField.value = convertToSlug(input.value);
|
|
});
|
|
});
|
|
// Ensure forms sends in AJAX
|
|
this.querySelectorAll<HTMLFormElement>("[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<SpinnerButton>("pb-spinner-button").forEach(sb => {
|
|
sb.setDone(PRIMARY_CLASS);
|
|
});
|
|
})
|
|
.catch((e) => {
|
|
console.error(e);
|
|
});
|
|
}
|
|
}
|
|
|
|
renderModal() {
|
|
return html`<div class="pf-c-backdrop">
|
|
<div class="pf-l-bullseye">
|
|
<div
|
|
class="pf-c-modal-box pf-m-lg"
|
|
role="dialog"
|
|
aria-modal="true"
|
|
aria-labelledby="modal-md-title"
|
|
aria-describedby="modal-md-description"
|
|
>
|
|
<button
|
|
@click=${() => (this.open = false)}
|
|
class="pf-c-button pf-m-plain"
|
|
type="button"
|
|
aria-label="Close dialog"
|
|
>
|
|
<i class="fas fa-times" aria-hidden="true"></i>
|
|
</button>
|
|
<slot name="modal"> </slot>
|
|
</div>
|
|
</div>
|
|
</div>`;
|
|
}
|
|
|
|
render() {
|
|
return html` <slot name="trigger" @click=${(e: any) => this.onClick(e)}></slot>
|
|
${this.open ? this.renderModal() : ""}`;
|
|
}
|
|
}
|