import type { CSSObject } from '@ant-design/cssinjs'; import type { FullToken, GenerateStyle } from '../../theme/internal'; import { genComponentStyleHook, mergeToken } from '../../theme/internal'; interface GridRowToken extends FullToken<'Grid'> {} interface GridColToken extends FullToken<'Grid'> { gridColumns: number; } // ============================== Row-Shared ============================== const genGridRowStyle: GenerateStyle = (token): CSSObject => { const { componentCls } = token; return { // Grid system [componentCls]: { display: 'flex', flexFlow: 'row wrap', minWidth: 0, '&::before, &::after': { display: 'flex', }, '&-no-wrap': { flexWrap: 'nowrap', }, // The origin of the X-axis '&-start': { justifyContent: 'flex-start', }, // The center of the X-axis '&-center': { justifyContent: 'center', }, // The opposite of the X-axis '&-end': { justifyContent: 'flex-end', }, '&-space-between': { justifyContent: 'space-between', }, '&-space-around': { justifyContent: 'space-around', }, '&-space-evenly': { justifyContent: 'space-evenly', }, // Align at the top '&-top': { alignItems: 'flex-start', }, // Align at the center '&-middle': { alignItems: 'center', }, '&-bottom': { alignItems: 'flex-end', }, }, }; }; // ============================== Col-Shared ============================== const genGridColStyle: GenerateStyle = (token): CSSObject => { const { componentCls } = token; return { // Grid system [componentCls]: { position: 'relative', maxWidth: '100%', // Prevent columns from collapsing when empty minHeight: 1, }, }; }; const genLoopGridColumnsStyle = (token: GridColToken, sizeCls: string): CSSObject => { const { componentCls, gridColumns } = token; const gridColumnsStyle: CSSObject = {}; for (let i = gridColumns; i >= 0; i--) { if (i === 0) { gridColumnsStyle[`${componentCls}${sizeCls}-${i}`] = { display: 'none', }; gridColumnsStyle[`${componentCls}-push-${i}`] = { insetInlineStart: 'auto', }; gridColumnsStyle[`${componentCls}-pull-${i}`] = { insetInlineEnd: 'auto', }; gridColumnsStyle[`${componentCls}${sizeCls}-push-${i}`] = { insetInlineStart: 'auto', }; gridColumnsStyle[`${componentCls}${sizeCls}-pull-${i}`] = { insetInlineEnd: 'auto', }; gridColumnsStyle[`${componentCls}${sizeCls}-offset-${i}`] = { marginInlineStart: 0, }; gridColumnsStyle[`${componentCls}${sizeCls}-order-${i}`] = { order: 0, }; } else { gridColumnsStyle[`${componentCls}${sizeCls}-${i}`] = { display: 'block', flex: `0 0 ${(i / gridColumns) * 100}%`, maxWidth: `${(i / gridColumns) * 100}%`, }; gridColumnsStyle[`${componentCls}${sizeCls}-push-${i}`] = { insetInlineStart: `${(i / gridColumns) * 100}%`, }; gridColumnsStyle[`${componentCls}${sizeCls}-pull-${i}`] = { insetInlineEnd: `${(i / gridColumns) * 100}%`, }; gridColumnsStyle[`${componentCls}${sizeCls}-offset-${i}`] = { marginInlineStart: `${(i / gridColumns) * 100}%`, }; gridColumnsStyle[`${componentCls}${sizeCls}-order-${i}`] = { order: i, }; } } return gridColumnsStyle; }; const genGridStyle = (token: GridColToken, sizeCls: string): CSSObject => genLoopGridColumnsStyle(token, sizeCls); const genGridMediaStyle = ( token: GridColToken, screenSize: number, sizeCls: string, ): CSSObject => ({ [`@media (min-width: ${screenSize}px)`]: { ...genGridStyle(token, sizeCls), }, }); // ============================== Export ============================== export const useRowStyle = genComponentStyleHook('Grid', (token) => [genGridRowStyle(token)]); export const useColStyle = genComponentStyleHook('Grid', (token) => { const gridToken: GridColToken = mergeToken(token, { gridColumns: 24, // Row is divided into 24 parts in Grid }); const gridMediaSizesMap = { '-sm': gridToken.screenSMMin, '-md': gridToken.screenMDMin, '-lg': gridToken.screenLGMin, '-xl': gridToken.screenXLMin, '-xxl': gridToken.screenXXLMin, }; return [ genGridColStyle(gridToken), genGridStyle(gridToken, ''), genGridStyle(gridToken, '-xs'), Object.keys(gridMediaSizesMap) .map((key: keyof typeof gridMediaSizesMap) => genGridMediaStyle(gridToken, gridMediaSizesMap[key], key), ) .reduce((pre, cur) => ({ ...pre, ...cur }), {}), ]; });