You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
185 lines
6.2 KiB
185 lines
6.2 KiB
import type { VNodeTypes, PropType, VNode, ExtractPropTypes, CSSProperties } from 'vue';
|
|
import { isVNode, defineComponent, renderSlot } from 'vue';
|
|
import Tabs from '../tabs';
|
|
import PropTypes from '../_util/vue-types';
|
|
import { flattenChildren, isEmptyElement, filterEmptyWithUndefined } from '../_util/props-util';
|
|
import type { SizeType } from '../config-provider';
|
|
import isPlainObject from 'lodash-es/isPlainObject';
|
|
import useConfigInject from '../config-provider/hooks/useConfigInject';
|
|
import devWarning from '../vc-util/devWarning';
|
|
import useStyle from './style';
|
|
import Skeleton from '../skeleton';
|
|
export interface CardTabListType {
|
|
key: string;
|
|
tab: any;
|
|
/** @deprecated Please use `customTab` instead. */
|
|
slots?: { tab: string };
|
|
disabled?: boolean;
|
|
}
|
|
|
|
export type CardType = 'inner';
|
|
export type CardSize = 'default' | 'small';
|
|
|
|
const { TabPane } = Tabs;
|
|
|
|
export const cardProps = () => ({
|
|
prefixCls: String,
|
|
title: PropTypes.any,
|
|
extra: PropTypes.any,
|
|
bordered: { type: Boolean, default: true },
|
|
bodyStyle: { type: Object as PropType<CSSProperties>, default: undefined as CSSProperties },
|
|
headStyle: { type: Object as PropType<CSSProperties>, default: undefined as CSSProperties },
|
|
loading: { type: Boolean, default: false },
|
|
hoverable: { type: Boolean, default: false },
|
|
type: { type: String as PropType<CardType> },
|
|
size: { type: String as PropType<CardSize> },
|
|
actions: PropTypes.any,
|
|
tabList: {
|
|
type: Array as PropType<CardTabListType[]>,
|
|
},
|
|
tabBarExtraContent: PropTypes.any,
|
|
activeTabKey: String,
|
|
defaultActiveTabKey: String,
|
|
cover: PropTypes.any,
|
|
onTabChange: {
|
|
type: Function as PropType<(key: string) => void>,
|
|
},
|
|
});
|
|
|
|
export type CardProps = Partial<ExtractPropTypes<ReturnType<typeof cardProps>>>;
|
|
|
|
const Card = defineComponent({
|
|
compatConfig: { MODE: 3 },
|
|
name: 'ACard',
|
|
inheritAttrs: false,
|
|
props: cardProps(),
|
|
slots: ['title', 'extra', 'tabBarExtraContent', 'actions', 'cover', 'customTab'],
|
|
setup(props, { slots, attrs }) {
|
|
const { prefixCls, direction, size } = useConfigInject('card', props);
|
|
const [wrapSSR, hashId] = useStyle(prefixCls);
|
|
const getAction = (actions: VNodeTypes[]) => {
|
|
const actionList = actions.map((action, index) =>
|
|
(isVNode(action) && !isEmptyElement(action)) || !isVNode(action) ? (
|
|
<li style={{ width: `${100 / actions.length}%` }} key={`action-${index}`}>
|
|
<span>{action}</span>
|
|
</li>
|
|
) : null,
|
|
);
|
|
return actionList;
|
|
};
|
|
const triggerTabChange = (key: string) => {
|
|
props.onTabChange?.(key);
|
|
};
|
|
const isContainGrid = (obj: VNode[] = []) => {
|
|
let containGrid: boolean;
|
|
obj.forEach(element => {
|
|
if (element && isPlainObject(element.type) && (element.type as any).__ANT_CARD_GRID) {
|
|
containGrid = true;
|
|
}
|
|
});
|
|
return containGrid;
|
|
};
|
|
|
|
return () => {
|
|
const {
|
|
headStyle = {},
|
|
bodyStyle = {},
|
|
loading,
|
|
bordered = true,
|
|
type,
|
|
tabList,
|
|
hoverable,
|
|
activeTabKey,
|
|
defaultActiveTabKey,
|
|
tabBarExtraContent = filterEmptyWithUndefined(slots.tabBarExtraContent?.()),
|
|
title = filterEmptyWithUndefined(slots.title?.()),
|
|
extra = filterEmptyWithUndefined(slots.extra?.()),
|
|
actions = filterEmptyWithUndefined(slots.actions?.()),
|
|
cover = filterEmptyWithUndefined(slots.cover?.()),
|
|
} = props;
|
|
const children = flattenChildren(slots.default?.());
|
|
const pre = prefixCls.value;
|
|
const classString = {
|
|
[`${pre}`]: true,
|
|
[hashId.value]: true,
|
|
[`${pre}-loading`]: loading,
|
|
[`${pre}-bordered`]: bordered,
|
|
[`${pre}-hoverable`]: !!hoverable,
|
|
[`${pre}-contain-grid`]: isContainGrid(children),
|
|
[`${pre}-contain-tabs`]: tabList && tabList.length,
|
|
[`${pre}-${size.value}`]: size.value,
|
|
[`${pre}-type-${type}`]: !!type,
|
|
[`${pre}-rtl`]: direction.value === 'rtl',
|
|
};
|
|
const loadingBlock = (
|
|
<Skeleton loading active paragraph={{ rows: 4 }} title={false}>
|
|
{children}
|
|
</Skeleton>
|
|
);
|
|
|
|
const hasActiveTabKey = activeTabKey !== undefined;
|
|
const tabsProps = {
|
|
size: 'large' as SizeType,
|
|
[hasActiveTabKey ? 'activeKey' : 'defaultActiveKey']: hasActiveTabKey
|
|
? activeTabKey
|
|
: defaultActiveTabKey,
|
|
onChange: triggerTabChange,
|
|
class: `${pre}-head-tabs`,
|
|
};
|
|
|
|
let head;
|
|
const tabs =
|
|
tabList && tabList.length ? (
|
|
<Tabs
|
|
{...tabsProps}
|
|
v-slots={{ rightExtra: tabBarExtraContent ? () => tabBarExtraContent : null }}
|
|
>
|
|
{tabList.map(item => {
|
|
const { tab: temp, slots: itemSlots } = item as CardTabListType;
|
|
const name = itemSlots?.tab;
|
|
devWarning(
|
|
!itemSlots,
|
|
'Card',
|
|
`tabList slots is deprecated, Please use \`customTab\` instead.`,
|
|
);
|
|
let tab = temp !== undefined ? temp : slots[name] ? slots[name](item) : null;
|
|
tab = renderSlot(slots, 'customTab', item as any, () => [tab]);
|
|
return <TabPane tab={tab} key={item.key} disabled={item.disabled} />;
|
|
})}
|
|
</Tabs>
|
|
) : null;
|
|
if (title || extra || tabs) {
|
|
head = (
|
|
<div class={`${pre}-head`} style={headStyle}>
|
|
<div class={`${pre}-head-wrapper`}>
|
|
{title && <div class={`${pre}-head-title`}>{title}</div>}
|
|
{extra && <div class={`${pre}-extra`}>{extra}</div>}
|
|
</div>
|
|
{tabs}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const coverDom = cover ? <div class={`${pre}-cover`}>{cover}</div> : null;
|
|
const body = (
|
|
<div class={`${pre}-body`} style={bodyStyle}>
|
|
{loading ? loadingBlock : children}
|
|
</div>
|
|
);
|
|
const actionDom =
|
|
actions && actions.length ? <ul class={`${pre}-actions`}>{getAction(actions)}</ul> : null;
|
|
|
|
return wrapSSR(
|
|
<div ref="cardContainerRef" {...attrs} class={[classString, attrs.class]}>
|
|
{head}
|
|
{coverDom}
|
|
{children && children.length ? body : null}
|
|
{actionDom}
|
|
</div>,
|
|
);
|
|
};
|
|
},
|
|
});
|
|
|
|
export default Card;
|