/** * 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 } from "mobx"; import { Observer, observer } from "mobx-react"; import { ConfirmDialog, ConfirmDialogParams } from "../confirm-dialog"; import { Table, TableCell, TableCellProps, TableHead, TableProps, TableRow, TableRowProps, TableSortCallbacks } from "../table"; import { boundMethod, cssNames, IClassName, isReactNode, prevDefault, stopPropagation } from "../../utils"; import { AddRemoveButtons, AddRemoveButtonsProps } from "../add-remove-buttons"; import { NoItems } from "../no-items"; import { Spinner } from "../spinner"; import type { ItemObject, ItemStore } from "../../../common/item.store"; import { Filter, pageFilters } from "./page-filters.store"; import { ThemeStore } from "../../theme.store"; import { MenuActions } from "../menu/menu-actions"; import { MenuItem } from "../menu"; import { Checkbox } from "../checkbox"; import { UserStore } from "../../../common/user-store"; export interface ItemListLayoutContentProps { getFilters: () => Filter[]; tableId?: string; className: IClassName; getItems: () => I[]; store: ItemStore; getIsReady: () => boolean; // show loading indicator while not ready isSelectable?: boolean; // show checkbox in rows for selecting items isConfigurable?: boolean; copyClassNameFromHeadCells?: boolean; sortingCallbacks?: TableSortCallbacks; tableProps?: Partial>; // low-level table configuration renderTableHeader: TableCellProps[] | null; renderTableContents: (item: I) => (ReactNode | TableCellProps)[]; renderItemMenu?: (item: I, store: ItemStore) => ReactNode; customizeTableRowProps?: (item: I) => Partial; addRemoveButtons?: Partial; virtual?: boolean; // item details view hasDetailsView?: boolean; detailsItem?: I; onDetails?: (item: I) => void; // other customizeRemoveDialog?: (selectedItems: I[]) => Partial; /** * Message to display when a store failed to load * * @default "Failed to load items" */ failedToLoadMessage?: React.ReactNode; } @observer export class ItemListLayoutContent extends React.Component> { constructor(props: ItemListLayoutContentProps) { super(props); makeObservable(this); } @computed get failedToLoad() { return this.props.store.failedLoading; } @boundMethod renderRow(item: I) { return this.getTableRow(item); } getTableRow(item: I) { const { isSelectable, renderTableHeader, renderTableContents, renderItemMenu, store, hasDetailsView, onDetails, copyClassNameFromHeadCells, customizeTableRowProps, detailsItem, } = this.props; const { isSelected } = store; return ( onDetails(item)) : undefined} {...customizeTableRowProps(item)} > {isSelectable && ( store.toggleSelection(item))} /> )} {renderTableContents(item).map((content, index) => { const cellProps: TableCellProps = isReactNode(content) ? { children: content } : content; const headCell = renderTableHeader?.[index]; if (copyClassNameFromHeadCells && headCell) { cellProps.className = cssNames( cellProps.className, headCell.className, ); } if (!headCell || this.showColumn(headCell)) { return ; } return null; })} {renderItemMenu && (
{renderItemMenu(item, store)}
)}
); } @boundMethod getRow(uid: string) { return (
{() => { const item = this.props.getItems().find(item => item.getId() === uid); if (!item) return null; return this.getTableRow(item); }}
); } @boundMethod removeItemsDialog(selectedItems: I[]) { const { customizeRemoveDialog, store } = this.props; const visibleMaxNamesCount = 5; const selectedNames = selectedItems.map(ns => ns.getName()).slice(0, visibleMaxNamesCount).join(", "); const dialogCustomProps = customizeRemoveDialog ? customizeRemoveDialog(selectedItems) : {}; const selectedCount = selectedItems.length; const tailCount = selectedCount > visibleMaxNamesCount ? selectedCount - visibleMaxNamesCount : 0; const tail = tailCount > 0 ? <>, and {tailCount} more : null; const message = selectedCount <= 1 ?

Remove item {selectedNames}?

:

Remove {selectedCount} items {selectedNames}{tail}?

; const onConfirm = store.removeItems ? () => store.removeItems(selectedItems) : store.removeSelectedItems; ConfirmDialog.open({ ok: onConfirm, labelOk: "Remove", message, ...dialogCustomProps, }); } renderNoItems() { if (this.failedToLoad) { return {this.props.failedToLoadMessage}; } if (!this.props.getIsReady()) { return ; } if (this.props.getFilters().length > 0) { return ( No items found.

pageFilters.reset()} className="contrast"> Reset filters?

); } return ; } renderItems() { if (this.props.virtual) { return null; } return this.props.getItems().map(item => this.getRow(item.getId())); } renderTableHeader() { const { customizeTableRowProps, renderTableHeader, isSelectable, isConfigurable, store } = this.props; if (!renderTableHeader) { return null; } const enabledItems = this.props.getItems().filter(item => !customizeTableRowProps(item).disabled); return ( {isSelectable && ( {() => ( store.toggleSelectionAll(enabledItems))} /> )} )} {renderTableHeader.map((cellProps, index) => ( this.showColumn(cellProps) && ( ) ))} {isConfigurable && this.renderColumnVisibilityMenu()} ); } render() { const { store, hasDetailsView, addRemoveButtons = {}, virtual, sortingCallbacks, detailsItem, className, tableProps = {}, tableId, getItems, } = this.props; const selectedItemId = detailsItem && detailsItem.getId(); const classNames = cssNames(className, "box", "grow", ThemeStore.getInstance().activeTheme.type); const items = getItems(); const selectedItems = store.pickOnlySelected(items); return (
{this.renderTableHeader()} {this.renderItems()}
{() => ( 0 ? () => this.removeItemsDialog(selectedItems) : null } removeTooltip={`Remove selected items (${selectedItems.length})`} {...addRemoveButtons} /> )}
); } showColumn({ id: columnId, showWithColumn }: TableCellProps): boolean { const { tableId, isConfigurable } = this.props; return !isConfigurable || !UserStore.getInstance().isTableColumnHidden(tableId, columnId, showWithColumn); } renderColumnVisibilityMenu() { const { renderTableHeader, tableId } = this.props; return ( {renderTableHeader.map((cellProps, index) => ( !cellProps.showWithColumn && ( `} value={this.showColumn(cellProps)} onChange={() => UserStore.getInstance().toggleTableColumnVisibility(tableId, cellProps.id)} /> ) ))} ); } }