import type { CSSObject } from '@ant-design/cssinjs'; import { Keyframes } from '@ant-design/cssinjs'; import type { FullToken, GenerateStyle } from '../../theme'; import { genComponentStyleHook, mergeToken } from '../../theme'; import { resetComponent } from '../../style'; /** Component only token. Which will handle additional calculation of alias token */ export interface ComponentToken { zIndexPopup: number; } type FloatButtonToken = FullToken<'FloatButton'> & { floatButtonColor: string; floatButtonBackgroundColor: string; floatButtonHoverBackgroundColor: string; floatButtonFontSize: number; floatButtonSize: number; floatButtonIconSize: number; // Position floatButtonInsetBlockEnd: number; floatButtonInsetInlineEnd: number; }; // ============================== Group ============================== const floatButtonGroupStyle: GenerateStyle = token => { const { componentCls, floatButtonSize, margin, borderRadius, motionDurationSlow } = token; const groupPrefixCls = `${componentCls}-group`; const moveDownIn = new Keyframes('antFloatButtonMoveDownIn', { '0%': { transform: `translate3d(0, ${floatButtonSize}px, 0)`, transformOrigin: '0 0', opacity: 0, }, '100%': { transform: 'translate3d(0, 0, 0)', transformOrigin: '0 0', opacity: 1, }, }); const moveDownOut = new Keyframes('antFloatButtonMoveDownOut', { '0%': { transform: 'translate3d(0, 0, 0)', transformOrigin: '0 0', opacity: 1, }, '100%': { transform: `translate3d(0, ${floatButtonSize}px, 0)`, transformOrigin: '0 0', opacity: 0, }, }); return { [groupPrefixCls]: { ...resetComponent(token), zIndex: 99, display: 'block', border: 'none', position: 'fixed', width: floatButtonSize, height: 'auto', boxShadow: 'none', minHeight: floatButtonSize, insetInlineEnd: token.floatButtonInsetInlineEnd, insetBlockEnd: token.floatButtonInsetBlockEnd, borderRadius: token.borderRadius, [`${groupPrefixCls}-wrap`]: { zIndex: -1, display: 'block', position: 'relative', marginBottom: margin, }, '&&-rtl': { direction: 'rtl', }, [componentCls]: { position: 'static', }, }, [`${groupPrefixCls}-circle`]: { [`${componentCls}-circle:not(:last-child)`]: { marginBottom: token.margin, [`${componentCls}-body`]: { width: floatButtonSize, height: floatButtonSize, }, }, }, [`${groupPrefixCls}-square`]: { [`${componentCls}-square`]: { borderRadius: 0, padding: 0, '&:first-child': { borderStartStartRadius: borderRadius, borderStartEndRadius: borderRadius, }, '&:last-child': { borderEndStartRadius: borderRadius, borderEndEndRadius: borderRadius, }, '&:not(:last-child)': { borderBottom: `${token.lineWidth}px ${token.lineType} ${token.colorSplit}`, }, }, [`${groupPrefixCls}-wrap`]: { display: 'block', borderRadius, boxShadow: token.boxShadowSecondary, overflow: 'hidden', [`${componentCls}-square`]: { boxShadow: 'none', marginTop: 0, borderRadius: 0, padding: token.paddingXXS, '&:first-child': { borderStartStartRadius: borderRadius, borderStartEndRadius: borderRadius, }, '&:last-child': { borderEndStartRadius: borderRadius, borderEndEndRadius: borderRadius, }, '&:not(:last-child)': { borderBottom: `${token.lineWidth}px ${token.lineType} ${token.colorSplit}`, }, [`${componentCls}-body`]: { width: floatButtonSize - token.paddingXXS * 2, height: floatButtonSize - token.paddingXXS * 2, }, }, }, }, [`${groupPrefixCls}-wrap-enter,${groupPrefixCls}-wrap-enter-active`]: { animationName: moveDownIn, animationDuration: motionDurationSlow, }, [`${groupPrefixCls}-wrap-leave`]: { animationName: moveDownOut, animationDuration: motionDurationSlow, }, [`${groupPrefixCls}-circle-shadow`]: { boxShadow: 'none', }, [`${groupPrefixCls}-square-shadow`]: { boxShadow: token.boxShadowSecondary, [`${componentCls}-square`]: { boxShadow: 'none', padding: token.paddingXXS, [`${componentCls}-body`]: { width: floatButtonSize - token.paddingXXS * 2, height: floatButtonSize - token.paddingXXS * 2, }, }, }, }; }; // ============================== Shared ============================== const sharedFloatButtonStyle: GenerateStyle = token => { const { componentCls, floatButtonIconSize, floatButtonSize } = token; return { [componentCls]: { ...resetComponent(token), border: 'none', position: 'fixed', cursor: 'pointer', overflow: 'hidden', zIndex: 99, display: 'block', justifyContent: 'center', alignItems: 'center', width: floatButtonSize, height: floatButtonSize, insetInlineEnd: token.floatButtonInsetInlineEnd, insetBlockEnd: token.floatButtonInsetBlockEnd, boxShadow: token.boxShadowSecondary, // Pure Panel '&-pure': { position: 'relative', inset: 'auto', }, '&:empty': { display: 'none', }, [`${componentCls}-body`]: { width: floatButtonSize, height: floatButtonSize, display: 'flex', justifyContent: 'center', alignItems: 'center', transition: `all ${token.motionDurationFast}`, [`${componentCls}-content`]: { overflow: 'hidden', textAlign: 'center', minHeight: floatButtonSize, display: 'flex', flexDirection: 'column', justifyContent: 'center', alignItems: 'center', padding: `2px 4px`, [`${componentCls}-icon`]: { textAlign: 'center', margin: 'auto', width: floatButtonIconSize, fontSize: floatButtonIconSize, lineHeight: 1, }, }, }, }, [`${componentCls}-circle`]: { height: floatButtonSize, borderRadius: '50%', [`${componentCls}-body`]: { borderRadius: '50%', }, }, [`${componentCls}-square`]: { height: 'auto', minHeight: floatButtonSize, borderRadius: token.borderRadius, [`${componentCls}-body`]: { height: 'auto', borderRadius: token.borderRadiusSM, }, }, [`${componentCls}-default`]: { backgroundColor: token.colorBgContainer, transition: `background-color ${token.motionDurationFast}`, [`${componentCls}-body`]: { backgroundColor: token.colorBgContainer, transition: `background-color ${token.motionDurationFast}`, '&:hover': { backgroundColor: token.colorFillContent, }, [`${componentCls}-content`]: { [`${componentCls}-icon`]: { color: token.colorText, }, [`${componentCls}-description`]: { display: 'flex', alignItems: 'center', lineHeight: `${token.fontSizeLG}px`, color: token.colorText, fontSize: token.fontSizeSM, }, }, }, }, [`${componentCls}-primary`]: { backgroundColor: token.colorPrimary, [`${componentCls}-body`]: { backgroundColor: token.colorPrimary, transition: `background-color ${token.motionDurationFast}`, '&:hover': { backgroundColor: token.colorPrimaryHover, }, [`${componentCls}-content`]: { [`${componentCls}-icon`]: { color: token.colorTextLightSolid, }, [`${componentCls}-description`]: { display: 'flex', alignItems: 'center', lineHeight: `${token.fontSizeLG}px`, color: token.colorTextLightSolid, fontSize: token.fontSizeSM, }, }, }, }, }; }; // ============================== Export ============================== export default genComponentStyleHook<'FloatButton'>('FloatButton', token => { const { colorTextLightSolid, colorBgContainer, controlHeightLG, marginXXL, marginLG, fontSize, fontSizeIcon, controlItemBgHover, } = token; const floatButtonToken = mergeToken(token, { floatButtonBackgroundColor: colorBgContainer, floatButtonColor: colorTextLightSolid, floatButtonHoverBackgroundColor: controlItemBgHover, floatButtonFontSize: fontSize, floatButtonIconSize: fontSizeIcon * 1.5, floatButtonSize: controlHeightLG, floatButtonInsetBlockEnd: marginXXL, floatButtonInsetInlineEnd: marginLG, }); return [floatButtonGroupStyle(floatButtonToken), sharedFloatButtonStyle(floatButtonToken)]; });