mirror of
https://gitee.com/ant-design/ant-design.git
synced 2024-12-03 04:30:06 +08:00
136 lines
3.9 KiB
JavaScript
136 lines
3.9 KiB
JavaScript
import React from 'react';
|
|
import Menu, { SubMenu, Item as MenuItem } from 'rc-menu';
|
|
import Dropdown from '../dropdown';
|
|
import Icon from '../icon';
|
|
import Checkbox from '../checkbox';
|
|
|
|
let FilterMenu = React.createClass({
|
|
getInitialState() {
|
|
return {
|
|
selectedKeys: this.props.selectedKeys,
|
|
keyPathOfSelectedItem: {}, // 记录所有有选中子菜单的祖先菜单
|
|
visible: false,
|
|
};
|
|
},
|
|
componentWillReceiveProps(nextProps) {
|
|
this.setState({
|
|
selectedKeys: nextProps.selectedKeys
|
|
});
|
|
},
|
|
getDefaultProps() {
|
|
return {
|
|
handleFilter() {},
|
|
column: null
|
|
};
|
|
},
|
|
setSelectedKeys({ selectedKeys }) {
|
|
this.setState({ selectedKeys });
|
|
},
|
|
handleClearFilters() {
|
|
this.setState({
|
|
selectedKeys: []
|
|
}, this.handleConfirm);
|
|
},
|
|
handleConfirm() {
|
|
this.setState({
|
|
visible: false
|
|
});
|
|
this.props.confirmFilter(this.props.column, this.state.selectedKeys);
|
|
},
|
|
onVisibleChange(visible) {
|
|
this.setState({
|
|
visible,
|
|
});
|
|
if (!visible) {
|
|
this.props.confirmFilter(this.props.column, this.state.selectedKeys);
|
|
}
|
|
},
|
|
renderMenuItem(item) {
|
|
return (
|
|
<MenuItem key={item.value}>
|
|
<Checkbox checked={this.state.selectedKeys.indexOf(item.value) >= 0} />
|
|
{item.text}
|
|
</MenuItem>
|
|
);
|
|
},
|
|
renderMenus(items) {
|
|
let menuItems = items.map(item => {
|
|
if (item.children && item.children.length > 0) {
|
|
const keyPathOfSelectedItem = this.state.keyPathOfSelectedItem;
|
|
const containSelected = Object.keys(keyPathOfSelectedItem).some(key => {
|
|
const keyPath = keyPathOfSelectedItem[key];
|
|
return keyPath.indexOf(item.value) >= 0;
|
|
});
|
|
const subMenuCls = containSelected ? 'ant-dropdown-submenu-contain-selected' : '';
|
|
return (
|
|
<SubMenu title={item.text} className={subMenuCls} key={item.value}>
|
|
{item.children.map(child => this.renderMenuItem(child))}
|
|
</SubMenu>
|
|
);
|
|
}
|
|
return this.renderMenuItem(item);
|
|
});
|
|
return menuItems;
|
|
},
|
|
handleMenuItemClick(info) {
|
|
if (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 });
|
|
},
|
|
render() {
|
|
let { column, locale } = this.props;
|
|
// default multiple selection in filter dropdown
|
|
let multiple = true;
|
|
if ('filterMultiple' in column) {
|
|
multiple = column.filterMultiple;
|
|
}
|
|
let menus = (
|
|
<div className="ant-table-filter-dropdown">
|
|
<Menu multiple={multiple}
|
|
prefixCls="ant-dropdown-menu"
|
|
onSelect={this.setSelectedKeys}
|
|
onDeselect={this.setSelectedKeys}
|
|
selectedKeys={this.state.selectedKeys}>
|
|
{this.renderMenus(column.filters)}
|
|
</Menu>
|
|
<div className="ant-table-filter-dropdown-btns">
|
|
<a className="ant-table-filter-dropdown-link confirm"
|
|
onClick={this.handleConfirm}>
|
|
{locale.filterConfirm}
|
|
</a>
|
|
<a className="ant-table-filter-dropdown-link clear"
|
|
onClick={this.handleClearFilters}>
|
|
{locale.filterReset}
|
|
</a>
|
|
</div>
|
|
</div>
|
|
);
|
|
|
|
let dropdownSelectedClass = '';
|
|
if (this.props.selectedKeys.length > 0) {
|
|
dropdownSelectedClass = 'ant-table-filter-selected';
|
|
}
|
|
|
|
return (
|
|
<Dropdown trigger={['click']}
|
|
overlay={menus}
|
|
visible={this.state.visible}
|
|
onVisibleChange={this.onVisibleChange}
|
|
closeOnSelect={false}>
|
|
<Icon title={locale.filterTitle} type="filter" className={dropdownSelectedClass} />
|
|
</Dropdown>
|
|
);
|
|
}
|
|
});
|
|
|
|
export default FilterMenu;
|