mirror of
https://github.com/ant-design/ant-design.git
synced 2024-12-05 09:49:57 +08:00
502dac12aa
* docs: fix code * feat: lint * feat: prettier * feat: test * feat: review * feat: format html * feat: format html
96 lines
2.7 KiB
TypeScript
96 lines
2.7 KiB
TypeScript
import * as React from 'react';
|
|
import LoadingOutlined from '@ant-design/icons/LoadingOutlined';
|
|
import classNames from 'classnames';
|
|
|
|
import type { TimelineProps } from './Timeline';
|
|
import TimelineItem from './TimelineItem';
|
|
import type { TimelineItemProps } from './TimelineItem';
|
|
|
|
const TimelineItemList: React.FC<TimelineProps & { hashId: string; direction?: string }> = ({
|
|
prefixCls,
|
|
className,
|
|
pending = false,
|
|
children,
|
|
items,
|
|
rootClassName,
|
|
reverse = false,
|
|
direction,
|
|
hashId,
|
|
pendingDot,
|
|
mode = '' as TimelineProps['mode'],
|
|
...restProps
|
|
}) => {
|
|
const getPositionCls = (position: string, idx: number) => {
|
|
if (mode === 'alternate') {
|
|
if (position === 'right') return `${prefixCls}-item-right`;
|
|
if (position === 'left') return `${prefixCls}-item-left`;
|
|
return idx % 2 === 0 ? `${prefixCls}-item-left` : `${prefixCls}-item-right`;
|
|
}
|
|
if (mode === 'left') return `${prefixCls}-item-left`;
|
|
if (mode === 'right') return `${prefixCls}-item-right`;
|
|
if (position === 'right') return `${prefixCls}-item-right`;
|
|
return '';
|
|
};
|
|
const mergedItems = [...(items || [])];
|
|
const pendingNode = typeof pending === 'boolean' ? null : pending;
|
|
|
|
if (pending) {
|
|
mergedItems.push({
|
|
pending: !!pending,
|
|
dot: pendingDot || <LoadingOutlined />,
|
|
children: pendingNode,
|
|
});
|
|
}
|
|
|
|
if (reverse) {
|
|
mergedItems.reverse();
|
|
}
|
|
const itemsCount = mergedItems.length;
|
|
const lastCls = `${prefixCls}-item-last`;
|
|
|
|
const itemsList = mergedItems
|
|
.filter((item: TimelineItemProps) => !!item)
|
|
.map((item: TimelineItemProps, idx: number) => {
|
|
const pendingClass = idx === itemsCount - 2 ? lastCls : '';
|
|
const readyClass = idx === itemsCount - 1 ? lastCls : '';
|
|
const { className: itemClassName, ...itemProps } = item;
|
|
|
|
return (
|
|
<TimelineItem
|
|
{...itemProps}
|
|
className={classNames([
|
|
itemClassName,
|
|
!reverse && !!pending ? pendingClass : readyClass,
|
|
getPositionCls(item?.position ?? '', idx),
|
|
])}
|
|
/* eslint-disable-next-line react/no-array-index-key */
|
|
key={item?.key || idx}
|
|
/>
|
|
);
|
|
});
|
|
|
|
const hasLabelItem = mergedItems.some((item: TimelineItemProps) => !!item?.label);
|
|
|
|
const classString = classNames(
|
|
prefixCls,
|
|
{
|
|
[`${prefixCls}-pending`]: !!pending,
|
|
[`${prefixCls}-reverse`]: !!reverse,
|
|
[`${prefixCls}-${mode}`]: !!mode && !hasLabelItem,
|
|
[`${prefixCls}-label`]: hasLabelItem,
|
|
[`${prefixCls}-rtl`]: direction === 'rtl',
|
|
},
|
|
className,
|
|
rootClassName,
|
|
hashId,
|
|
);
|
|
|
|
return (
|
|
<ul {...restProps} className={classString}>
|
|
{itemsList}
|
|
</ul>
|
|
);
|
|
};
|
|
|
|
export default TimelineItemList;
|