2017-08-09 15:45:08 +08:00
|
|
|
import React, { Component } from 'react';
|
2017-09-06 15:53:25 +08:00
|
|
|
import PropTypes from 'prop-types';
|
2017-08-09 15:45:08 +08:00
|
|
|
import classNames from 'classnames';
|
2017-11-17 11:25:29 +08:00
|
|
|
import LocaleReceiver from '../locale-provider/LocaleReceiver';
|
|
|
|
import defaultLocale from '../locale-provider/default';
|
2017-08-09 15:45:08 +08:00
|
|
|
|
|
|
|
import Spin from '../spin';
|
|
|
|
import Pagination from '../pagination';
|
2017-08-14 11:08:19 +08:00
|
|
|
import { Row } from '../grid';
|
2017-08-09 15:45:08 +08:00
|
|
|
|
|
|
|
import Item from './Item';
|
|
|
|
|
2017-09-25 22:14:49 +08:00
|
|
|
export { ListItemProps, ListItemMetaProps } from './Item';
|
|
|
|
|
2017-10-10 10:04:03 +08:00
|
|
|
export type ColumnType = 1 | 2 | 3 | 4 | 6 | 8 | 12 | 24;
|
|
|
|
|
2017-08-14 22:25:17 +08:00
|
|
|
export interface ListGridType {
|
2017-08-14 11:08:19 +08:00
|
|
|
gutter?: number;
|
2017-10-10 10:04:03 +08:00
|
|
|
column?: ColumnType;
|
|
|
|
xs?: ColumnType;
|
|
|
|
sm?: ColumnType;
|
|
|
|
md?: ColumnType;
|
|
|
|
lg?: ColumnType;
|
|
|
|
xl?: ColumnType;
|
2017-08-14 11:08:19 +08:00
|
|
|
}
|
|
|
|
|
2017-09-26 16:11:35 +08:00
|
|
|
export type ListSize = 'small' | 'default' | 'large';
|
|
|
|
|
2017-08-09 15:45:08 +08:00
|
|
|
export interface ListProps {
|
|
|
|
bordered?: boolean;
|
|
|
|
className?: string;
|
|
|
|
children?: React.ReactNode;
|
2017-09-26 16:11:35 +08:00
|
|
|
dataSource: any;
|
2017-08-09 15:45:08 +08:00
|
|
|
extra?: React.ReactNode;
|
2017-09-26 16:11:35 +08:00
|
|
|
grid?: ListGridType;
|
2017-08-09 15:45:08 +08:00
|
|
|
id?: string;
|
2017-09-26 16:11:35 +08:00
|
|
|
itemLayout?: string;
|
2017-08-09 15:45:08 +08:00
|
|
|
loading?: boolean;
|
2017-09-25 15:24:16 +08:00
|
|
|
loadMore?: React.ReactNode;
|
2017-08-09 15:45:08 +08:00
|
|
|
pagination?: any;
|
|
|
|
prefixCls?: string;
|
2017-09-25 15:24:16 +08:00
|
|
|
rowKey?: any;
|
|
|
|
renderItem: any;
|
2017-09-26 16:11:35 +08:00
|
|
|
size?: ListSize;
|
|
|
|
split?: boolean;
|
|
|
|
header?: React.ReactNode;
|
|
|
|
footer?: React.ReactNode;
|
2017-11-17 11:25:29 +08:00
|
|
|
locale?: Object;
|
2017-08-09 15:45:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export default class List extends Component<ListProps> {
|
|
|
|
static Item: typeof Item = Item;
|
|
|
|
|
2017-09-06 15:53:25 +08:00
|
|
|
static childContextTypes = {
|
|
|
|
grid: PropTypes.any,
|
|
|
|
};
|
|
|
|
|
2017-11-17 11:25:29 +08:00
|
|
|
static defaultProps = {
|
|
|
|
dataSource: [],
|
|
|
|
prefixCls: 'ant-list',
|
|
|
|
bordered: false,
|
|
|
|
split: true,
|
|
|
|
loading: false,
|
|
|
|
pagination: false,
|
|
|
|
};
|
|
|
|
|
2017-09-25 15:24:16 +08:00
|
|
|
private keys = {};
|
2017-09-17 15:48:44 +08:00
|
|
|
|
2017-09-06 15:53:25 +08:00
|
|
|
getChildContext() {
|
|
|
|
return {
|
|
|
|
grid: this.props.grid,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-09-25 15:24:16 +08:00
|
|
|
renderItem = (item, index) => {
|
|
|
|
const { dataSource, renderItem, rowKey } = this.props;
|
|
|
|
let key;
|
2017-09-06 15:53:25 +08:00
|
|
|
|
2017-09-25 15:24:16 +08:00
|
|
|
if (typeof rowKey === 'function') {
|
|
|
|
key = rowKey(dataSource[index]);
|
|
|
|
} else if (typeof rowKey === 'string') {
|
|
|
|
key = dataSource[rowKey];
|
|
|
|
} else {
|
|
|
|
key = dataSource.key;
|
2017-09-06 15:53:25 +08:00
|
|
|
}
|
|
|
|
|
2017-09-25 15:24:16 +08:00
|
|
|
if (!key) {
|
|
|
|
key = `list-item-${index}`;
|
2017-09-06 15:53:25 +08:00
|
|
|
}
|
|
|
|
|
2017-09-25 15:24:16 +08:00
|
|
|
this.keys[index] = key;
|
2017-09-06 15:53:25 +08:00
|
|
|
|
2017-09-25 15:24:16 +08:00
|
|
|
return renderItem(item, index);
|
2017-09-06 15:53:25 +08:00
|
|
|
}
|
|
|
|
|
2017-10-26 15:36:48 +08:00
|
|
|
isSomethingAfterLastTtem() {
|
|
|
|
const { loadMore, pagination, footer } = this.props;
|
|
|
|
return !!(loadMore || pagination || footer);
|
|
|
|
}
|
|
|
|
|
2017-11-17 11:25:29 +08:00
|
|
|
renderEmpty = (contextLocale) => {
|
|
|
|
const locale = { ...contextLocale, ...this.props.locale };
|
|
|
|
return <div className={`${this.props.prefixCls}-empty-text`}>{locale.emptyText}</div>;
|
|
|
|
}
|
|
|
|
|
2017-08-09 15:45:08 +08:00
|
|
|
render() {
|
|
|
|
const {
|
2017-11-17 11:25:29 +08:00
|
|
|
bordered,
|
|
|
|
split,
|
2017-08-09 15:45:08 +08:00
|
|
|
className,
|
|
|
|
children,
|
2017-11-17 11:25:29 +08:00
|
|
|
loading,
|
2017-08-09 15:45:08 +08:00
|
|
|
itemLayout,
|
2017-09-25 15:24:16 +08:00
|
|
|
loadMore,
|
2017-11-17 11:25:29 +08:00
|
|
|
pagination,
|
|
|
|
prefixCls,
|
2017-08-14 11:08:19 +08:00
|
|
|
grid,
|
2017-11-17 11:25:29 +08:00
|
|
|
dataSource,
|
2017-09-26 16:11:35 +08:00
|
|
|
size,
|
2017-09-25 15:24:16 +08:00
|
|
|
rowKey,
|
|
|
|
renderItem,
|
2017-09-26 16:11:35 +08:00
|
|
|
header,
|
|
|
|
footer,
|
2017-08-30 15:16:48 +08:00
|
|
|
...rest,
|
2017-08-21 21:14:20 +08:00
|
|
|
} = this.props;
|
2017-08-09 15:45:08 +08:00
|
|
|
|
2017-09-26 16:11:35 +08:00
|
|
|
// large => lg
|
|
|
|
// small => sm
|
|
|
|
let sizeCls = '';
|
|
|
|
switch (size) {
|
|
|
|
case 'large':
|
|
|
|
sizeCls = 'lg';
|
|
|
|
break;
|
|
|
|
case 'small':
|
|
|
|
sizeCls = 'sm';
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-08-09 15:45:08 +08:00
|
|
|
const classString = classNames(prefixCls, className, {
|
|
|
|
[`${prefixCls}-vertical`]: itemLayout === 'vertical',
|
2017-09-26 16:11:35 +08:00
|
|
|
[`${prefixCls}-${sizeCls}`]: sizeCls,
|
|
|
|
[`${prefixCls}-split`]: split,
|
2017-08-09 15:45:08 +08:00
|
|
|
[`${prefixCls}-bordered`]: bordered,
|
|
|
|
[`${prefixCls}-loading`]: loading,
|
2017-08-14 11:08:19 +08:00
|
|
|
[`${prefixCls}-grid`]: grid,
|
2017-10-26 15:36:48 +08:00
|
|
|
[`${prefixCls}-something-after-last-item`]: this.isSomethingAfterLastTtem(),
|
2017-08-09 15:45:08 +08:00
|
|
|
});
|
|
|
|
|
|
|
|
const paginationContent = (
|
|
|
|
<div className={`${prefixCls}-pagination`}>
|
|
|
|
<Pagination {...pagination} />
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
2017-11-17 11:25:29 +08:00
|
|
|
let childrenContent;
|
|
|
|
if (dataSource.length > 0) {
|
|
|
|
const childrenList = React.Children.map(dataSource.map((item: any, index) => this.renderItem(item, index)),
|
|
|
|
(child: any, index) => React.cloneElement(child, {
|
|
|
|
key: this.keys[index],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
childrenContent = grid ? (
|
|
|
|
<Row gutter={grid.gutter}>{childrenList}</Row>
|
|
|
|
) : childrenList;
|
|
|
|
} else if (!children) {
|
|
|
|
childrenContent = (
|
|
|
|
<LocaleReceiver
|
|
|
|
componentName="Table"
|
|
|
|
defaultLocale={defaultLocale.Table}
|
|
|
|
>
|
|
|
|
{this.renderEmpty}
|
|
|
|
</LocaleReceiver>
|
|
|
|
);
|
|
|
|
}
|
2017-08-14 11:08:19 +08:00
|
|
|
|
2017-10-26 15:36:48 +08:00
|
|
|
const content = (
|
2017-09-06 15:53:25 +08:00
|
|
|
<div>
|
2017-10-26 15:36:48 +08:00
|
|
|
<Spin spinning={loading}>{childrenContent}</Spin>
|
2017-09-25 15:24:16 +08:00
|
|
|
{loadMore}
|
2017-10-26 15:36:48 +08:00
|
|
|
{(!loadMore && pagination) ? paginationContent : null}
|
2017-09-06 15:53:25 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
2017-09-25 15:24:16 +08:00
|
|
|
<div className={classString} {...rest}>
|
2017-09-26 16:11:35 +08:00
|
|
|
{header && <div className={`${prefixCls}-header`}>{header}</div>}
|
2017-09-06 15:53:25 +08:00
|
|
|
{content}
|
2017-09-25 15:24:16 +08:00
|
|
|
{children}
|
2017-09-26 16:11:35 +08:00
|
|
|
{footer && <div className={`${prefixCls}-footer`}>{footer}</div>}
|
2017-08-09 15:45:08 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|