web/admin: Dual select state management, custom event dispatching. (#14490)
* web/admin: Fix issues surrounding dual select state management. * web: Fix nested path. * web: Use PatternFly variable.
This commit is contained in:
@ -3,6 +3,7 @@ import {
|
||||
CustomEmitterElement,
|
||||
CustomListenerElement,
|
||||
} from "@goauthentik/elements/utils/eventEmitter";
|
||||
import { match } from "ts-pattern";
|
||||
|
||||
import { msg, str } from "@lit/localize";
|
||||
import { PropertyValues, html, nothing } from "lit";
|
||||
@ -15,34 +16,41 @@ import { globalVariables, mainStyles } from "./components/styles.css";
|
||||
import PFButton from "@patternfly/patternfly/components/Button/button.css";
|
||||
import PFBase from "@patternfly/patternfly/patternfly-base.css";
|
||||
|
||||
import "./components/ak-dual-select-available-pane";
|
||||
import { AkDualSelectAvailablePane } from "./components/ak-dual-select-available-pane";
|
||||
import "./components/ak-dual-select-controls";
|
||||
import "./components/ak-dual-select-selected-pane";
|
||||
import { AkDualSelectSelectedPane } from "./components/ak-dual-select-selected-pane";
|
||||
import "./components/ak-pagination";
|
||||
import "./components/ak-search-bar";
|
||||
import "./components/ak-dual-select-available-pane.js";
|
||||
import { AkDualSelectAvailablePane } from "./components/ak-dual-select-available-pane.js";
|
||||
import "./components/ak-dual-select-controls.js";
|
||||
import "./components/ak-dual-select-selected-pane.js";
|
||||
import { AkDualSelectSelectedPane } from "./components/ak-dual-select-selected-pane.js";
|
||||
import "./components/ak-pagination.js";
|
||||
import "./components/ak-search-bar.js";
|
||||
import {
|
||||
EVENT_ADD_ALL,
|
||||
EVENT_ADD_ONE,
|
||||
EVENT_ADD_SELECTED,
|
||||
EVENT_DELETE_ALL,
|
||||
EVENT_REMOVE_ALL,
|
||||
EVENT_REMOVE_ONE,
|
||||
EVENT_REMOVE_SELECTED,
|
||||
} from "./constants";
|
||||
import type { BasePagination, DualSelectPair, SearchbarEvent } from "./types";
|
||||
BasePagination,
|
||||
DualSelectEventType,
|
||||
DualSelectPair,
|
||||
SearchbarEventDetail,
|
||||
SearchbarEventSource,
|
||||
} from "./types.js";
|
||||
|
||||
function alphaSort([_k1, v1, s1]: DualSelectPair, [_k2, v2, s2]: DualSelectPair) {
|
||||
const [l, r] = [s1 !== undefined ? s1 : v1, s2 !== undefined ? s2 : v2];
|
||||
return l < r ? -1 : l > r ? 1 : 0;
|
||||
function localeComparator(a: DualSelectPair, b: DualSelectPair) {
|
||||
const aSortBy = a[2];
|
||||
const bSortBy = b[2];
|
||||
|
||||
return aSortBy.localeCompare(bSortBy);
|
||||
}
|
||||
|
||||
function mapDualPairs(pairs: DualSelectPair[]) {
|
||||
return new Map(pairs.map(([k, v, _]) => [k, v]));
|
||||
function keyfinder(key: string) {
|
||||
return ([k]: DualSelectPair) => k === key;
|
||||
}
|
||||
|
||||
const styles = [PFBase, PFButton, globalVariables, mainStyles];
|
||||
const DelegatedEvents = [
|
||||
DualSelectEventType.AddSelected,
|
||||
DualSelectEventType.RemoveSelected,
|
||||
DualSelectEventType.AddAll,
|
||||
DualSelectEventType.RemoveAll,
|
||||
DualSelectEventType.DeleteAll,
|
||||
DualSelectEventType.AddOne,
|
||||
DualSelectEventType.RemoveOne,
|
||||
] as const satisfies DualSelectEventType[];
|
||||
|
||||
/**
|
||||
* @element ak-dual-select
|
||||
@ -53,24 +61,25 @@ const styles = [PFBase, PFButton, globalVariables, mainStyles];
|
||||
*
|
||||
* @fires ak-dual-select-change - A custom change event with the current `selected` list.
|
||||
*/
|
||||
|
||||
const keyfinder =
|
||||
(key: string) =>
|
||||
([k]: DualSelectPair) =>
|
||||
k === key;
|
||||
|
||||
@customElement("ak-dual-select")
|
||||
export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKElement)) {
|
||||
static get styles() {
|
||||
return styles;
|
||||
}
|
||||
static styles = [PFBase, PFButton, globalVariables, mainStyles];
|
||||
|
||||
/* The list of options to *currently* show. Note that this is not *all* the options, only the
|
||||
* currently shown list of options from a pagination collection. */
|
||||
//#region Properties
|
||||
|
||||
/**
|
||||
* The list of options to *currently* show.
|
||||
*
|
||||
* Note that this is not *all* the options,
|
||||
* only the currently shown list of options from a pagination collection.
|
||||
*/
|
||||
@property({ type: Array })
|
||||
options: DualSelectPair[] = [];
|
||||
|
||||
/* The list of options selected. This is the *entire* list and will not be paginated. */
|
||||
/**
|
||||
* The list of options selected.
|
||||
* This is the *entire* list and will not be paginated.
|
||||
*/
|
||||
@property({ type: Array })
|
||||
selected: DualSelectPair[] = [];
|
||||
|
||||
@ -83,138 +92,133 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
@property({ attribute: "selected-label" })
|
||||
selectedLabel = msg("Selected options");
|
||||
|
||||
//#endregion
|
||||
|
||||
//#region State
|
||||
|
||||
@state()
|
||||
selectedFilter: string = "";
|
||||
protected selectedFilter: string = "";
|
||||
|
||||
#selectedKeys: Set<string> = new Set();
|
||||
|
||||
//#endregion
|
||||
|
||||
//#region Refs
|
||||
|
||||
availablePane: Ref<AkDualSelectAvailablePane> = createRef();
|
||||
|
||||
selectedPane: Ref<AkDualSelectSelectedPane> = createRef();
|
||||
|
||||
selectedKeys: Set<string> = new Set();
|
||||
//#endregion
|
||||
|
||||
//#region Lifecycle
|
||||
|
||||
constructor() {
|
||||
super();
|
||||
this.handleMove = this.handleMove.bind(this);
|
||||
this.handleSearch = this.handleSearch.bind(this);
|
||||
[
|
||||
EVENT_ADD_ALL,
|
||||
EVENT_ADD_SELECTED,
|
||||
EVENT_DELETE_ALL,
|
||||
EVENT_REMOVE_ALL,
|
||||
EVENT_REMOVE_SELECTED,
|
||||
EVENT_ADD_ONE,
|
||||
EVENT_REMOVE_ONE,
|
||||
].forEach((eventName: string) => {
|
||||
this.addCustomListener(eventName, (event: Event) => this.handleMove(eventName, event));
|
||||
});
|
||||
|
||||
for (const eventName of DelegatedEvents) {
|
||||
this.addCustomListener(eventName, this.#moveListener);
|
||||
}
|
||||
|
||||
this.addCustomListener("ak-dual-select-move", () => {
|
||||
this.requestUpdate();
|
||||
});
|
||||
this.addCustomListener("ak-search", this.handleSearch);
|
||||
|
||||
this.addCustomListener("ak-search", this.#searchListener);
|
||||
}
|
||||
|
||||
willUpdate(changedProperties: PropertyValues<this>) {
|
||||
if (changedProperties.has("selected")) {
|
||||
this.selectedKeys = new Set(this.selected.map(([key, _]) => key));
|
||||
this.#selectedKeys = new Set(this.selected.map(([key]) => key));
|
||||
}
|
||||
|
||||
// Pagination invalidates available moveables.
|
||||
if (changedProperties.has("options") && this.availablePane.value) {
|
||||
this.availablePane.value.clearMove();
|
||||
}
|
||||
}
|
||||
|
||||
handleMove(eventName: string, event: Event) {
|
||||
if (!(event instanceof CustomEvent)) {
|
||||
throw new Error(`Expected move event here, got ${eventName}`);
|
||||
}
|
||||
//#endregion
|
||||
|
||||
switch (eventName) {
|
||||
case EVENT_ADD_SELECTED: {
|
||||
this.addSelected();
|
||||
break;
|
||||
}
|
||||
case EVENT_REMOVE_SELECTED: {
|
||||
this.removeSelected();
|
||||
break;
|
||||
}
|
||||
case EVENT_ADD_ALL: {
|
||||
this.addAllVisible();
|
||||
break;
|
||||
}
|
||||
case EVENT_REMOVE_ALL: {
|
||||
this.removeAllVisible();
|
||||
break;
|
||||
}
|
||||
case EVENT_DELETE_ALL: {
|
||||
this.removeAll();
|
||||
break;
|
||||
}
|
||||
case EVENT_ADD_ONE: {
|
||||
this.addOne(event.detail);
|
||||
break;
|
||||
}
|
||||
case EVENT_REMOVE_ONE: {
|
||||
this.removeOne(event.detail);
|
||||
break;
|
||||
}
|
||||
//#region Event Listeners
|
||||
|
||||
#moveListener = (event: CustomEvent<string>) => {
|
||||
match(event.type)
|
||||
.with(DualSelectEventType.AddSelected, () => this.addSelected())
|
||||
.with(DualSelectEventType.RemoveSelected, () => this.removeSelected())
|
||||
.with(DualSelectEventType.AddAll, () => this.addAllVisible())
|
||||
.with(DualSelectEventType.RemoveAll, () => this.removeAllVisible())
|
||||
.with(DualSelectEventType.DeleteAll, () => this.removeAll())
|
||||
.with(DualSelectEventType.AddOne, () => this.addOne(event.detail))
|
||||
.with(DualSelectEventType.RemoveOne, () => this.removeOne(event.detail))
|
||||
.otherwise(() => {
|
||||
throw new Error(`Expected move event here, got ${event.type}`);
|
||||
});
|
||||
|
||||
this.dispatchCustomEvent(DualSelectEventType.Change, { value: this.value });
|
||||
|
||||
default:
|
||||
throw new Error(
|
||||
`AkDualSelect.handleMove received unknown event type: ${eventName}`,
|
||||
);
|
||||
}
|
||||
this.dispatchCustomEvent("ak-dual-select-change", { value: this.value });
|
||||
event.stopPropagation();
|
||||
}
|
||||
};
|
||||
|
||||
protected addSelected() {
|
||||
if (this.availablePane.value!.moveable.length === 0) return;
|
||||
|
||||
addSelected() {
|
||||
if (this.availablePane.value!.moveable.length === 0) {
|
||||
return;
|
||||
}
|
||||
this.selected = this.availablePane.value!.moveable.reduce(
|
||||
(acc, key) => {
|
||||
const value = this.options.find(keyfinder(key));
|
||||
|
||||
return value && !acc.find(keyfinder(value[0])) ? [...acc, value] : acc;
|
||||
},
|
||||
[...this.selected],
|
||||
);
|
||||
|
||||
// This is where the information gets... lossy. Dammit.
|
||||
this.availablePane.value!.clearMove();
|
||||
}
|
||||
|
||||
addOne(key: string) {
|
||||
protected addOne(key: string) {
|
||||
const requested = this.options.find(keyfinder(key));
|
||||
if (requested && !this.selected.find(keyfinder(requested[0]))) {
|
||||
this.selected = [...this.selected, requested];
|
||||
}
|
||||
|
||||
if (!requested) return;
|
||||
if (this.selected.find(keyfinder(requested[0]))) return;
|
||||
|
||||
this.selected = [...this.selected, requested];
|
||||
}
|
||||
|
||||
// These are the *currently visible* options; the parent node is responsible for paginating and
|
||||
// updating the list of currently visible options;
|
||||
addAllVisible() {
|
||||
protected addAllVisible() {
|
||||
// Create a new array of all current options and selected, and de-dupe.
|
||||
const selected = mapDualPairs([...this.options, ...this.selected]);
|
||||
this.selected = Array.from(selected.entries());
|
||||
const selected = new Map<string, DualSelectPair>([
|
||||
...this.options.map((pair) => [pair[0], pair] as const),
|
||||
...this.selected.map((pair) => [pair[0], pair] as const),
|
||||
]);
|
||||
|
||||
this.selected = Array.from(selected.values());
|
||||
|
||||
this.availablePane.value!.clearMove();
|
||||
}
|
||||
|
||||
removeSelected() {
|
||||
if (this.selectedPane.value!.moveable.length === 0) {
|
||||
return;
|
||||
}
|
||||
protected removeSelected() {
|
||||
if (this.selectedPane.value!.moveable.length === 0) return;
|
||||
|
||||
const deselected = new Set(this.selectedPane.value!.moveable);
|
||||
|
||||
this.selected = this.selected.filter(([key]) => !deselected.has(key));
|
||||
|
||||
this.selectedPane.value!.clearMove();
|
||||
}
|
||||
|
||||
removeOne(key: string) {
|
||||
protected removeOne(key: string) {
|
||||
this.selected = this.selected.filter(([k]) => k !== key);
|
||||
}
|
||||
|
||||
removeAllVisible() {
|
||||
protected removeAllVisible() {
|
||||
// Remove all the items from selected that are in the *currently visible* options list
|
||||
const options = new Set(this.options.map(([k, _]) => k));
|
||||
const options = new Set(this.options.map(([k]) => k));
|
||||
|
||||
this.selected = this.selected.filter(([k]) => !options.has(k));
|
||||
|
||||
this.selectedPane.value!.clearMove();
|
||||
}
|
||||
|
||||
@ -223,24 +227,25 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
this.selectedPane.value!.clearMove();
|
||||
}
|
||||
|
||||
handleSearch(event: SearchbarEvent) {
|
||||
switch (event.detail.source) {
|
||||
case "ak-dual-list-available-search":
|
||||
return this.handleAvailableSearch(event.detail.value);
|
||||
case "ak-dual-list-selected-search":
|
||||
return this.handleSelectedSearch(event.detail.value);
|
||||
}
|
||||
#searchListener = (event: CustomEvent<SearchbarEventDetail>) => {
|
||||
const { source, value } = event.detail;
|
||||
|
||||
match(source)
|
||||
.with(SearchbarEventSource.Available, () => {
|
||||
this.dispatchCustomEvent(DualSelectEventType.Search, value);
|
||||
})
|
||||
.with(SearchbarEventSource.Selected, () => {
|
||||
this.selectedFilter = value;
|
||||
this.selectedPane.value!.clearMove();
|
||||
})
|
||||
.exhaustive();
|
||||
|
||||
event.stopPropagation();
|
||||
}
|
||||
};
|
||||
|
||||
handleAvailableSearch(value: string) {
|
||||
this.dispatchCustomEvent("ak-dual-select-search", value);
|
||||
}
|
||||
//#endregion
|
||||
|
||||
handleSelectedSearch(value: string) {
|
||||
this.selectedFilter = value;
|
||||
this.selectedPane.value!.clearMove();
|
||||
}
|
||||
//#region Public Getters
|
||||
|
||||
get value() {
|
||||
return this.selected;
|
||||
@ -251,7 +256,7 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
// added.
|
||||
const allMoved =
|
||||
this.options.length ===
|
||||
this.options.filter(([key, _]) => this.selectedKeys.has(key)).length;
|
||||
this.options.filter(([key, _]) => this.#selectedKeys.has(key)).length;
|
||||
|
||||
return this.options.length > 0 && !allMoved;
|
||||
}
|
||||
@ -259,7 +264,8 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
get canRemoveAll() {
|
||||
// False if no visible option can be found in the selected list
|
||||
return (
|
||||
this.options.length > 0 && !!this.options.find(([key, _]) => this.selectedKeys.has(key))
|
||||
this.options.length > 0 &&
|
||||
!!this.options.find(([key, _]) => this.#selectedKeys.has(key))
|
||||
);
|
||||
}
|
||||
|
||||
@ -267,6 +273,10 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
return (this.pages?.next ?? 0) > 0 || (this.pages?.previous ?? 0) > 0;
|
||||
}
|
||||
|
||||
//#endregion
|
||||
|
||||
//#region Render
|
||||
|
||||
render() {
|
||||
const selected =
|
||||
this.selectedFilter === ""
|
||||
@ -282,11 +292,15 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
const availableCount = this.availablePane.value?.toMove.size ?? 0;
|
||||
const selectedCount = this.selectedPane.value?.toMove.size ?? 0;
|
||||
const selectedTotal = selected.length;
|
||||
|
||||
const availableStatus =
|
||||
availableCount > 0 ? msg(str`${availableCount} item(s) marked to add.`) : " ";
|
||||
|
||||
const selectedTotalStatus = msg(str`${selectedTotal} item(s) selected.`);
|
||||
|
||||
const selectedCountStatus =
|
||||
selectedCount > 0 ? " " + msg(str`${selectedCount} item(s) marked to remove.`) : "";
|
||||
|
||||
const selectedStatus = `${selectedTotalStatus} ${selectedCountStatus}`;
|
||||
|
||||
return html`
|
||||
@ -310,7 +324,7 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
<ak-dual-select-available-pane
|
||||
${ref(this.availablePane)}
|
||||
.options=${this.options}
|
||||
.selected=${this.selectedKeys}
|
||||
.selected=${this.#selectedKeys}
|
||||
></ak-dual-select-available-pane>
|
||||
${this.needPagination
|
||||
? html`<ak-pagination .pages=${this.pages}></ak-pagination>`
|
||||
@ -344,12 +358,14 @@ export class AkDualSelect extends CustomEmitterElement(CustomListenerElement(AKE
|
||||
|
||||
<ak-dual-select-selected-pane
|
||||
${ref(this.selectedPane)}
|
||||
.selected=${selected.toSorted(alphaSort)}
|
||||
.selected=${selected.toSorted(localeComparator)}
|
||||
></ak-dual-select-selected-pane>
|
||||
</div>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
//#endregion
|
||||
}
|
||||
|
||||
declare global {
|
||||
|
||||
Reference in New Issue
Block a user