307 lines
9.3 KiB
JavaScript
307 lines
9.3 KiB
JavaScript
|
import { TinyColor } from '@ctrl/tinycolor';
|
||
|
import { genModalMaskStyle } from '../../modal/style';
|
||
|
import { textEllipsis } from '../../style';
|
||
|
import { initFadeMotion, initZoomMotion } from '../../style/motion';
|
||
|
import { genComponentStyleHook, mergeToken } from '../../theme/internal';
|
||
|
export const genBoxStyle = position => ({
|
||
|
position: position || 'absolute',
|
||
|
inset: 0
|
||
|
});
|
||
|
export const genImageMaskStyle = token => {
|
||
|
const {
|
||
|
iconCls,
|
||
|
motionDurationSlow,
|
||
|
paddingXXS,
|
||
|
marginXXS,
|
||
|
prefixCls,
|
||
|
colorTextLightSolid
|
||
|
} = token;
|
||
|
return {
|
||
|
position: 'absolute',
|
||
|
inset: 0,
|
||
|
display: 'flex',
|
||
|
alignItems: 'center',
|
||
|
justifyContent: 'center',
|
||
|
color: colorTextLightSolid,
|
||
|
background: new TinyColor('#000').setAlpha(0.5).toRgbString(),
|
||
|
cursor: 'pointer',
|
||
|
opacity: 0,
|
||
|
transition: `opacity ${motionDurationSlow}`,
|
||
|
[`.${prefixCls}-mask-info`]: Object.assign(Object.assign({}, textEllipsis), {
|
||
|
padding: `0 ${paddingXXS}px`,
|
||
|
[iconCls]: {
|
||
|
marginInlineEnd: marginXXS,
|
||
|
svg: {
|
||
|
verticalAlign: 'baseline'
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
};
|
||
|
};
|
||
|
export const genPreviewOperationsStyle = token => {
|
||
|
const {
|
||
|
previewCls,
|
||
|
modalMaskBg,
|
||
|
paddingSM,
|
||
|
marginXL,
|
||
|
margin,
|
||
|
paddingLG,
|
||
|
previewOperationColorDisabled,
|
||
|
previewOperationHoverColor,
|
||
|
motionDurationSlow,
|
||
|
iconCls,
|
||
|
colorTextLightSolid
|
||
|
} = token;
|
||
|
const operationBg = new TinyColor(modalMaskBg).setAlpha(0.1);
|
||
|
const operationBgHover = operationBg.clone().setAlpha(0.2);
|
||
|
return {
|
||
|
[`${previewCls}-footer`]: {
|
||
|
position: 'fixed',
|
||
|
bottom: marginXL,
|
||
|
left: {
|
||
|
_skip_check_: true,
|
||
|
value: 0
|
||
|
},
|
||
|
width: '100%',
|
||
|
display: 'flex',
|
||
|
flexDirection: 'column',
|
||
|
alignItems: 'center',
|
||
|
color: token.previewOperationColor
|
||
|
},
|
||
|
[`${previewCls}-progress`]: {
|
||
|
marginBottom: margin
|
||
|
},
|
||
|
[`${previewCls}-close`]: {
|
||
|
position: 'fixed',
|
||
|
top: marginXL,
|
||
|
right: {
|
||
|
_skip_check_: true,
|
||
|
value: marginXL
|
||
|
},
|
||
|
display: 'flex',
|
||
|
color: colorTextLightSolid,
|
||
|
backgroundColor: operationBg.toRgbString(),
|
||
|
borderRadius: '50%',
|
||
|
padding: paddingSM,
|
||
|
outline: 0,
|
||
|
border: 0,
|
||
|
cursor: 'pointer',
|
||
|
transition: `all ${motionDurationSlow}`,
|
||
|
'&:hover': {
|
||
|
backgroundColor: operationBgHover.toRgbString()
|
||
|
},
|
||
|
[`& > ${iconCls}`]: {
|
||
|
fontSize: token.previewOperationSize
|
||
|
}
|
||
|
},
|
||
|
[`${previewCls}-operations`]: {
|
||
|
display: 'flex',
|
||
|
alignItems: 'center',
|
||
|
padding: `0 ${paddingLG}px`,
|
||
|
backgroundColor: operationBg.toRgbString(),
|
||
|
borderRadius: 100,
|
||
|
'&-operation': {
|
||
|
marginInlineStart: paddingSM,
|
||
|
padding: paddingSM,
|
||
|
cursor: 'pointer',
|
||
|
transition: `all ${motionDurationSlow}`,
|
||
|
userSelect: 'none',
|
||
|
[`&:not(${previewCls}-operations-operation-disabled):hover > ${iconCls}`]: {
|
||
|
color: previewOperationHoverColor
|
||
|
},
|
||
|
'&-disabled': {
|
||
|
color: previewOperationColorDisabled,
|
||
|
cursor: 'not-allowed'
|
||
|
},
|
||
|
'&:first-of-type': {
|
||
|
marginInlineStart: 0
|
||
|
},
|
||
|
[`& > ${iconCls}`]: {
|
||
|
fontSize: token.previewOperationSize
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
};
|
||
|
export const genPreviewSwitchStyle = token => {
|
||
|
const {
|
||
|
modalMaskBg,
|
||
|
iconCls,
|
||
|
previewOperationColorDisabled,
|
||
|
previewCls,
|
||
|
zIndexPopup,
|
||
|
motionDurationSlow
|
||
|
} = token;
|
||
|
const operationBg = new TinyColor(modalMaskBg).setAlpha(0.1);
|
||
|
const operationBgHover = operationBg.clone().setAlpha(0.2);
|
||
|
return {
|
||
|
[`${previewCls}-switch-left, ${previewCls}-switch-right`]: {
|
||
|
position: 'fixed',
|
||
|
insetBlockStart: '50%',
|
||
|
zIndex: zIndexPopup + 1,
|
||
|
display: 'flex',
|
||
|
alignItems: 'center',
|
||
|
justifyContent: 'center',
|
||
|
width: token.imagePreviewSwitchSize,
|
||
|
height: token.imagePreviewSwitchSize,
|
||
|
marginTop: -token.imagePreviewSwitchSize / 2,
|
||
|
color: token.previewOperationColor,
|
||
|
background: operationBg.toRgbString(),
|
||
|
borderRadius: '50%',
|
||
|
transform: `translateY(-50%)`,
|
||
|
cursor: 'pointer',
|
||
|
transition: `all ${motionDurationSlow}`,
|
||
|
userSelect: 'none',
|
||
|
'&:hover': {
|
||
|
background: operationBgHover.toRgbString()
|
||
|
},
|
||
|
[`&-disabled`]: {
|
||
|
'&, &:hover': {
|
||
|
color: previewOperationColorDisabled,
|
||
|
background: 'transparent',
|
||
|
cursor: 'not-allowed',
|
||
|
[`> ${iconCls}`]: {
|
||
|
cursor: 'not-allowed'
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
[`> ${iconCls}`]: {
|
||
|
fontSize: token.previewOperationSize
|
||
|
}
|
||
|
},
|
||
|
[`${previewCls}-switch-left`]: {
|
||
|
insetInlineStart: token.marginSM
|
||
|
},
|
||
|
[`${previewCls}-switch-right`]: {
|
||
|
insetInlineEnd: token.marginSM
|
||
|
}
|
||
|
};
|
||
|
};
|
||
|
export const genImagePreviewStyle = token => {
|
||
|
const {
|
||
|
motionEaseOut,
|
||
|
previewCls,
|
||
|
motionDurationSlow,
|
||
|
componentCls
|
||
|
} = token;
|
||
|
return [{
|
||
|
[`${componentCls}-preview-root`]: {
|
||
|
[previewCls]: {
|
||
|
height: '100%',
|
||
|
textAlign: 'center',
|
||
|
pointerEvents: 'none'
|
||
|
},
|
||
|
[`${previewCls}-body`]: Object.assign(Object.assign({}, genBoxStyle()), {
|
||
|
overflow: 'hidden'
|
||
|
}),
|
||
|
[`${previewCls}-img`]: {
|
||
|
maxWidth: '100%',
|
||
|
maxHeight: '70%',
|
||
|
verticalAlign: 'middle',
|
||
|
transform: 'scale3d(1, 1, 1)',
|
||
|
cursor: 'grab',
|
||
|
transition: `transform ${motionDurationSlow} ${motionEaseOut} 0s`,
|
||
|
userSelect: 'none',
|
||
|
'&-wrapper': Object.assign(Object.assign({}, genBoxStyle()), {
|
||
|
transition: `transform ${motionDurationSlow} ${motionEaseOut} 0s`,
|
||
|
// https://github.com/ant-design/ant-design/issues/39913
|
||
|
// TailwindCSS will reset img default style.
|
||
|
// Let's set back.
|
||
|
display: 'flex',
|
||
|
justifyContent: 'center',
|
||
|
alignItems: 'center',
|
||
|
'& > *': {
|
||
|
pointerEvents: 'auto'
|
||
|
},
|
||
|
'&::before': {
|
||
|
display: 'inline-block',
|
||
|
width: 1,
|
||
|
height: '50%',
|
||
|
marginInlineEnd: -1,
|
||
|
content: '""'
|
||
|
}
|
||
|
})
|
||
|
},
|
||
|
[`${previewCls}-moving`]: {
|
||
|
[`${previewCls}-preview-img`]: {
|
||
|
cursor: 'grabbing',
|
||
|
'&-wrapper': {
|
||
|
transitionDuration: '0s'
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
// Override
|
||
|
{
|
||
|
[`${componentCls}-preview-root`]: {
|
||
|
[`${previewCls}-wrap`]: {
|
||
|
zIndex: token.zIndexPopup
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
// Preview operations & switch
|
||
|
{
|
||
|
[`${componentCls}-preview-operations-wrapper`]: {
|
||
|
position: 'fixed',
|
||
|
zIndex: token.zIndexPopup + 1
|
||
|
},
|
||
|
'&': [genPreviewOperationsStyle(token), genPreviewSwitchStyle(token)]
|
||
|
}];
|
||
|
};
|
||
|
const genImageStyle = token => {
|
||
|
const {
|
||
|
componentCls
|
||
|
} = token;
|
||
|
return {
|
||
|
// ============================== image ==============================
|
||
|
[componentCls]: {
|
||
|
position: 'relative',
|
||
|
display: 'inline-block',
|
||
|
[`${componentCls}-img`]: {
|
||
|
width: '100%',
|
||
|
height: 'auto',
|
||
|
verticalAlign: 'middle'
|
||
|
},
|
||
|
[`${componentCls}-img-placeholder`]: {
|
||
|
backgroundColor: token.colorBgContainerDisabled,
|
||
|
backgroundImage: "url('')",
|
||
|
backgroundRepeat: 'no-repeat',
|
||
|
backgroundPosition: 'center center',
|
||
|
backgroundSize: '30%'
|
||
|
},
|
||
|
[`${componentCls}-mask`]: Object.assign({}, genImageMaskStyle(token)),
|
||
|
[`${componentCls}-mask:hover`]: {
|
||
|
opacity: 1
|
||
|
},
|
||
|
[`${componentCls}-placeholder`]: Object.assign({}, genBoxStyle())
|
||
|
}
|
||
|
};
|
||
|
};
|
||
|
const genPreviewMotion = token => {
|
||
|
const {
|
||
|
previewCls
|
||
|
} = token;
|
||
|
return {
|
||
|
[`${previewCls}-root`]: initZoomMotion(token, 'zoom'),
|
||
|
[`&`]: initFadeMotion(token, true)
|
||
|
};
|
||
|
};
|
||
|
// ============================== Export ==============================
|
||
|
export default genComponentStyleHook('Image', token => {
|
||
|
const previewCls = `${token.componentCls}-preview`;
|
||
|
const imageToken = mergeToken(token, {
|
||
|
previewCls,
|
||
|
modalMaskBg: new TinyColor('#000').setAlpha(0.45).toRgbString(),
|
||
|
imagePreviewSwitchSize: token.controlHeightLG
|
||
|
});
|
||
|
return [genImageStyle(imageToken), genImagePreviewStyle(imageToken), genModalMaskStyle(mergeToken(imageToken, {
|
||
|
componentCls: previewCls
|
||
|
})), genPreviewMotion(imageToken)];
|
||
|
}, token => ({
|
||
|
zIndexPopup: token.zIndexPopupBase + 80,
|
||
|
previewOperationColor: new TinyColor(token.colorTextLightSolid).setAlpha(0.65).toRgbString(),
|
||
|
previewOperationHoverColor: new TinyColor(token.colorTextLightSolid).setAlpha(0.85).toRgbString(),
|
||
|
previewOperationColorDisabled: new TinyColor(token.colorTextLightSolid).setAlpha(0.25).toRgbString(),
|
||
|
previewOperationSize: token.fontSizeIcon * 1.5 // FIXME: fontSizeIconLG
|
||
|
}));
|