import { unit } from '@ant-design/cssinjs'; import type { CSSObject } from '@ant-design/cssinjs'; import { clearFix, resetComponent, textEllipsis } from '../../style'; import type { FullToken, GenerateStyle, GetDefaultToken } from '../../theme/internal'; import { genStyleHooks, mergeToken } from '../../theme/internal'; export interface ComponentToken { /** * @desc 卡片头部背景色 * @descEN Background color of card header */ headerBg: string; /** * @desc 卡片头部文字大小 * @descEN Font size of card header */ headerFontSize: number | string; /** * @desc 小号卡片头部文字大小 * @descEN Font size of small card header */ headerFontSizeSM: number | string; /** * @desc 卡片头部高度 * @descEN Height of card header */ headerHeight: number | string; /** * @desc 小号卡片头部高度 * @descEN Height of small card header */ headerHeightSM: number | string; /** * @desc 操作区背景色 * @descEN Background color of card actions */ actionsBg: string; /** * @desc 操作区每一项的外间距 * @descEN Margin of each item in card actions */ actionsLiMargin: string; /** * @desc 内置标签页组件下间距 * @descEN Margin bottom of tabs component */ tabsMarginBottom: number; /** * @desc 额外区文字颜色 * @descEN Text color of extra area */ extraColor: string; } interface CardToken extends FullToken<'Card'> { cardShadow: string; cardHeadPadding: number; cardPaddingSM: number; cardPaddingBase: number; cardActionsIconSize: number; } // ============================== Styles ============================== // ============================== Head ============================== const genCardHeadStyle: GenerateStyle = (token): CSSObject => { const { antCls, componentCls, headerHeight, cardPaddingBase, tabsMarginBottom } = token; return { display: 'flex', justifyContent: 'center', flexDirection: 'column', minHeight: headerHeight, marginBottom: -1, // Fix card grid overflow bug: https://gw.alipayobjects.com/zos/rmsportal/XonYxBikwpgbqIQBeuhk.png padding: `0 ${unit(cardPaddingBase)}`, color: token.colorTextHeading, fontWeight: token.fontWeightStrong, fontSize: token.headerFontSize, background: token.headerBg, borderBottom: `${unit(token.lineWidth)} ${token.lineType} ${token.colorBorderSecondary}`, borderRadius: `${unit(token.borderRadiusLG)} ${unit(token.borderRadiusLG)} 0 0`, ...clearFix(), '&-wrapper': { width: '100%', display: 'flex', alignItems: 'center', }, '&-title': { display: 'inline-block', flex: 1, ...textEllipsis, [` > ${componentCls}-typography, > ${componentCls}-typography-edit-content `]: { insetInlineStart: 0, marginTop: 0, marginBottom: 0, }, }, [`${antCls}-tabs-top`]: { clear: 'both', marginBottom: tabsMarginBottom, color: token.colorText, fontWeight: 'normal', fontSize: token.fontSize, '&-bar': { borderBottom: `${unit(token.lineWidth)} ${token.lineType} ${token.colorBorderSecondary}`, }, }, }; }; // ============================== Grid ============================== const genCardGridStyle: GenerateStyle = (token): CSSObject => { const { cardPaddingBase, colorBorderSecondary, cardShadow, lineWidth } = token; return { width: '33.33%', padding: cardPaddingBase, border: 0, borderRadius: 0, boxShadow: ` ${unit(lineWidth)} 0 0 0 ${colorBorderSecondary}, 0 ${unit(lineWidth)} 0 0 ${colorBorderSecondary}, ${unit(lineWidth)} ${unit(lineWidth)} 0 0 ${colorBorderSecondary}, ${unit(lineWidth)} 0 0 0 ${colorBorderSecondary} inset, 0 ${unit(lineWidth)} 0 0 ${colorBorderSecondary} inset; `, transition: `all ${token.motionDurationMid}`, '&-hoverable:hover': { position: 'relative', zIndex: 1, boxShadow: cardShadow, }, }; }; // ============================== Actions ============================== const genCardActionsStyle: GenerateStyle = (token): CSSObject => { const { componentCls, iconCls, actionsLiMargin, cardActionsIconSize, colorBorderSecondary, actionsBg, } = token; return { margin: 0, padding: 0, listStyle: 'none', background: actionsBg, borderTop: `${unit(token.lineWidth)} ${token.lineType} ${colorBorderSecondary}`, display: 'flex', borderRadius: `0 0 ${unit(token.borderRadiusLG)} ${unit(token.borderRadiusLG)}`, ...clearFix(), '& > li': { margin: actionsLiMargin, color: token.colorTextDescription, textAlign: 'center', '> span': { position: 'relative', display: 'block', minWidth: token.calc(token.cardActionsIconSize).mul(2).equal(), fontSize: token.fontSize, lineHeight: token.lineHeight, cursor: 'pointer', '&:hover': { color: token.colorPrimary, transition: `color ${token.motionDurationMid}`, }, [`a:not(${componentCls}-btn), > ${iconCls}`]: { display: 'inline-block', width: '100%', color: token.colorTextDescription, lineHeight: unit(token.fontHeight), transition: `color ${token.motionDurationMid}`, '&:hover': { color: token.colorPrimary, }, }, [`> ${iconCls}`]: { fontSize: cardActionsIconSize, lineHeight: unit(token.calc(cardActionsIconSize).mul(token.lineHeight).equal()), }, }, '&:not(:last-child)': { borderInlineEnd: `${unit(token.lineWidth)} ${token.lineType} ${colorBorderSecondary}`, }, }, }; }; // ============================== Meta ============================== const genCardMetaStyle: GenerateStyle = (token): CSSObject => ({ margin: `${unit(token.calc(token.marginXXS).mul(-1).equal())} 0`, display: 'flex', ...clearFix(), '&-avatar': { paddingInlineEnd: token.padding, }, '&-detail': { overflow: 'hidden', flex: 1, '> div:not(:last-child)': { marginBottom: token.marginXS, }, }, '&-title': { color: token.colorTextHeading, fontWeight: token.fontWeightStrong, fontSize: token.fontSizeLG, ...textEllipsis, }, '&-description': { color: token.colorTextDescription, }, }); // ============================== Inner ============================== const genCardTypeInnerStyle: GenerateStyle = (token): CSSObject => { const { componentCls, cardPaddingBase, colorFillAlter } = token; return { [`${componentCls}-head`]: { padding: `0 ${unit(cardPaddingBase)}`, background: colorFillAlter, '&-title': { fontSize: token.fontSize, }, }, [`${componentCls}-body`]: { padding: `${unit(token.padding)} ${unit(cardPaddingBase)}`, }, }; }; // ============================== Loading ============================== const genCardLoadingStyle: GenerateStyle = (token): CSSObject => { const { componentCls } = token; return { overflow: 'hidden', [`${componentCls}-body`]: { userSelect: 'none', }, }; }; // ============================== Basic ============================== const genCardStyle: GenerateStyle = (token): CSSObject => { const { antCls, componentCls, cardShadow, cardHeadPadding, colorBorderSecondary, boxShadowTertiary, cardPaddingBase, extraColor, } = token; return { [componentCls]: { ...resetComponent(token), position: 'relative', background: token.colorBgContainer, borderRadius: token.borderRadiusLG, [`&:not(${componentCls}-bordered)`]: { boxShadow: boxShadowTertiary, }, [`${componentCls}-head`]: genCardHeadStyle(token), [`${componentCls}-extra`]: { // https://stackoverflow.com/a/22429853/3040605 marginInlineStart: 'auto', color: extraColor, fontWeight: 'normal', fontSize: token.fontSize, }, [`${componentCls}-body`]: { padding: cardPaddingBase, borderRadius: `0 0 ${unit(token.borderRadiusLG)} ${unit(token.borderRadiusLG)}`, ...clearFix(), }, [`${componentCls}-grid`]: genCardGridStyle(token), [`${componentCls}-cover`]: { '> *': { display: 'block', width: '100%', }, [`img, img + ${antCls}-image-mask`]: { borderRadius: `${unit(token.borderRadiusLG)} ${unit(token.borderRadiusLG)} 0 0`, }, }, [`${componentCls}-actions`]: genCardActionsStyle(token), [`${componentCls}-meta`]: genCardMetaStyle(token), }, [`${componentCls}-bordered`]: { border: `${unit(token.lineWidth)} ${token.lineType} ${colorBorderSecondary}`, [`${componentCls}-cover`]: { marginTop: -1, marginInlineStart: -1, marginInlineEnd: -1, }, }, [`${componentCls}-hoverable`]: { cursor: 'pointer', transition: `box-shadow ${token.motionDurationMid}, border-color ${token.motionDurationMid}`, '&:hover': { borderColor: 'transparent', boxShadow: cardShadow, }, }, [`${componentCls}-contain-grid`]: { borderRadius: `${unit(token.borderRadiusLG)} ${unit(token.borderRadiusLG)} 0 0 `, [`${componentCls}-body`]: { display: 'flex', flexWrap: 'wrap', }, [`&:not(${componentCls}-loading) ${componentCls}-body`]: { marginBlockStart: token.calc(token.lineWidth).mul(-1).equal(), marginInlineStart: token.calc(token.lineWidth).mul(-1).equal(), padding: 0, }, }, [`${componentCls}-contain-tabs`]: { [`> div${componentCls}-head`]: { minHeight: 0, [`${componentCls}-head-title, ${componentCls}-extra`]: { paddingTop: cardHeadPadding, }, }, }, [`${componentCls}-type-inner`]: genCardTypeInnerStyle(token), [`${componentCls}-loading`]: genCardLoadingStyle(token), [`${componentCls}-rtl`]: { direction: 'rtl', }, }; }; // ============================== Size ============================== const genCardSizeStyle: GenerateStyle = (token): CSSObject => { const { componentCls, cardPaddingSM, headerHeightSM, headerFontSizeSM } = token; return { [`${componentCls}-small`]: { [`> ${componentCls}-head`]: { minHeight: headerHeightSM, padding: `0 ${unit(cardPaddingSM)}`, fontSize: headerFontSizeSM, [`> ${componentCls}-head-wrapper`]: { [`> ${componentCls}-extra`]: { fontSize: token.fontSize, }, }, }, [`> ${componentCls}-body`]: { padding: cardPaddingSM, }, }, [`${componentCls}-small${componentCls}-contain-tabs`]: { [`> ${componentCls}-head`]: { [`${componentCls}-head-title, ${componentCls}-extra`]: { paddingTop: 0, display: 'flex', alignItems: 'center', }, }, }, }; }; export const prepareComponentToken: GetDefaultToken<'Card'> = (token) => ({ headerBg: 'transparent', headerFontSize: token.fontSizeLG, headerFontSizeSM: token.fontSize, headerHeight: token.fontSizeLG * token.lineHeightLG + token.padding * 2, headerHeightSM: token.fontSize * token.lineHeight + token.paddingXS * 2, actionsBg: token.colorBgContainer, actionsLiMargin: `${token.paddingSM}px 0`, tabsMarginBottom: -token.padding - token.lineWidth, extraColor: token.colorText, }); // ============================== Export ============================== export default genStyleHooks( 'Card', (token) => { const cardToken = mergeToken(token, { cardShadow: token.boxShadowCard, cardHeadPadding: token.padding, cardPaddingBase: token.paddingLG, cardActionsIconSize: token.fontSize, cardPaddingSM: 12, // Fixed padding. }); return [ // Style genCardStyle(cardToken), // Size genCardSizeStyle(cardToken), ]; }, prepareComponentToken, );