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

542 lines
14 KiB
TypeScript
Raw Normal View History

// deps-lint-skip-all
2022-04-06 22:14:16 +08:00
import type { CSSObject } from '@ant-design/cssinjs';
import { zoomIn } from '../../style/motion';
2022-04-06 22:14:16 +08:00
import type { AliasToken, FullToken, GenerateStyle } from '../../_util/theme';
import { genComponentStyleHook, mergeToken, resetComponent } from '../../_util/theme';
2022-04-06 22:14:16 +08:00
interface FormToken extends FullToken<'Form'> {
formItemCls: string;
rootPrefixCls: string;
}
const resetForm = (token: AliasToken): CSSObject => ({
legend: {
display: 'block',
width: '100%',
2022-06-02 22:21:21 +08:00
marginBottom: token.marginLG,
padding: 0,
color: token.colorTextSecondary,
fontSize: token.fontSizeLG,
lineHeight: 'inherit',
border: 0,
borderBottom: `${token.controlLineWidth}px ${token.controlLineType} ${token.colorBorder}`,
},
label: {
fontSize: token.fontSize,
},
'input[type="search"]': {
boxSizing: 'border-box',
},
// Position radios and checkboxes better
'input[type="radio"], input[type="checkbox"]': {
lineHeight: 'normal',
},
'input[type="file"]': {
display: 'block',
},
// Make range inputs behave like textual form controls
'input[type="range"]': {
display: 'block',
width: '100%',
},
// Make multiple select elements height not fixed
'select[multiple], select[size]': {
height: 'auto',
},
// Focus for file, radio, and checkbox
[`input[type='file']:focus,
input[type='radio']:focus,
input[type='checkbox']:focus`]: {
outline: 0,
boxShadow: `0 0 0 ${token.controlOutlineWidth}px ${token.colorPrimaryOutline}`,
},
// Adjust output element
output: {
display: 'block',
paddingTop: 15,
color: token.colorText,
fontSize: token.fontSize,
lineHeight: token.lineHeight,
},
});
const genFormSize = (token: FormToken, height: number): CSSObject => ({
[`${token.formItemCls}-label > label`]: {
height,
},
[`${token.formItemCls}-control-input`]: {
minHeight: height,
},
});
const genFormStyle: GenerateStyle<FormToken> = token => {
2022-04-06 22:14:16 +08:00
const { componentCls } = token;
return {
2022-04-06 22:14:16 +08:00
[token.componentCls]: {
...resetComponent(token),
...resetForm(token),
2022-04-06 22:14:16 +08:00
[`${componentCls}-text`]: {
display: 'inline-block',
paddingInlineEnd: token.paddingSM,
},
// ================================================================
// = Size =
// ================================================================
'&-small': {
...genFormSize(token, token.controlHeightSM),
},
'&-large': {
...genFormSize(token, token.controlHeightLG),
},
},
};
};
const genFormItemStyle: GenerateStyle<FormToken> = token => {
2022-04-06 22:14:16 +08:00
const { formItemCls, iconCls, componentCls, rootPrefixCls } = token;
return {
[formItemCls]: {
...resetComponent(token),
marginBottom: token.marginLG,
verticalAlign: 'top',
// We delay one frame (0.017s) here to let CSSMotion goes
transition: `margin-bottom ${token.motionDurationSlow} 0.017s linear`,
'&-with-help': {
marginBottom: 0,
transition: 'none',
},
[`&-hidden,
&-hidden.${rootPrefixCls}-row`]: {
// https://github.com/ant-design/ant-design/issues/26141
display: 'none',
},
'&-has-warning': {
[`${formItemCls}-split`]: {
color: token.colorError,
},
},
'&-has-error': {
[`${formItemCls}-split`]: {
color: token.colorWarning,
},
},
// ==============================================================
// = Label =
// ==============================================================
[`${formItemCls}-label`]: {
display: 'inline-block',
flexGrow: 0,
overflow: 'hidden',
whiteSpace: 'nowrap',
textAlign: 'end',
verticalAlign: 'middle',
'&-left': {
textAlign: 'start',
},
'&-wrap': {
overflow: 'unset',
lineHeight: `${token.lineHeight} - 0.25em`,
whiteSpace: 'unset',
},
'> label': {
position: 'relative',
display: 'inline-flex',
alignItems: 'center',
maxWidth: '100%',
height: token.controlHeight,
color: token.colorTextHeading,
fontSize: token.fontSize,
[`> ${iconCls}`]: {
fontSize: token.fontSize,
verticalAlign: 'top',
},
// Required mark
[`&${formItemCls}-required:not(${formItemCls}-required-mark-optional)::before`]: {
display: 'inline-block',
marginInlineEnd: token.marginXXS,
color: token.colorError,
fontSize: token.fontSize,
fontFamily: 'SimSun, sans-serif',
lineHeight: 1,
content: '"*"',
2022-04-06 22:14:16 +08:00
[`${componentCls}-hide-required-mark &`]: {
display: 'none',
},
},
// Optional mark
[`${formItemCls}-optional`]: {
display: 'inline-block',
marginInlineStart: token.marginXXS,
color: token.colorTextSecondary,
2022-04-06 22:14:16 +08:00
[`${componentCls}-hide-required-mark &`]: {
display: 'none',
},
},
// Optional mark
[`${formItemCls}-tooltip`]: {
color: token.colorTextSecondary,
cursor: 'help',
writingMode: 'horizontal-tb',
marginInlineStart: token.marginXXS,
},
'&::after': {
content: '":"',
position: 'relative',
marginBlock: 0,
2022-06-02 22:21:21 +08:00
marginInlineStart: token.marginXXS / 2,
marginInlineEnd: token.marginXS,
},
[`&${formItemCls}-no-colon::after`]: {
content: '" "',
},
},
},
// ==============================================================
// = Input =
// ==============================================================
[`${formItemCls}-control`]: {
display: 'flex',
flexDirection: 'column',
flexGrow: 1,
[`&:first-child:not([class^=~"'${rootPrefixCls}-col-'"]):not([class*=~"' ${rootPrefixCls}-col-'"])`]:
{
width: '100%',
},
'&-input': {
position: 'relative',
display: 'flex',
alignItems: 'center',
minHeight: token.controlHeight,
'&-content': {
flex: 'auto',
maxWidth: '100%',
},
},
},
// ==============================================================
// = Explain =
// ==============================================================
[formItemCls]: {
'&-explain, &-extra': {
clear: 'both',
color: token.colorTextSecondary,
fontSize: token.fontSize,
lineHeight: token.lineHeight,
transition: `color ${token.motionDurationMid} ${token.motionEaseOut}`, // sync input color transition
},
'&-explain-connected': {
height: 0,
minHeight: 0,
opacity: 0,
},
'&-extra': {
minHeight: token.controlHeightSM,
},
'&-explain': {
'&-error': {
color: token.colorError,
},
'&-warning': {
color: token.colorWarning,
},
},
},
[`&-with-help ${formItemCls}-explain`]: {
height: 'auto',
minHeight: token.controlHeightSM,
opacity: 1,
},
// ==============================================================
// = Feedback Icon =
// ==============================================================
[`${formItemCls}-feedback-icon`]: {
fontSize: token.fontSize,
textAlign: 'center',
visibility: 'visible',
animationName: zoomIn,
animationDuration: token.motionDurationMid,
animationTimingFunction: token.motionEaseOutBack,
pointerEvents: 'none',
'&-success': {
color: token.colorSuccess,
},
'&-error': {
color: token.colorError,
},
'&-warning': {
color: token.colorWarning,
},
'&-validating': {
color: token.colorPrimary,
},
},
},
};
};
const genFormMotionStyle: GenerateStyle<FormToken> = token => {
2022-04-06 22:14:16 +08:00
const { componentCls, rootPrefixCls } = token;
return {
2022-04-06 22:14:16 +08:00
[componentCls]: {
// Explain holder
[`.${rootPrefixCls}-show-help`]: {
transition: `height ${token.motionDurationSlow} linear,
min-height ${token.motionDurationSlow} linear,
margin-bottom ${token.motionDurationSlow} ${token.motionEaseInOut},
opacity ${token.motionDurationSlow} ${token.motionEaseInOut}`,
'&-leave': {
minHeight: token.controlHeightSM,
'&-active': {
minHeight: 0,
},
},
},
// Explain
[`.${rootPrefixCls}-show-help-item`]: {
overflow: 'hidden',
transition: `height ${token.motionDurationSlow} ${token.motionEaseInOut},
opacity ${token.motionDurationSlow} ${token.motionEaseInOut},
transform ${token.motionDurationSlow} ${token.motionEaseInOut} !important`,
[`&-appear,
&-enter`]: {
transform: `translateY(-5px)`,
opacity: 0,
'&-active': {
transform: 'translateY(0)',
opacity: 1,
},
},
'&-leave-active': {
transform: `translateY(-5px)`,
},
},
},
};
};
const genHorizontalStyle: GenerateStyle<FormToken> = token => {
2022-04-06 22:14:16 +08:00
const { componentCls, formItemCls, rootPrefixCls } = token;
return {
2022-04-06 22:14:16 +08:00
[`${componentCls}-horizontal`]: {
[`${formItemCls}-label`]: {
flexGrow: 0,
},
[`${formItemCls}-control`]: {
flex: '1 1 0',
// https://github.com/ant-design/ant-design/issues/32777
// https://github.com/ant-design/ant-design/issues/33773
minWidth: 0,
},
// https://github.com/ant-design/ant-design/issues/32980
[`${formItemCls}-label.${rootPrefixCls}-col-24 + ${formItemCls}-control`]: {
minWidth: 'unset',
},
},
};
};
const genInlineStyle: GenerateStyle<FormToken> = token => {
2022-04-06 22:14:16 +08:00
const { componentCls, formItemCls } = token;
return {
2022-04-06 22:14:16 +08:00
[`${componentCls}-inline`]: {
display: 'flex',
flexWrap: 'wrap',
[formItemCls]: {
flex: 'none',
flexWrap: 'nowrap',
marginInlineEnd: token.margin,
marginBottom: 0,
'&-with-help': {
marginBottom: token.marginLG,
},
[`> ${formItemCls}-label,
> ${formItemCls}-control`]: {
display: 'inline-block',
verticalAlign: 'top',
},
[`> ${formItemCls}-label`]: {
flex: 'none',
},
2022-04-06 22:14:16 +08:00
[`${componentCls}-text`]: {
display: 'inline-block',
},
[`${formItemCls}-has-feedback`]: {
display: 'inline-block',
},
},
},
};
};
const makeVerticalLayoutLabel = (token: FormToken): CSSObject => ({
margin: 0,
padding: `0 0 ${token.paddingXS}px`,
whiteSpace: 'initial',
textAlign: 'start',
'> label': {
margin: 0,
'&::after': {
display: 'none',
},
},
});
const makeVerticalLayout = (token: FormToken): CSSObject => {
2022-04-06 22:14:16 +08:00
const { componentCls, formItemCls } = token;
return {
[`${formItemCls} ${formItemCls}-label`]: {
...makeVerticalLayoutLabel(token),
},
2022-04-06 22:14:16 +08:00
[componentCls]: {
[formItemCls]: {
flexWrap: 'wrap',
[`${formItemCls}-label,
${formItemCls}-control`]: {
flex: '0 0 100%',
maxWidth: '100%',
},
},
},
};
};
const genVerticalStyle: GenerateStyle<FormToken> = token => {
2022-04-06 22:14:16 +08:00
const { componentCls, formItemCls, rootPrefixCls } = token;
return {
2022-04-06 22:14:16 +08:00
[`${componentCls}-vertical`]: {
[formItemCls]: {
flexDirection: 'column',
'&-label > label': {
height: 'auto',
},
},
},
2022-04-06 22:14:16 +08:00
[`${componentCls}-vertical ${formItemCls}-label,
.${rootPrefixCls}-col-24${formItemCls}-label,
.${rootPrefixCls}-col-xl-24${formItemCls}-label`]: {
...makeVerticalLayoutLabel(token),
},
[`@media (max-width: ${token.screenSMMax}px)`]: {
...makeVerticalLayout(token),
2022-04-06 22:14:16 +08:00
[componentCls]: {
[`.${rootPrefixCls}-col-xs-24${formItemCls}-label`]: {
...makeVerticalLayoutLabel(token),
},
},
},
[`@media (max-width: ${token.screenSMMax}px)`]: {
2022-04-06 22:14:16 +08:00
[componentCls]: {
[`.${rootPrefixCls}-col-sm-24${formItemCls}-label`]: {
...makeVerticalLayoutLabel(token),
},
},
},
[`@media (max-width: ${token.screenMDMax}px)`]: {
2022-04-06 22:14:16 +08:00
[componentCls]: {
[`.${rootPrefixCls}-col-md-24${formItemCls}-label`]: {
...makeVerticalLayoutLabel(token),
},
},
},
[`@media (max-width: ${token.screenLGMax}px)`]: {
2022-04-06 22:14:16 +08:00
[componentCls]: {
[`.${rootPrefixCls}-col-lg-24${formItemCls}-label`]: {
...makeVerticalLayoutLabel(token),
},
},
},
};
};
// ============================== Export ==============================
2022-04-06 22:14:16 +08:00
export default genComponentStyleHook('Form', (token, { rootPrefixCls }) => {
const formToken = mergeToken<FormToken>(token, {
2022-04-06 22:14:16 +08:00
formItemCls: `${token.componentCls}-item`,
rootPrefixCls,
});
return [
2022-04-06 22:14:16 +08:00
genFormStyle(formToken),
genFormItemStyle(formToken),
genFormMotionStyle(formToken),
genHorizontalStyle(formToken),
genInlineStyle(formToken),
genVerticalStyle(formToken),
zoomIn,
];
2022-04-06 22:14:16 +08:00
});