import React, { useMemo, useRef, useState } from 'react'; import DownOutlined from '@ant-design/icons/DownOutlined'; import classNames from 'classnames'; import omit from 'rc-util/lib/omit'; import { groupKeysMap } from '../_util/transKeys'; import Checkbox from '../checkbox'; import Dropdown from '../dropdown'; import type { MenuProps } from '../menu'; import type { KeyWiseTransferItem, RenderResult, RenderResultObject, SelectAllLabel, TransferDirection, TransferLocale, TransferSearchOption, } from './index'; import type { PaginationType, TransferKey } from './interface'; import type { ListBodyRef, TransferListBodyProps } from './ListBody'; import DefaultListBody, { OmitProps } from './ListBody'; import Search from './search'; const defaultRender = () => null; function isRenderResultPlainObject(result: RenderResult): result is RenderResultObject { return !!( result && !React.isValidElement(result) && Object.prototype.toString.call(result) === '[object Object]' ); } function getEnabledItemKeys(items: RecordType[]) { return items.filter((data) => !data.disabled).map((data) => data.key); } const isValidIcon = (icon: React.ReactNode) => icon !== undefined; export interface RenderedItem { renderedText: string; renderedEl: React.ReactNode; item: RecordType; } type RenderListFunction = (props: TransferListBodyProps) => React.ReactNode; export interface TransferListProps extends TransferLocale { prefixCls: string; titleText: React.ReactNode; dataSource: RecordType[]; filterOption?: (filterText: string, item: RecordType, direction: TransferDirection) => boolean; style?: React.CSSProperties; checkedKeys: TransferKey[]; handleFilter: (e: React.ChangeEvent) => void; onItemSelect: ( key: TransferKey, check: boolean, e?: React.MouseEvent, ) => void; onItemSelectAll: (dataSource: TransferKey[], checkAll: boolean | 'replace') => void; onItemRemove?: (keys: TransferKey[]) => void; handleClear: () => void; /** Render item */ render?: (item: RecordType) => RenderResult; showSearch?: boolean | TransferSearchOption; searchPlaceholder: string; itemUnit: string; itemsUnit: string; renderList?: RenderListFunction; footer?: ( props: TransferListProps, info?: { direction: TransferDirection }, ) => React.ReactNode; onScroll: (e: React.UIEvent) => void; disabled?: boolean; direction: TransferDirection; showSelectAll?: boolean; selectAllLabel?: SelectAllLabel; showRemove?: boolean; pagination?: PaginationType; selectionsIcon?: React.ReactNode; } export interface TransferCustomListBodyProps extends TransferListBodyProps {} const useShowSearchOption = (showSearch: boolean | TransferSearchOption) => { if (showSearch && typeof showSearch === 'object') { return { ...showSearch, defaultValue: showSearch.defaultValue || '', }; } return { defaultValue: '', placeholder: '', }; }; const TransferList = ( props: TransferListProps, ) => { const { prefixCls, dataSource = [], titleText = '', checkedKeys, disabled, showSearch = false, style, searchPlaceholder, notFoundContent, selectAll, deselectAll, selectCurrent, selectInvert, removeAll, removeCurrent, showSelectAll = true, showRemove, pagination, direction, itemsUnit, itemUnit, selectAllLabel, selectionsIcon, footer, renderList, onItemSelectAll, onItemRemove, handleFilter, handleClear, filterOption, render = defaultRender, } = props; const searchOptions = useShowSearchOption(showSearch); const [filterValue, setFilterValue] = useState(searchOptions.defaultValue); const listBodyRef = useRef>({}); const internalHandleFilter = (e: React.ChangeEvent) => { setFilterValue(e.target.value); handleFilter(e); }; const internalHandleClear = () => { setFilterValue(''); handleClear(); }; const matchFilter = (text: string, item: RecordType) => { if (filterOption) { return filterOption(filterValue, item, direction); } return text.includes(filterValue); }; const renderListBody = (listProps: TransferListBodyProps) => { let bodyContent: React.ReactNode = renderList ? renderList({ ...listProps, onItemSelect: (key, check) => listProps.onItemSelect(key, check), }) : null; const customize: boolean = !!bodyContent; if (!customize) { // @ts-ignore bodyContent = ; } return { customize, bodyContent }; }; const renderItem = (item: RecordType): RenderedItem => { const renderResult = render(item); const isRenderResultPlain = isRenderResultPlainObject(renderResult); return { item, renderedEl: isRenderResultPlain ? renderResult.label : renderResult, renderedText: isRenderResultPlain ? renderResult.value : (renderResult as string), }; }; const notFoundContentEle = useMemo( () => Array.isArray(notFoundContent) ? notFoundContent[direction === 'left' ? 0 : 1] : notFoundContent, [notFoundContent, direction], ); const [filteredItems, filteredRenderItems] = useMemo(() => { const filterItems: RecordType[] = []; const filterRenderItems: RenderedItem[] = []; dataSource.forEach((item) => { const renderedItem = renderItem(item); if (filterValue && !matchFilter(renderedItem.renderedText, item)) { return; } filterItems.push(item); filterRenderItems.push(renderedItem); }); return [filterItems, filterRenderItems] as const; }, [dataSource, filterValue]); const checkedActiveItems = useMemo(() => { return filteredItems.filter((item) => checkedKeys.includes(item.key) && !item.disabled); }, [checkedKeys, filteredItems]); const checkStatus = useMemo(() => { if (checkedActiveItems.length === 0) { return 'none'; } const checkedKeysMap = groupKeysMap(checkedKeys); if (filteredItems.every((item) => checkedKeysMap.has(item.key) || !!item.disabled)) { return 'all'; } return 'part'; }, [checkedKeys, checkedActiveItems]); const listBody = useMemo(() => { const search = showSearch ? (
) : null; const { customize, bodyContent } = renderListBody({ ...omit(props, OmitProps), filteredItems, filteredRenderItems, selectedKeys: checkedKeys, }); let bodyNode: React.ReactNode; // We should wrap customize list body in a classNamed div to use flex layout. if (customize) { bodyNode =
{bodyContent}
; } else { bodyNode = filteredItems.length ? ( bodyContent ) : (
{notFoundContentEle}
); } return (
{search} {bodyNode}
); }, [ showSearch, prefixCls, searchPlaceholder, filterValue, disabled, checkedKeys, filteredItems, filteredRenderItems, notFoundContentEle, ]); const checkBox = ( !d.disabled).length === 0 || disabled} checked={checkStatus === 'all'} indeterminate={checkStatus === 'part'} className={`${prefixCls}-checkbox`} onChange={() => { // Only select enabled items onItemSelectAll?.( filteredItems.filter((item) => !item.disabled).map(({ key }) => key), checkStatus !== 'all', ); }} /> ); const getSelectAllLabel = (selectedCount: number, totalCount: number): React.ReactNode => { if (selectAllLabel) { return typeof selectAllLabel === 'function' ? selectAllLabel({ selectedCount, totalCount }) : selectAllLabel; } const unit = totalCount > 1 ? itemsUnit : itemUnit; return ( <> {(selectedCount > 0 ? `${selectedCount}/` : '') + totalCount} {unit} ); }; // Custom Layout const footerDom = footer && (footer.length < 2 ? footer(props) : footer(props, { direction })); const listCls = classNames(prefixCls, { [`${prefixCls}-with-pagination`]: !!pagination, [`${prefixCls}-with-footer`]: !!footerDom, }); // ====================== Get filtered, checked item list ====================== const listFooter = footerDom ?
{footerDom}
: null; const checkAllCheckbox = !showRemove && !pagination && checkBox; let items: MenuProps['items']; if (showRemove) { items = [ /* Remove Current Page */ pagination ? { key: 'removeCurrent', label: removeCurrent, onClick() { const pageKeys = getEnabledItemKeys( (listBodyRef.current?.items || []).map((entity) => entity.item), ); onItemRemove?.(pageKeys); }, } : null, /* Remove All */ { key: 'removeAll', label: removeAll, onClick() { onItemRemove?.(getEnabledItemKeys(filteredItems)); }, }, ].filter(Boolean); } else { items = [ { key: 'selectAll', label: checkStatus === 'all' ? deselectAll : selectAll, onClick() { const keys = getEnabledItemKeys(filteredItems); onItemSelectAll?.(keys, keys.length !== checkedKeys.length); }, }, pagination ? { key: 'selectCurrent', label: selectCurrent, onClick() { const pageItems = listBodyRef.current?.items || []; onItemSelectAll?.(getEnabledItemKeys(pageItems.map((entity) => entity.item)), true); }, } : null, { key: 'selectInvert', label: selectInvert, onClick() { const availablePageItemKeys = getEnabledItemKeys( (listBodyRef.current?.items || []).map((entity) => entity.item), ); const checkedKeySet = new Set(checkedKeys); const newCheckedKeysSet = new Set(checkedKeySet); availablePageItemKeys.forEach((key) => { if (checkedKeySet.has(key)) { newCheckedKeysSet.delete(key); } else { newCheckedKeysSet.add(key); } }); onItemSelectAll?.(Array.from(newCheckedKeysSet), 'replace'); }, }, ]; } const dropdown: React.ReactNode = ( {isValidIcon(selectionsIcon) ? selectionsIcon : } ); return (
{/* Header */}
{showSelectAll ? ( <> {checkAllCheckbox} {dropdown} ) : null} {getSelectAllLabel(checkedActiveItems.length, filteredItems.length)} {titleText}
{listBody} {listFooter}
); }; if (process.env.NODE_ENV !== 'production') { TransferList.displayName = 'TransferList'; } export default TransferList;