ant-design-vue/components/card/Card.tsx
Carl Chen d870f3f8e0
fix: fixed error with no expected value in expandColumnTitle slot (#7265)
* fix: fixed error report with no expected value in `expandColumnTitle` slot

* fix: optimize optional chain

* fix: use default render

* refactor: use `customRenderSlot` replace `renderSlot`

* style: code format

* perf: optimize useColumns code

* fix: fix path

* feat: add customRenderSlot unit test
2024-01-16 20:49:38 +08:00

196 lines
6.4 KiB
Vue

import type { VNodeTypes, PropType, VNode, ExtractPropTypes, CSSProperties } from 'vue';
import { isVNode, defineComponent } 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';
import type { CustomSlotsType } from '../_util/type';
import { customRenderSlot } from '../_util/vnode';
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: Object as CustomSlotsType<{
title: any;
extra: any;
tabBarExtraContent: any;
actions: any;
cover: any;
customTab: CardTabListType;
default: any;
}>,
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 = customRenderSlot(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;