ant-design/components/segmented/style/index.tsx

247 lines
7.3 KiB
TypeScript
Raw Normal View History

import type { CSSObject } from '@ant-design/cssinjs';
2023-05-10 14:56:58 +08:00
import { resetComponent, textEllipsis } from '../../style';
import type { FullToken, GenerateStyle } from '../../theme/internal';
import { genComponentStyleHook, mergeToken } from '../../theme/internal';
export interface ComponentToken {
/**
* @desc
* @descEN Text color of item
*/
itemColor: string;
/**
* @desc
* @descEN Text color of item when hover
*/
itemHoverColor: string;
/**
* @desc
* @descEN Background color of item when hover
*/
itemHoverBg: string;
/**
* @desc
* @descEN Background color of item when active
*/
itemActiveBg: string;
/**
* @desc
* @descEN Background color of item when selected
*/
itemSelectedBg: string;
}
interface SegmentedToken extends FullToken<'Segmented'> {
2023-05-25 14:22:44 +08:00
segmentedPadding: number;
segmentedBgColor: string;
segmentedPaddingHorizontal: number;
segmentedPaddingHorizontalSM: number;
}
// ============================== Mixins ==============================
function getItemDisabledStyle(cls: string, token: SegmentedToken): CSSObject {
return {
[`${cls}, ${cls}:hover, ${cls}:focus`]: {
color: token.colorTextDisabled,
cursor: 'not-allowed',
},
};
}
function getItemSelectedStyle(token: SegmentedToken): CSSObject {
return {
backgroundColor: token.itemSelectedBg,
boxShadow: token.boxShadowTertiary,
};
}
const segmentedTextEllipsisCss: CSSObject = {
overflow: 'hidden',
// handle text ellipsis
...textEllipsis,
};
// ============================== Styles ==============================
const genSegmentedStyle: GenerateStyle<SegmentedToken> = (token: SegmentedToken) => {
const { componentCls } = token;
return {
[componentCls]: {
...resetComponent(token),
display: 'inline-block',
2023-05-25 14:22:44 +08:00
padding: token.segmentedPadding,
color: token.itemColor,
2023-05-25 14:22:44 +08:00
backgroundColor: token.segmentedBgColor,
borderRadius: token.borderRadius,
transition: `all ${token.motionDurationMid} ${token.motionEaseInOut}`,
2022-05-26 09:14:09 +08:00
[`${componentCls}-group`]: {
position: 'relative',
display: 'flex',
alignItems: 'stretch',
justifyItems: 'flex-start',
width: '100%',
},
// RTL styles
[`&${componentCls}-rtl`]: {
direction: 'rtl',
},
// block styles
[`&${componentCls}-block`]: {
display: 'flex',
},
[`&${componentCls}-block ${componentCls}-item`]: {
flex: 1,
minWidth: 0,
},
// item styles
[`${componentCls}-item`]: {
position: 'relative',
textAlign: 'center',
cursor: 'pointer',
transition: `color ${token.motionDurationMid} ${token.motionEaseInOut}`,
borderRadius: token.borderRadiusSM,
'&-selected': {
...getItemSelectedStyle(token),
color: token.itemHoverColor,
},
'&::after': {
content: '""',
position: 'absolute',
width: '100%',
height: '100%',
top: 0,
insetInlineStart: 0,
borderRadius: 'inherit',
transition: `background-color ${token.motionDurationMid}`,
// This is mandatory to make it not clickable or hoverable
// Ref: https://github.com/ant-design/ant-design/issues/40888
pointerEvents: 'none',
},
[`&:hover:not(${componentCls}-item-selected):not(${componentCls}-item-disabled)`]: {
color: token.itemHoverColor,
'&::after': {
backgroundColor: token.itemHoverBg,
},
},
2023-05-10 14:56:58 +08:00
[`&:active:not(${componentCls}-item-selected):not(${componentCls}-item-disabled)`]: {
color: token.itemHoverColor,
2023-05-10 14:56:58 +08:00
'&::after': {
backgroundColor: token.itemActiveBg,
2023-05-10 14:56:58 +08:00
},
},
'&-label': {
2023-05-25 14:22:44 +08:00
minHeight: token.controlHeight - token.segmentedPadding * 2,
lineHeight: `${token.controlHeight - token.segmentedPadding * 2}px`,
padding: `0 ${token.segmentedPaddingHorizontal}px`,
...segmentedTextEllipsisCss,
},
// syntactic sugar to add `icon` for Segmented Item
'&-icon + *': {
2022-12-15 11:38:08 +08:00
marginInlineStart: token.marginSM / 2,
},
'&-input': {
position: 'absolute',
insetBlockStart: 0,
insetInlineStart: 0,
width: 0,
height: 0,
opacity: 0,
pointerEvents: 'none',
},
},
// thumb styles
[`${componentCls}-thumb`]: {
...getItemSelectedStyle(token),
position: 'absolute',
insetBlockStart: 0,
insetInlineStart: 0,
width: 0,
height: '100%',
padding: `${token.paddingXXS}px 0`,
borderRadius: token.borderRadiusSM,
[`& ~ ${componentCls}-item:not(${componentCls}-item-selected):not(${componentCls}-item-disabled)::after`]:
{
backgroundColor: 'transparent',
},
},
// size styles
[`&${componentCls}-lg`]: {
borderRadius: token.borderRadiusLG,
[`${componentCls}-item-label`]: {
2023-05-25 14:22:44 +08:00
minHeight: token.controlHeightLG - token.segmentedPadding * 2,
lineHeight: `${token.controlHeightLG - token.segmentedPadding * 2}px`,
padding: `0 ${token.segmentedPaddingHorizontal}px`,
fontSize: token.fontSizeLG,
},
[`${componentCls}-item, ${componentCls}-thumb`]: {
borderRadius: token.borderRadius,
},
},
[`&${componentCls}-sm`]: {
borderRadius: token.borderRadiusSM,
2022-09-21 10:35:14 +08:00
[`${componentCls}-item-label`]: {
2023-05-25 14:22:44 +08:00
minHeight: token.controlHeightSM - token.segmentedPadding * 2,
lineHeight: `${token.controlHeightSM - token.segmentedPadding * 2}px`,
padding: `0 ${token.segmentedPaddingHorizontalSM}px`,
},
[`${componentCls}-item, ${componentCls}-thumb`]: {
borderRadius: token.borderRadiusXS,
},
},
// disabled styles
...getItemDisabledStyle(`&-disabled ${componentCls}-item`, token),
...getItemDisabledStyle(`${componentCls}-item-disabled`, token),
// transition effect when `appear-active`
[`${componentCls}-thumb-motion-appear-active`]: {
transition: `transform ${token.motionDurationSlow} ${token.motionEaseInOut}, width ${token.motionDurationSlow} ${token.motionEaseInOut}`,
willChange: 'transform, width',
},
},
};
};
// ============================== Export ==============================
export default genComponentStyleHook(
'Segmented',
(token) => {
2023-05-25 14:22:44 +08:00
const { lineWidth, lineWidthBold, colorBgLayout } = token;
const segmentedToken = mergeToken<SegmentedToken>(token, {
2023-05-25 14:22:44 +08:00
segmentedPadding: lineWidthBold,
segmentedBgColor: colorBgLayout,
segmentedPaddingHorizontal: token.controlPaddingHorizontal - lineWidth,
segmentedPaddingHorizontalSM: token.controlPaddingHorizontalSM - lineWidth,
});
return [genSegmentedStyle(segmentedToken)];
},
(token) => {
const { colorTextLabel, colorText, colorFillSecondary, colorBgElevated, colorFill } = token;
return {
itemColor: colorTextLabel,
itemHoverColor: colorText,
itemHoverBg: colorFillSecondary,
itemSelectedBg: colorBgElevated,
itemActiveBg: colorFill,
};
},
);