91 lines
3.1 KiB
TypeScript
91 lines
3.1 KiB
TypeScript
import "#elements/chips/Chip";
|
|
import "#elements/chips/ChipGroup";
|
|
import "#admin/applications/ProviderSelectModal";
|
|
|
|
import { AKElement } from "#elements/Base";
|
|
|
|
import { Provider } from "@goauthentik/api";
|
|
|
|
import { TemplateResult, html, nothing } from "lit";
|
|
import { customElement, property } from "lit/decorators.js";
|
|
import { ifDefined } from "lit/directives/if-defined.js";
|
|
import { map } from "lit/directives/map.js";
|
|
|
|
@customElement("ak-backchannel-providers-input")
|
|
export class AkBackchannelProvidersInput extends AKElement {
|
|
// Render into the lightDOM. This effectively erases the shadowDOM nature of this component, but
|
|
// we're not actually using that and, for the meantime, we need the form handlers to be able to
|
|
// find the children of this component.
|
|
//
|
|
// This field is so highly specialized that it would make more sense if we put the API and the
|
|
// fetcher here.
|
|
//
|
|
// TODO: This abstraction is wrong; it's putting *more* layers in as a way of managing the
|
|
// visual clutter and legibility issues of ak-form-elemental-horizontal and patternfly in
|
|
// general.
|
|
|
|
protected createRenderRoot() {
|
|
return this as HTMLElement;
|
|
}
|
|
|
|
@property({ type: String })
|
|
name!: string;
|
|
|
|
@property({ type: String })
|
|
label = "";
|
|
|
|
@property({ type: Array })
|
|
providers: Provider[] = [];
|
|
|
|
@property({ type: Object })
|
|
tooltip?: TemplateResult;
|
|
|
|
@property({ attribute: false, type: Object })
|
|
confirm!: ({ items }: { items: Provider[] }) => Promise<void>;
|
|
|
|
@property({ attribute: false, type: Object })
|
|
remover!: (provider: Provider) => () => void;
|
|
|
|
@property({ type: String })
|
|
value = "";
|
|
|
|
@property({ type: Boolean })
|
|
required = false;
|
|
|
|
@property({ type: String })
|
|
help = "";
|
|
|
|
render() {
|
|
const renderOneChip = (provider: Provider) =>
|
|
html`<ak-chip
|
|
.removable=${true}
|
|
value=${ifDefined(provider.pk)}
|
|
@remove=${this.remover(provider)}
|
|
>${provider.name}</ak-chip
|
|
>`;
|
|
|
|
return html`
|
|
<ak-form-element-horizontal label=${this.label} name=${this.name}>
|
|
<div class="pf-c-input-group">
|
|
<ak-provider-select-table ?backchannel=${true} .confirm=${this.confirm}>
|
|
<button slot="trigger" class="pf-c-button pf-m-control" type="button">
|
|
${this.tooltip ? this.tooltip : nothing}
|
|
<i class="fas fa-plus" aria-hidden="true"></i>
|
|
</button>
|
|
</ak-provider-select-table>
|
|
<div class="pf-c-form-control">
|
|
<ak-chip-group> ${map(this.providers, renderOneChip)} </ak-chip-group>
|
|
</div>
|
|
</div>
|
|
${this.help ? html`<p class="pf-c-form__helper-text">${this.help}</p>` : nothing}
|
|
</ak-form-element-horizontal>
|
|
`;
|
|
}
|
|
}
|
|
|
|
declare global {
|
|
interface HTMLElementTagNameMap {
|
|
"ak-backchannel-providers-input": AkBackchannelProvidersInput;
|
|
}
|
|
}
|