import difference from 'lodash/difference'; import React, { useState } from 'react'; import { ConfigProvider, Space, Switch, Table, Tag, Transfer } from 'antd'; import type { ColumnsType, TableRowSelection } from 'antd/es/table/interface'; import type { TransferDirection, TransferItem, TransferProps } from 'antd/es/transfer'; interface RecordType { key: string; title: string; description: string; disabled: boolean; tag: string; } interface DataType { key: string; title: string; description: string; disabled: boolean; tag: string; } interface TableTransferProps extends TransferProps { dataSource: DataType[]; leftColumns: ColumnsType; rightColumns: ColumnsType; } // Customize Table Transfer const TableTransfer = ({ leftColumns, rightColumns, ...restProps }: TableTransferProps) => ( {({ direction, filteredItems, onItemSelectAll, onItemSelect, selectedKeys: listSelectedKeys, disabled: listDisabled, }) => { const columns = direction === 'left' ? leftColumns : rightColumns; const rowSelection: TableRowSelection = { getCheckboxProps: (item) => ({ disabled: listDisabled || item.disabled }), onSelectAll(selected, selectedRows) { const treeSelectedKeys = selectedRows .filter((item) => !item.disabled) .map(({ key }) => key); const diffKeys = selected ? difference(treeSelectedKeys, listSelectedKeys) : difference(listSelectedKeys, treeSelectedKeys); onItemSelectAll(diffKeys as string[], selected); }, onSelect({ key }, selected) { onItemSelect(key as string, selected); }, selectedRowKeys: listSelectedKeys, }; return ( ({ onClick: () => { if (itemDisabled || listDisabled) return; onItemSelect(key as string, !listSelectedKeys.includes(key as string)); }, })} /> ); }} ); const mockTags = ['cat', 'dog', 'bird']; const mockData: RecordType[] = Array.from({ length: 20 }).map((_, i) => ({ key: i.toString(), title: `content${i + 1}`, description: `description of content${i + 1}`, disabled: i % 4 === 0, tag: mockTags[i % 3], })); const leftTableColumns: ColumnsType = [ { dataIndex: 'title', title: 'Name', }, { dataIndex: 'tag', title: 'Tag', render: (tag) => {tag}, }, { dataIndex: 'description', title: 'Description', }, ]; const rightTableColumns: ColumnsType> = [ { dataIndex: 'title', title: 'Name', }, ]; const initialTargetKeys = mockData.filter((item) => Number(item.key) > 10).map((item) => item.key); const App: React.FC = () => { const [targetKeys, setTargetKeys] = useState(initialTargetKeys); const [selectedKeys, setSelectedKeys] = useState([]); const onChange = (nextTargetKeys: string[], direction: TransferDirection, moveKeys: string[]) => { console.log('targetKeys:', nextTargetKeys); console.log('direction:', direction); console.log('moveKeys:', moveKeys); setTargetKeys(nextTargetKeys); }; const onSelectChange = (sourceSelectedKeys: string[], targetSelectedKeys: string[]) => { console.log('sourceSelectedKeys:', sourceSelectedKeys); console.log('targetSelectedKeys:', targetSelectedKeys); setSelectedKeys([...sourceSelectedKeys, ...targetSelectedKeys]); }; const onScroll = (direction: TransferDirection, e: React.SyntheticEvent) => { console.log('direction:', direction); console.log('target:', e.target); }; const [disabled, setDisabled] = useState(false); const [showSearch, setShowSearch] = useState(false); const secondOnChange = (nextTargetKeys: string[]) => { setTargetKeys(nextTargetKeys); }; const triggerDisable = (checked: boolean) => { setDisabled(checked); }; const triggerShowSearch = (checked: boolean) => { setShowSearch(checked); }; return ( item.title} /> item.title!.indexOf(inputValue) !== -1 || item.tag.indexOf(inputValue) !== -1 } leftColumns={leftTableColumns} rightColumns={rightTableColumns} /> ); }; export default App;