mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
294 lines
9.7 KiB
TypeScript
294 lines
9.7 KiB
TypeScript
/**
|
|
* Copyright (c) OpenLens Authors. All rights reserved.
|
|
* Licensed under MIT License. See LICENSE in root directory for more information.
|
|
*/
|
|
|
|
import "./item-list-layout.scss";
|
|
|
|
import React, { ReactNode } from "react";
|
|
import { computed, makeObservable, untracked } from "mobx";
|
|
import type { ConfirmDialogParams } from "../confirm-dialog";
|
|
import type { TableCellProps, TableProps, TableRowProps, TableSortCallbacks } from "../table";
|
|
import { boundMethod, cssNames, IClassName, noop, StorageHelper } from "../../utils";
|
|
import type { AddRemoveButtonsProps } from "../add-remove-buttons";
|
|
import type { ItemObject, ItemStore } from "../../../common/item.store";
|
|
import type { SearchInputUrlProps } from "../input";
|
|
import { FilterType, pageFilters } from "./page-filters.store";
|
|
import { PageFiltersList } from "./page-filters-list";
|
|
import type { NamespaceStore } from "../+namespaces/namespace-store/namespace.store";
|
|
import namespaceStoreInjectable from "../+namespaces/namespace-store/namespace-store.injectable";
|
|
import { withInjectables } from "@ogre-tools/injectable-react";
|
|
import itemListLayoutStorageInjectable from "./storage.injectable";
|
|
import { ItemListLayoutContent } from "./content";
|
|
import { ItemListLayoutHeader } from "./header";
|
|
import groupBy from "lodash/groupBy";
|
|
import { ItemListLayoutFilters } from "./filters";
|
|
import { observer } from "mobx-react";
|
|
|
|
export type SearchFilter<I extends ItemObject> = (item: I) => string | number | (string | number)[];
|
|
export type SearchFilters<I extends ItemObject> = Record<string, SearchFilter<I>>;
|
|
export type ItemsFilter<I extends ItemObject> = (items: I[]) => I[];
|
|
export type ItemsFilters<I extends ItemObject> = Record<string, ItemsFilter<I>>;
|
|
|
|
export interface HeaderPlaceholders {
|
|
title?: ReactNode;
|
|
searchProps?: SearchInputUrlProps;
|
|
filters?: ReactNode;
|
|
info?: ReactNode;
|
|
}
|
|
|
|
export type HeaderCustomizer = (placeholders: HeaderPlaceholders) => HeaderPlaceholders;
|
|
export interface ItemListLayoutProps<I extends ItemObject> {
|
|
tableId?: string;
|
|
className: IClassName;
|
|
getItems: () => I[];
|
|
store: ItemStore<I>;
|
|
dependentStores?: ItemStore<ItemObject>[];
|
|
preloadStores?: boolean;
|
|
hideFilters?: boolean;
|
|
searchFilters?: SearchFilter<I>[];
|
|
/** @deprecated */
|
|
filterItems?: ItemsFilter<I>[];
|
|
|
|
// header (title, filtering, searching, etc.)
|
|
showHeader?: boolean;
|
|
headerClassName?: IClassName;
|
|
renderHeaderTitle?: ReactNode | ((parent: NonInjectedItemListLayout<I>) => ReactNode);
|
|
customizeHeader?: HeaderCustomizer | HeaderCustomizer[];
|
|
|
|
// items list configuration
|
|
isReady?: boolean; // show loading indicator while not ready
|
|
isSelectable?: boolean; // show checkbox in rows for selecting items
|
|
isConfigurable?: boolean;
|
|
copyClassNameFromHeadCells?: boolean;
|
|
sortingCallbacks?: TableSortCallbacks<I>;
|
|
tableProps?: Partial<TableProps<I>>; // low-level table configuration
|
|
renderTableHeader: TableCellProps[] | null;
|
|
renderTableContents: (item: I) => (ReactNode | TableCellProps)[];
|
|
renderItemMenu?: (item: I, store: ItemStore<I>) => ReactNode;
|
|
customizeTableRowProps?: (item: I) => Partial<TableRowProps>;
|
|
addRemoveButtons?: Partial<AddRemoveButtonsProps>;
|
|
virtual?: boolean;
|
|
|
|
// item details view
|
|
hasDetailsView?: boolean;
|
|
detailsItem?: I;
|
|
onDetails?: (item: I) => void;
|
|
|
|
// other
|
|
customizeRemoveDialog?: (selectedItems: I[]) => Partial<ConfirmDialogParams>;
|
|
renderFooter?: (parent: NonInjectedItemListLayout<I>) => React.ReactNode;
|
|
|
|
/**
|
|
* Message to display when a store failed to load
|
|
*
|
|
* @default "Failed to load items"
|
|
*/
|
|
failedToLoadMessage?: React.ReactNode;
|
|
|
|
filterCallbacks?: ItemsFilters<I>;
|
|
}
|
|
|
|
const defaultProps: Partial<ItemListLayoutProps<ItemObject>> = {
|
|
showHeader: true,
|
|
isSelectable: true,
|
|
isConfigurable: false,
|
|
copyClassNameFromHeadCells: true,
|
|
preloadStores: true,
|
|
dependentStores: [],
|
|
searchFilters: [],
|
|
customizeHeader: [],
|
|
filterItems: [],
|
|
hasDetailsView: true,
|
|
onDetails: noop,
|
|
virtual: true,
|
|
customizeTableRowProps: () => ({}),
|
|
failedToLoadMessage: "Failed to load items",
|
|
};
|
|
|
|
interface Dependencies {
|
|
namespaceStore: NamespaceStore;
|
|
itemListLayoutStorage: StorageHelper<{ showFilters: boolean }>;
|
|
}
|
|
|
|
@observer
|
|
class NonInjectedItemListLayout<I extends ItemObject> extends React.Component<ItemListLayoutProps<I> & Dependencies> {
|
|
static defaultProps = defaultProps as object;
|
|
|
|
constructor(props: ItemListLayoutProps<I> & Dependencies) {
|
|
super(props);
|
|
makeObservable(this);
|
|
}
|
|
|
|
async componentDidMount() {
|
|
const { isConfigurable, tableId, preloadStores } = this.props;
|
|
|
|
if (isConfigurable && !tableId) {
|
|
throw new Error("[ItemListLayout]: configurable list require props.tableId to be specified");
|
|
}
|
|
|
|
if (preloadStores) {
|
|
this.loadStores();
|
|
}
|
|
}
|
|
|
|
private loadStores() {
|
|
const { store, dependentStores } = this.props;
|
|
const stores = Array.from(new Set([store, ...dependentStores]));
|
|
|
|
stores.forEach(store => store.loadAll(this.props.namespaceStore.contextNamespaces));
|
|
}
|
|
|
|
get showFilters(): boolean {
|
|
return this.props.itemListLayoutStorage.get().showFilters;
|
|
}
|
|
|
|
set showFilters(showFilters: boolean) {
|
|
this.props.itemListLayoutStorage.merge({ showFilters });
|
|
}
|
|
|
|
@computed get filters() {
|
|
let { activeFilters } = pageFilters;
|
|
const { searchFilters } = this.props;
|
|
|
|
if (searchFilters.length === 0) {
|
|
activeFilters = activeFilters.filter(({ type }) => type !== FilterType.SEARCH);
|
|
}
|
|
|
|
return activeFilters;
|
|
}
|
|
|
|
@boundMethod
|
|
toggleFilters() {
|
|
this.showFilters = !this.showFilters;
|
|
}
|
|
|
|
@computed get isReady() {
|
|
return this.props.isReady ?? this.props.store.isLoaded;
|
|
}
|
|
|
|
renderFilters() {
|
|
const { hideFilters } = this.props;
|
|
const { isReady, filters } = this;
|
|
|
|
if (!isReady || !filters.length || hideFilters || !this.showFilters) {
|
|
return null;
|
|
}
|
|
|
|
return <PageFiltersList filters={filters} />;
|
|
}
|
|
|
|
private filterCallbacks: ItemsFilters<I> = {
|
|
[FilterType.SEARCH]: items => {
|
|
const { searchFilters } = this.props;
|
|
const search = pageFilters.getValues(FilterType.SEARCH)[0] || "";
|
|
|
|
if (search && searchFilters.length) {
|
|
const normalizeText = (text: string) => String(text).toLowerCase();
|
|
const searchTexts = [search].map(normalizeText);
|
|
|
|
return items.filter(item => {
|
|
return searchFilters.some(getTexts => {
|
|
const sourceTexts: string[] = [getTexts(item)].flat().map(normalizeText);
|
|
|
|
return sourceTexts.some(source => searchTexts.some(search => source.includes(search)));
|
|
});
|
|
});
|
|
}
|
|
|
|
return items;
|
|
},
|
|
};
|
|
|
|
@computed get items() {
|
|
const filterGroups = groupBy(this.filters, ({ type }) => type);
|
|
const filterItems: ItemsFilter<I>[] = [];
|
|
|
|
for (const [type, filtersGroup] of Object.entries(filterGroups)) {
|
|
const filterCallback = this.filterCallbacks[type] ?? this.props.filterCallbacks?.[type];
|
|
|
|
if (filterCallback && filtersGroup.length > 0) {
|
|
filterItems.push(filterCallback);
|
|
}
|
|
}
|
|
|
|
const items = this.props.getItems();
|
|
|
|
return applyFilters(filterItems.concat(this.props.filterItems), items);
|
|
}
|
|
|
|
render() {
|
|
return untracked(() => (
|
|
<div
|
|
className={cssNames("ItemListLayout flex column", this.props.className)}
|
|
>
|
|
<ItemListLayoutHeader
|
|
getItems={() => this.items}
|
|
getFilters={() => this.filters}
|
|
toggleFilters={this.toggleFilters}
|
|
store={this.props.store}
|
|
searchFilters={this.props.searchFilters}
|
|
showHeader={this.props.showHeader}
|
|
headerClassName={this.props.headerClassName}
|
|
renderHeaderTitle={this.props.renderHeaderTitle}
|
|
customizeHeader={this.props.customizeHeader}
|
|
/>
|
|
|
|
<ItemListLayoutFilters
|
|
getIsReady={() => this.isReady}
|
|
getFilters={() => this.filters}
|
|
getFiltersAreShown={() => this.showFilters}
|
|
hideFilters={this.props.hideFilters}
|
|
/>
|
|
|
|
<ItemListLayoutContent
|
|
getItems={() => this.items}
|
|
getFilters={() => this.filters}
|
|
tableId={this.props.tableId}
|
|
className={this.props.className}
|
|
store={this.props.store}
|
|
getIsReady={() => this.isReady}
|
|
isSelectable={this.props.isSelectable}
|
|
isConfigurable={this.props.isConfigurable}
|
|
copyClassNameFromHeadCells={this.props.copyClassNameFromHeadCells}
|
|
sortingCallbacks={this.props.sortingCallbacks}
|
|
tableProps={this.props.tableProps}
|
|
renderTableHeader={this.props.renderTableHeader}
|
|
renderTableContents={this.props.renderTableContents}
|
|
renderItemMenu={this.props.renderItemMenu}
|
|
customizeTableRowProps={this.props.customizeTableRowProps}
|
|
addRemoveButtons={this.props.addRemoveButtons}
|
|
virtual={this.props.virtual}
|
|
hasDetailsView={this.props.hasDetailsView}
|
|
detailsItem={this.props.detailsItem}
|
|
onDetails={this.props.onDetails}
|
|
customizeRemoveDialog={this.props.customizeRemoveDialog}
|
|
failedToLoadMessage={this.props.failedToLoadMessage}
|
|
/>
|
|
|
|
{this.props.renderFooter?.(this)}
|
|
</div>
|
|
));
|
|
}
|
|
}
|
|
|
|
const InjectedItemListLayout = withInjectables<Dependencies, ItemListLayoutProps<ItemObject>>(NonInjectedItemListLayout, {
|
|
getProps: (di, props) => ({
|
|
namespaceStore: di.inject(namespaceStoreInjectable),
|
|
itemListLayoutStorage: di.inject(itemListLayoutStorageInjectable),
|
|
...props,
|
|
}),
|
|
});
|
|
|
|
export function ItemListLayout<I extends ItemObject>(props: ItemListLayoutProps<I>) {
|
|
return <InjectedItemListLayout {...props} />;
|
|
}
|
|
|
|
function applyFilters<I extends ItemObject>(filters: ItemsFilter<I>[], items: I[]): I[] {
|
|
if (!filters || !filters.length) {
|
|
return items;
|
|
}
|
|
|
|
return filters.reduce((items, filter) => filter(items), items);
|
|
}
|