import * as React from 'react'; import * as ReactDOM from 'react-dom'; import Menu, { SubMenu, Item as MenuItem } from 'rc-menu'; import closest from 'dom-closest'; import classNames from 'classnames'; import shallowequal from 'shallowequal'; import Dropdown from '../dropdown'; import Icon from '../icon'; import Checkbox from '../checkbox'; import Radio from '../radio'; import FilterDropdownMenuWrapper from './FilterDropdownMenuWrapper'; import { FilterMenuProps, FilterMenuState, ColumnProps, ColumnFilterItem } from './interface'; export default class FilterMenu extends React.Component, FilterMenuState> { static defaultProps = { handleFilter() {}, column: {}, }; neverShown: boolean; constructor(props: FilterMenuProps) { super(props); const visible = ('filterDropdownVisible' in props.column) ? props.column.filterDropdownVisible : false; this.state = { selectedKeys: props.selectedKeys, keyPathOfSelectedItem: {}, // 记录所有有选中子菜单的祖先菜单 visible, }; } componentDidMount() { const { column } = this.props; this.setNeverShown(column); } componentWillReceiveProps(nextProps: FilterMenuProps) { const { column } = nextProps; this.setNeverShown(column); const newState = {} as { selectedKeys: string[]; visible: boolean; }; /** * if the state is visible the component should ignore updates on selectedKeys prop to avoid * that the user selection is lost * this happens frequently when a table is connected on some sort of realtime data * Fixes https://github.com/ant-design/ant-design/issues/10289 and * https://github.com/ant-design/ant-design/issues/10209 */ if ('selectedKeys' in nextProps && !shallowequal(this.props.selectedKeys, nextProps.selectedKeys)) { newState.selectedKeys = nextProps.selectedKeys; } if ('filterDropdownVisible' in column) { newState.visible = column.filterDropdownVisible as boolean; } if (Object.keys(newState).length > 0) { this.setState(newState); } } setNeverShown = (column: ColumnProps) => { const rootNode = ReactDOM.findDOMNode(this); const filterBelongToScrollBody = !!closest(rootNode, `.ant-table-scroll`); if (filterBelongToScrollBody) { // When fixed column have filters, there will be two dropdown menus // Filter dropdown menu inside scroll body should never be shown // To fix https://github.com/ant-design/ant-design/issues/5010 and // https://github.com/ant-design/ant-design/issues/7909 this.neverShown = !!column.fixed; } } setSelectedKeys = ({ selectedKeys }: { selectedKeys: string[] }) => { this.setState({ selectedKeys }); } setVisible(visible: boolean) { const { column } = this.props; if (!('filterDropdownVisible' in column)) { this.setState({ visible }); } if (column.onFilterDropdownVisibleChange) { column.onFilterDropdownVisibleChange(visible); } } handleClearFilters = () => { this.setState({ selectedKeys: [], }, this.handleConfirm); } handleConfirm = () => { this.setVisible(false); // Call `setSelectedKeys` & `confirm` in the same time will make filter data not up to date // https://github.com/ant-design/ant-design/issues/12284 this.setState({}, this.confirmFilter); } onVisibleChange = (visible: boolean) => { this.setVisible(visible); if (!visible) { this.confirmFilter(); } } confirmFilter() { const { selectedKeys } = this.state; if (!shallowequal(selectedKeys, this.props.selectedKeys)) { this.props.confirmFilter(this.props.column, selectedKeys); } } renderMenuItem(item: ColumnFilterItem) { const { column } = this.props; const { selectedKeys } = this.state; const multiple = ('filterMultiple' in column) ? column.filterMultiple : true; const input = multiple ? = 0} /> : = 0} />; return ( {input} {item.text} ); } hasSubMenu() { const { column: { filters = [] } } = this.props; return filters.some(item => !!(item.children && item.children.length > 0)); } renderMenus(items: ColumnFilterItem[]): React.ReactElement[] { return items.map(item => { if (item.children && item.children.length > 0) { const { keyPathOfSelectedItem } = this.state; const containSelected = Object.keys(keyPathOfSelectedItem).some( key => keyPathOfSelectedItem[key].indexOf(item.value) >= 0, ); const subMenuCls = containSelected ? `${this.props.dropdownPrefixCls}-submenu-contain-selected` : ''; return ( {this.renderMenus(item.children)} ); } return this.renderMenuItem(item); }); } handleMenuItemClick = (info: { keyPath: string, key: string }) => { if (!info.keyPath || info.keyPath.length <= 1) { return; } const keyPathOfSelectedItem = this.state.keyPathOfSelectedItem; if (this.state.selectedKeys.indexOf(info.key) >= 0) { // deselect SubMenu child delete keyPathOfSelectedItem[info.key]; } else { // select SubMenu child keyPathOfSelectedItem[info.key] = info.keyPath; } this.setState({ keyPathOfSelectedItem }); } renderFilterIcon = () => { const { column, locale, prefixCls } = this.props; const filterd = this.props.selectedKeys.length > 0; let filterIcon = column.filterIcon as any; if (typeof filterIcon === 'function') { filterIcon = filterIcon(filterd); } const dropdownSelectedClass = filterd ? `${prefixCls}-selected` : ''; return filterIcon ? React.cloneElement(filterIcon as any, { title: locale.filterTitle, className: classNames(`${prefixCls}-icon`, filterIcon.props.className), }) : ; } render() { const { column, locale, prefixCls, dropdownPrefixCls, getPopupContainer } = this.props; // default multiple selection in filter dropdown const multiple = ('filterMultiple' in column) ? column.filterMultiple : true; const dropdownMenuClass = classNames({ [`${dropdownPrefixCls}-menu-without-submenu`]: !this.hasSubMenu(), }); let { filterDropdown } = column; if (typeof filterDropdown === 'function') { filterDropdown = (filterDropdown as Function)({ prefixCls: `${dropdownPrefixCls}-custom`, setSelectedKeys: (selectedKeys: Array) => this.setSelectedKeys({ selectedKeys }), selectedKeys: this.state.selectedKeys, confirm: this.handleConfirm, clearFilters: this.handleClearFilters, filters: column.filters, getPopupContainer: (triggerNode: HTMLElement) => triggerNode.parentNode, }); } const menus = filterDropdown ? ( {filterDropdown} ) : ( triggerNode.parentNode} > {this.renderMenus(column.filters!)} ); return ( {this.renderFilterIcon()} ); } }