250 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			250 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			TypeScript
		
	
	
| import type { CSSInterpolation } from '../_util/cssinjs';
 | |
| import type { AliasToken } from '../theme/internal';
 | |
| import type { TokenWithCommonCls } from '../theme/util/genComponentStyleHook';
 | |
| import { roundedArrow } from './roundedArrow';
 | |
| 
 | |
| function connectArrowCls(classList: string[], showArrowCls = '') {
 | |
|   return classList.map(cls => `${showArrowCls}${cls}`).join(',');
 | |
| }
 | |
| 
 | |
| export const MAX_VERTICAL_CONTENT_RADIUS = 8;
 | |
| 
 | |
| export function getArrowOffset(options: {
 | |
|   sizePopupArrow: number;
 | |
|   contentRadius: number;
 | |
|   borderRadiusOuter: number;
 | |
|   limitVerticalRadius?: boolean;
 | |
| }) {
 | |
|   const maxVerticalContentRadius = MAX_VERTICAL_CONTENT_RADIUS;
 | |
|   const { sizePopupArrow, contentRadius, borderRadiusOuter, limitVerticalRadius } = options;
 | |
|   const arrowInnerOffset = sizePopupArrow / 2 - Math.ceil(borderRadiusOuter * (Math.sqrt(2) - 1));
 | |
|   const dropdownArrowOffset = (contentRadius > 12 ? contentRadius + 2 : 12) - arrowInnerOffset;
 | |
|   const dropdownArrowOffsetVertical = limitVerticalRadius
 | |
|     ? maxVerticalContentRadius - arrowInnerOffset
 | |
|     : dropdownArrowOffset;
 | |
|   return { dropdownArrowOffset, dropdownArrowOffsetVertical };
 | |
| }
 | |
| 
 | |
