import { provide, inject, defineComponent, App, Plugin, ExtractPropTypes } from 'vue'; import omit from 'omit.js'; import PropTypes, { withUndefined } from '../_util/vue-types'; import classNames from '../_util/classNames'; import { defaultConfigProvider } from '../config-provider'; import Spin from '../spin'; import Pagination, { PaginationConfig } from '../pagination'; import { Row } from '../grid'; import Item, { ListItemMeta } from './Item'; import { getComponent, getSlot } from '../_util/props-util'; import initDefaultProps from '../_util/props-util/initDefaultProps'; import { cloneElement } from '../_util/vnode'; import { tuple } from '../_util/type'; export { ListItemProps, ListItemMetaProps, ListItemMeta } from './Item'; export const ColumnCount = ['', 1, 2, 3, 4, 6, 8, 12, 24]; export const ColumnType = ['gutter', 'column', 'xs', 'sm', 'md', 'lg', 'xl', 'xxl']; export const ListGridType = { gutter: PropTypes.number, column: PropTypes.oneOf(ColumnCount), xs: PropTypes.oneOf(ColumnCount), sm: PropTypes.oneOf(ColumnCount), md: PropTypes.oneOf(ColumnCount), lg: PropTypes.oneOf(ColumnCount), xl: PropTypes.oneOf(ColumnCount), xxl: PropTypes.oneOf(ColumnCount), }; export const ListSize = tuple('small', 'default', 'large'); const paginationProps = PaginationConfig(); export const ListProps = () => ({ bordered: PropTypes.looseBool, dataSource: PropTypes.array, extra: PropTypes.any, grid: PropTypes.shape(ListGridType).loose, itemLayout: PropTypes.string, loading: withUndefined(PropTypes.oneOfType([PropTypes.looseBool, PropTypes.object])), loadMore: PropTypes.any, pagination: withUndefined( PropTypes.oneOfType([ PropTypes.shape>>(paginationProps).loose, PropTypes.looseBool, ]), ), prefixCls: PropTypes.string, rowKey: PropTypes.any, renderItem: PropTypes.any, size: PropTypes.oneOf(ListSize), split: PropTypes.looseBool, header: PropTypes.any, footer: PropTypes.any, locale: PropTypes.object, }); const List = defineComponent({ name: 'AList', inheritAttrs: false, Item, props: initDefaultProps(ListProps(), { dataSource: [], bordered: false, split: true, loading: false, pagination: false, }), setup() { return { keys: [], defaultPaginationProps: {}, onPaginationChange: null, onPaginationShowSizeChange: null, configProvider: inject('configProvider', defaultConfigProvider), }; }, data() { const { pagination } = this.$props; const paginationObj: Partial = pagination && typeof pagination === 'object' ? pagination : {}; return { paginationCurrent: paginationObj.defaultCurrent || 1, paginationSize: paginationObj.defaultPageSize || 10, }; }, created() { provide('listContext', this); this.defaultPaginationProps = { current: 1, pageSize: 10, onChange: (page: number, pageSize: number) => { const { pagination } = this; this.paginationCurrent = page; if (pagination && (pagination as any).onChange) { (pagination as any).onChange(page, pageSize); } }, total: 0, }; this.onPaginationChange = this.triggerPaginationEvent('onChange'); this.onPaginationShowSizeChange = this.triggerPaginationEvent('onShowSizeChange'); }, methods: { triggerPaginationEvent(eventName) { return (page, pageSize) => { const { pagination } = this.$props; this.paginationCurrent = page; this.paginationSize = pageSize; if (pagination && pagination[eventName]) { pagination[eventName](page, pageSize); } }; }, innerRenderItem(item, index) { const { $slots: { renderItem }, rowKey, } = this; const renderer = this.renderItem || renderItem; if (!renderer) return null; let key; if (typeof rowKey === 'function') { key = rowKey(item); } else if (typeof rowKey === 'string') { key = item[rowKey]; } else { key = item.key; } if (!key) { key = `list-item-${index}`; } this.keys[index] = key; return renderer({ item, index }); }, isSomethingAfterLastItem() { const { pagination } = this; const loadMore = getComponent(this, 'loadMore'); const footer = getComponent(this, 'footer'); return !!(loadMore || pagination || footer); }, renderEmpty(prefixCls, renderEmpty) { const { locale } = this; return (
{(locale && locale.emptyText) || renderEmpty('List')}
); }, }, render() { const { prefixCls: customizePrefixCls, bordered, split, itemLayout, pagination, grid, dataSource = [], size, loading, paginationCurrent, paginationSize, $attrs, } = this; const { getPrefixCls } = this.configProvider; const prefixCls = getPrefixCls('list', customizePrefixCls); const { class: className, ...restAttrs } = $attrs; const loadMore = getComponent(this, 'loadMore'); const footer = getComponent(this, 'footer'); const header = getComponent(this, 'header'); const children = getSlot(this); let loadingProp = loading; if (typeof loadingProp === 'boolean') { loadingProp = { spinning: loadingProp, }; } const isLoading = loadingProp && loadingProp.spinning; // large => lg // small => sm let sizeCls = ''; switch (size) { case 'large': sizeCls = 'lg'; break; case 'small': sizeCls = 'sm'; break; default: break; } const classString = classNames( prefixCls, { [`${prefixCls}-vertical`]: itemLayout === 'vertical', [`${prefixCls}-${sizeCls}`]: sizeCls, [`${prefixCls}-split`]: split, [`${prefixCls}-bordered`]: bordered, [`${prefixCls}-loading`]: isLoading, [`${prefixCls}-grid`]: grid, [`${prefixCls}-something-after-last-item`]: this.isSomethingAfterLastItem(), }, className, ); const paginationProps = { ...this.defaultPaginationProps, total: dataSource.length, current: paginationCurrent, pageSize: paginationSize, ...((pagination as any) || {}), }; classString; const largestPage = Math.ceil(paginationProps.total / paginationProps.pageSize); if (paginationProps.current > largestPage) { paginationProps.current = largestPage; } const { class: cls, style, ...restProps } = paginationProps; const paginationContent = pagination ? (
) : null; let splitDataSource = [...dataSource]; if (pagination) { if (dataSource.length > (paginationProps.current - 1) * paginationProps.pageSize) { splitDataSource = [...dataSource].splice( (paginationProps.current - 1) * paginationProps.pageSize, paginationProps.pageSize, ); } } let childrenContent; childrenContent = isLoading &&
; if (splitDataSource.length > 0) { const items = splitDataSource.map((item, index) => this.innerRenderItem(item, index)); const childrenList = items.map((child, index) => cloneElement(child, { key: this.keys[index], }), ); childrenContent = grid ? ( {childrenList} ) : (
    {childrenList}
); } else if (!children.length && !isLoading) { const renderEmpty = this.configProvider.renderEmpty; childrenContent = this.renderEmpty(prefixCls, renderEmpty); } const paginationPosition = paginationProps.position || 'bottom'; return (
{(paginationPosition === 'top' || paginationPosition === 'both') && paginationContent} {header &&
{header}
} {childrenContent} {children} {footer &&
{footer}
} {loadMore || ((paginationPosition === 'bottom' || paginationPosition === 'both') && paginationContent)}
); }, }); /* istanbul ignore next */ List.install = function(app: App) { app.component(List.name, List); app.component(List.Item.name, List.Item); app.component(List.Item.Meta.displayName, List.Item.Meta); return app; }; export default List as typeof List & Plugin & { readonly Item: typeof Item & { readonly Meta: typeof ListItemMeta; }; };