| export default function getArrowStyle<Token extends TokenWithCommonCls<AliasToken>>(
 | |
|   token: Token,
 | |
|   options: {
 | |
|     colorBg: string;
 | |
|     showArrowCls?: string;
 | |
|     contentRadius?: number;
 | |
|     limitVerticalRadius?: boolean;
 | |
|   },
 | |
| ): CSSInterpolation {
 | |
|   const {
 | |
|     componentCls,
 | |
|     sizePopupArrow,
 | |
|     marginXXS,
 | |
|     borderRadiusXS,
 | |
|     borderRadiusOuter,
 | |
|     boxShadowPopoverArrow,
 | |
|   } = token;
 | |
| 
 | |
|   const {
 | |
|     colorBg,
 | |
|     showArrowCls,
 | |
|     contentRadius = token.borderRadiusLG,
 | |
|     limitVerticalRadius,
 | |
|   } = options;
 | |
| 
 | |
|   const { dropdownArrowOffsetVertical, dropdownArrowOffset } = getArrowOffset({
 | |
|     sizePopupArrow,
 | |
|     contentRadius,
 | |
|     borderRadiusOuter,
 | |
|     limitVerticalRadius,
 | |
|   });
 | |
|   const dropdownArrowDistance = sizePopupArrow / 2 + marginXXS;
 | |
| 
 | |
|   return {
 | |
|     [componentCls]: {
 | |
|       // ============================ Basic ============================
 | |
|       [`${componentCls}-arrow`]: [
 | |
|         {
 | |
|           position: 'absolute',
 | |
|           zIndex: 1, // lift it up so the menu wouldn't cask shadow on it
 | |
|           display: 'block',
 | |
| 
 | |
|           ...roundedArrow(
 | |
|             sizePopupArrow,
 | |
|             borderRadiusXS,
 | |
|             borderRadiusOuter,
 | |
|             colorBg,
 | |
|             boxShadowPopoverArrow,
 | |
|           ),
 | |
| 
 | |
|           '&:before': {
 | |
|             background: colorBg,
 | |
|           },
 | |
|         },
 | |
|       ],
 | |
| 
 | |
|       // ========================== Placement ==========================
 | |
|       // Here handle the arrow position and rotate stuff
 | |
|       // >>>>> Top
 | |
|       [[
 | |
|         `&-placement-top ${componentCls}-arrow`,
 | |
|         `&-placement-topLeft ${componentCls}-arrow`,
 | |
|         `&-placement-topRight ${componentCls}-arrow`,
 | |
|       ].join(',')]: {
 | |
|         bottom: 0,
 | |
|         transform: 'translateY(100%) rotate(180deg)',
 | |
|       },
 | |
| 
 | |
|       [`&-placement-top ${componentCls}-arrow`]: {
 | |
|         left: {
 | |
|           _skip_check_: true,
 | |
|           value: '50%',
 | |
|         },
 | |
|         transform: 'translateX(-50%) translateY(100%) rotate(180deg)',
 | |
|       },
 | |
| 
 | |
|       [`&-placement-topLeft ${componentCls}-arrow`]: {
 | |
|         left: {
 | |
|           _skip_check_: true,
 | |
|           value: dropdownArrowOffset,
 | |
|         },
 | |
|       },
 | |
| 
 | |
|       [`&-placement-topRight ${componentCls}-arrow`]: {
 | |
|         right: {
 | |
|           _skip_check_: true,
 | |
|           value: dropdownArrowOffset,
 | |
|         },
 | |
|       },
 | |
| 
 | |
|       // >>>>> Bottom
 | |
|       [[
 | |
|         `&-placement-bottom ${componentCls}-arrow`,
 | |
|         `&-placement-bottomLeft ${componentCls}-arrow`,
 | |
|         `&-placement-bottomRight ${componentCls}-arrow`,
 | |
|       ].join(',')]: {
 | |
|         top: 0,
 | |
|         transform: `translateY(-100%)`,
 | |
|       },
 | |
| 
 | |
|       [`&-placement-bottom ${componentCls}-arrow`]: {
 | |
|         left: {
 | |
|           _skip_check_: true,
 | |
|           value: '50%',
 | |
|         },
 | |
|         transform: `translateX(-50%) translateY(-100%)`,
 | |
|       },
 | |
| 
 | |
|       [`&-placement-bottomLeft ${componentCls}-arrow`]: {
 | |
|         left: {
 | |
|           _skip_check_: true,
 | |
|           value: dropdownArrowOffset,
 | |
|         },
 | |
|       },
 | |
| 
 | |
|       [`&-placement-bottomRight ${componentCls}-arrow`]: {
 | |
|         right: {
 | |
|           _skip_check_: true,
 | |
|           value: dropdownArrowOffset,
 | |
|         },
 | |
|       },
 | |
| 
 | |
|       // >>>>> Left
 | |
|       [[
 | |
|         `&-placement-left ${componentCls}-arrow`,
 | |
|         `&-placement-leftTop ${componentCls}-arrow`,
 | |
|         `&-placement-leftBottom ${componentCls}-arrow`,
 | |
|       ].join(',')]: {
 | |
|         right: {
 | |
|           _skip_check_: true,
 | |
|           value: 0,
 | |
|         },
 | |
|         transform: 'translateX(100%) rotate(90deg)',
 | |
|       },
 | |
| 
 | |
|       [`&-placement-left ${componentCls}-arrow`]: {
 | |
|         top: {
 | |
|           _skip_check_: true,
 | |
|           value: '50%',
 | |
|         },
 | |
|         transform: 'translateY(-50%) translateX(100%) rotate(90deg)',
 | |
|       },
 | |
| 
 | |
|       [`&-placement-leftTop ${componentCls}-arrow`]: {
 | |
|         top: dropdownArrowOffsetVertical,
 | |
|       },
 | |
| 
 | |
|       [`&-placement-leftBottom ${componentCls}-arrow`]: {
 | |
|         bottom: dropdownArrowOffsetVertical,
 | |
|       },
 | |
| 
 | |
|       // >>>>> Right
 | |
|       [[
 | |
|         `&-placement-right ${componentCls}-arrow`,
 | |
|         `&-placement-rightTop ${componentCls}-arrow`,
 | |
|         `&-placement-rightBottom ${componentCls}-arrow`,
 | |
|       ].join(',')]: {
 | |
|         left: {
 | |
|           _skip_check_: true,
 | |
|           value: 0,
 | |
|         },
 | |
|         transform: 'translateX(-100%) rotate(-90deg)',
 | |
|       },
 | |
| 
 | |
|       [`&-placement-right ${componentCls}-arrow`]: {
 | |
|         top: {
 | |
|           _skip_check_: true,
 | |
|           value: '50%',
 | |
|         },
 | |
|         transform: 'translateY(-50%) translateX(-100%) rotate(-90deg)',
 | |
|       },
 | |
| 
 | |
|       [`&-placement-rightTop ${componentCls}-arrow`]: {
 | |
|         top: dropdownArrowOffsetVertical,
 | |
|       },
 | |
| 
 | |
|       [`&-placement-rightBottom ${componentCls}-arrow`]: {
 | |
|         bottom: dropdownArrowOffsetVertical,
 | |
|       },
 | |
| 
 | |
|       // =========================== Offset ============================
 | |
|       // Offset the popover to account for the dropdown arrow
 | |
|       // >>>>> Top
 | |
|       [connectArrowCls(
 | |
|         [`&-placement-topLeft`, `&-placement-top`, `&-placement-topRight`],
 | |
|         showArrowCls,
 | |
|       )]: {
 | |
|         paddingBottom: dropdownArrowDistance,
 | |
|       },
 | |
| 
 | |
|       // >>>>> Bottom
 | |
|       [connectArrowCls(
 | |
|         [`&-placement-bottomLeft`, `&-placement-bottom`, `&-placement-bottomRight`],
 | |
|         showArrowCls,
 | |
|       )]: {
 | |
|         paddingTop: dropdownArrowDistance,
 | |
|       },
 | |
| 
 | |
|       // >>>>> Left
 | |
|       [connectArrowCls(
 | |
|         [`&-placement-leftTop`, `&-placement-left`, `&-placement-leftBottom`],
 | |
|         showArrowCls,
 | |
|       )]: {
 | |
|         paddingRight: {
 | |
|           _skip_check_: true,
 | |
|           value: dropdownArrowDistance,
 | |
|         },
 | |
|       },
 | |
| 
 | |
|       // >>>>> Right
 | |
|       [connectArrowCls(
 | |
|         [`&-placement-rightTop`, `&-placement-right`, `&-placement-rightBottom`],
 | |
|         showArrowCls,
 | |
|       )]: {
 | |
|         paddingLeft: {
 | |
|           _skip_check_: true,
 | |
|           value: dropdownArrowDistance,
 | |
|         },
 | |
|       },
 | |
|     },
 | |
|   };
 | |
| }
 |