diff --git a/src/layout/col-div.style.tsx b/src/layout/col-div.style.tsx new file mode 100644 index 000000000..ad53633bc --- /dev/null +++ b/src/layout/col-div.style.tsx @@ -0,0 +1,76 @@ +import styled, { css } from "styled-components"; +import { MediaQuery } from "../theme/breakpoint/media-query-helper"; + +export interface StyledDivStyleProps { + $xxsStart?: number | undefined; + $xxsSpan?: number | undefined; + $xsStart?: number | undefined; + $xsSpan?: number | undefined; + $smStart?: number | undefined; + $smSpan?: number | undefined; + $mdStart?: number | undefined; + $mdSpan?: number | undefined; + $lgStart?: number | undefined; + $lgSpan?: number | undefined; + $xlStart?: number | undefined; + $xlSpan?: number | undefined; + $xxlStart?: number | undefined; + $xxlSpan?: number | undefined; +} + +export const StyledDiv = styled.div` + position: relative; + + ${(props) => { + const { + $xxlStart, + $xxlSpan, + + $xlStart, + $xlSpan, + + $lgStart, + $lgSpan, + + $mdStart, + $mdSpan, + + $smStart, + $smSpan, + + $xsStart, + $xsSpan, + + $xxsStart, + $xxsSpan, + } = props; + + return css` + grid-column: ${$xxlStart || "auto"} / span ${$xxlSpan || 1}; + + ${MediaQuery.MaxWidth.xl} { + grid-column: ${$xlStart || "auto"} / span ${$xlSpan || 1}; + } + + ${MediaQuery.MaxWidth.lg} { + grid-column: ${$lgStart || "auto"} / span ${$lgSpan || 1}; + } + + ${MediaQuery.MaxWidth.md} { + grid-column: ${$mdStart || "auto"} / span ${$mdSpan || 1}; + } + + ${MediaQuery.MaxWidth.sm} { + grid-column: ${$smStart || "auto"} / span ${$smSpan || 1}; + } + + ${MediaQuery.MaxWidth.xs} { + grid-column: ${$xsStart || "auto"} / span ${$xsSpan || 1}; + } + + ${MediaQuery.MaxWidth.xxs} { + grid-column: ${$xxsStart || "auto"} / span ${$xxsSpan || 1}; + } + `; + }} +`; diff --git a/src/layout/col-div.tsx b/src/layout/col-div.tsx new file mode 100644 index 000000000..af86ab67a --- /dev/null +++ b/src/layout/col-div.tsx @@ -0,0 +1,126 @@ +import React from "react"; +import { ColDivProps } from "./types"; +import { StyledDiv } from "./col-div.style"; +import { BreakpointValues } from "../theme/breakpoint/theme-helper"; +import { ThemeSpec } from "../theme/types"; +import { useTheme } from "styled-components"; + +const Component = ( + props: ColDivProps, + ref: React.Ref +): JSX.Element => { + const theme = useTheme() as ThemeSpec; + + const { + xxlCols, + xlCols, + lgCols, + mdCols, + smCols, + xsCols, + xxsCols, + ...otherProps + } = props; + + const getColSpan = ( + cols: number | [number, number] | undefined, + maxCols: number, + label: string + ) => { + if (!cols) return { start: undefined, span: undefined }; + + if (process.env.NODE_ENV === "development") { + if (typeof cols === "number" && cols > maxCols) { + console.warn( + `Warning: ${label}Cols exceeds maximum (${maxCols}): ${cols}` + ); + } else if ( + Array.isArray(cols) && + (cols[0] > maxCols + 1 || cols[1] > maxCols + 1) + ) { + console.warn( + `Warning: ${label}Cols array exceeds maximum (${maxCols}): [${cols[0]}, ${cols[1]}]` + ); + } + } + + if (Array.isArray(cols)) { + const [start, end] = cols; + const span = Math.min(end - start, maxCols); + return { start, span }; + } + + return { start: undefined, span: Math.min(cols, maxCols) }; + }; + + const getStyleProps = () => { + const xxlColumnCount = BreakpointValues["xxl-column"]({ theme }); + const xlColumnCount = BreakpointValues["xl-column"]({ theme }); + const lgColumnCount = BreakpointValues["lg-column"]({ theme }); + const mdColumnCount = BreakpointValues["md-column"]({ theme }); + const smColumnCount = BreakpointValues["sm-column"]({ theme }); + const xsColumnCount = BreakpointValues["xs-column"]({ theme }); + const xxsColumnCount = BreakpointValues["xxs-column"]({ theme }); + + const xxlStartSpan = getColSpan( + xxlCols || + xlCols || + lgCols || + mdCols || + smCols || + xsCols || + xxsCols, + xxlColumnCount, + "xxl" + ); + + const xlStartSpan = getColSpan( + xlCols || lgCols || mdCols || smCols || xsCols || xxsCols, + xlColumnCount, + "xl" + ); + + const lgStartSpan = getColSpan( + lgCols || mdCols || smCols || xsCols || xxsCols, + lgColumnCount, + "lg" + ); + + const mdStartSpan = getColSpan( + mdCols || smCols || xsCols || xxsCols, + mdColumnCount, + "md" + ); + + const smStartSpan = getColSpan( + smCols || xsCols || xxsCols, + smColumnCount, + "sm" + ); + + const xsStartSpan = getColSpan(xsCols || xxsCols, xsColumnCount, "xs"); + + const xxsStartSpan = getColSpan(xxsCols, xxsColumnCount, "xxs"); + + return { + $xxlStart: xxlStartSpan.start, + $xxlSpan: xxlStartSpan.span, + $xlStart: xlStartSpan.start, + $xlSpan: xlStartSpan.span, + $lgStart: lgStartSpan.start, + $lgSpan: lgStartSpan.span, + $mdStart: mdStartSpan.start, + $mdSpan: mdStartSpan.span, + $smStart: smStartSpan.start, + $smSpan: smStartSpan.span, + $xsStart: xsStartSpan.start, + $xsSpan: xsStartSpan.span, + $xxsStart: xxsStartSpan.start, + $xxsSpan: xxsStartSpan.span, + }; + }; + + return ; +}; + +export const ColDiv = React.forwardRef(Component); diff --git a/src/layout/index.ts b/src/layout/index.ts index 0767e37d0..3337d54de 100644 --- a/src/layout/index.ts +++ b/src/layout/index.ts @@ -1,3 +1,4 @@ +import { ColDiv } from "./col-div"; import { Container } from "./container"; import { Content } from "./content"; import { Section } from "./section"; @@ -6,6 +7,7 @@ export const Layout = { Section: Section, Container: Container, Content: Content, + ColDiv: ColDiv, }; export * from "./types"; diff --git a/src/layout/types.ts b/src/layout/types.ts index a6d3bc227..7b97d93d1 100644 --- a/src/layout/types.ts +++ b/src/layout/types.ts @@ -1,3 +1,6 @@ +import { DefaultTheme } from "styled-components"; +import { AddOne, Range } from "../util/utility-types"; + interface CommonLayoutProps extends React.HTMLAttributes { children: React.ReactNode; "data-testid"?: string | undefined; @@ -14,3 +17,34 @@ export interface ContainerProps extends CommonLayoutProps { export interface SectionProps extends CommonLayoutProps {} export interface ContentProps extends ContainerProps {} + +export type ColSpan = Max extends number + ? Range | [Range>, Range>] | undefined + : number | [number, number] | undefined; + +export type BreakpointSpan< + Breakpoint extends keyof DefaultTheme["maxColumns"] +> = DefaultTheme["maxColumns"] extends Record< + Breakpoint, + infer Max extends number +> + ? ColSpan + : number | [number, number] | undefined; +export interface ColProps { + /** + * Specifies the number of columns to be spanned across for any breakpoint. + * If an array is specified, the format is [startCol, endCol]. + */ + xxlCols?: BreakpointSpan<"xxl">; + xlCols?: BreakpointSpan<"xl">; + lgCols?: BreakpointSpan<"lg">; + mdCols?: BreakpointSpan<"md">; + smCols?: BreakpointSpan<"sm">; + xsCols?: BreakpointSpan<"xs">; + xxsCols?: BreakpointSpan<"xxs">; +} +export interface ColDivProps + extends React.HTMLAttributes, + ColProps { + "data-testid"?: string; +} diff --git a/src/util/utility-types.ts b/src/util/utility-types.ts index 404a0b9dd..cff985297 100644 --- a/src/util/utility-types.ts +++ b/src/util/utility-types.ts @@ -10,3 +10,19 @@ export type RequiredKeys = { // eslint-disable-next-line @typescript-eslint/ban-types [K in keyof T]-?: {} extends { [P in K]: T[K] } ? never : K; }[keyof T]; + +// Gets a union of numbers from 1 to N e.g. Range<3> evaluates to 1 | 2 |3 +export type Range< + N extends number, + Result extends number[] = [] +> = Result["length"] extends N + ? Exclude + : Range; + +//Increments a numeric literal e.g. AddOne<1> evaluates to 2 +export type AddOne< + N extends number, + Result extends number[] = [] +> = Result["length"] extends N + ? [...Result, Result["length"]]["length"] + : AddOne; diff --git a/stories/layout-test/Coldiv.stories.tsx b/stories/layout-test/Coldiv.stories.tsx new file mode 100644 index 000000000..8daa77027 --- /dev/null +++ b/stories/layout-test/Coldiv.stories.tsx @@ -0,0 +1,94 @@ +import { ThemeProvider } from "styled-components"; +import { ColDiv } from "../../src/layout/col-div"; +import { Content } from "../../src/layout/content"; +import { mockTheme } from "./mock-theme"; +import { Layout } from "../../src/layout"; + +export default { + title: "Layout-Test/ColDiv", + component: ColDiv, +}; + +// Default view span 1 column +export const Default = () => ( + + + + Default ColDiv + + + +); + +// Mobile config: spans across 3 columns in mobile view +export const MobileColsOnly = () => ( + + + Mobile: Span 3 columns + + +); + +// Tablet config: spans across 6 columns in tablet view +export const TabletColsOnly = () => ( + + + + Tablet: Span 6 columns + + + +); + +// Desktop config: spans across 8 columns in desktop view +export const DesktopColsOnly = () => ( + + + + Desktop: Span 8 columns + + + +); + +// Mixed config: spans across different columns in mobile, tablet, and desktop view +export const MixedCols = () => ( + + + + Mobile: Span 2 columns, Tablet: Span 4 columns, Desktop: Span 6 + columns + + + +); + +// Using start and span values: spans across a specific range of columns +export const StartAndSpan = () => ( + + + + Mobile: Takes up span 4 columns starting from 1 | Tablet: Takes + up span 3 columns starting from 3 | Desktop: Takes up span 6 + columns starting from 4 + + + +); diff --git a/tests/layout/layout-coldiv.spec.tsx b/tests/layout/layout-coldiv.spec.tsx new file mode 100644 index 000000000..772e6c196 --- /dev/null +++ b/tests/layout/layout-coldiv.spec.tsx @@ -0,0 +1,110 @@ +import { render } from "@testing-library/react"; +import "jest-styled-components"; +import { ThemeProvider } from "styled-components"; +import { ColDiv } from "../../src/layout/col-div"; +import { ThemeSpec } from "../../src/theme/types"; + +const mockTheme: ThemeSpec = { + colourScheme: "lifesg", + fontScheme: "lifesg", + animationScheme: "lifesg", + borderScheme: "lifesg", + spacingScheme: "lifesg", + radiusScheme: "lifesg", + breakpointScheme: "lifesg", +}; + +describe("ColDiv Component", () => { + it("should render with default settings (spanning 1 column)", () => { + const { container } = render( + + Default ColDiv + + ); + + expect(container.firstChild).toHaveStyleRule( + "grid-column", + "auto / span 1" + ); + }); + + it("should correctly apply xxs column span", () => { + const { container } = render( + + XXS ColDiv + + ); + + expect(container.firstChild).toHaveStyleRule( + "grid-column", + "auto / span 2", + { + media: `screen and (max-width: 320px)`, + } + ); + }); + + it("should correctly apply xs column span", () => { + const { container } = render( + + XS ColDiv + + ); + + expect(container.firstChild).toHaveStyleRule( + "grid-column", + "auto / span 3", + { + media: `screen and (max-width: 375px)`, + } + ); + }); + + it("should correctly apply sm column span", () => { + const { container } = render( + + SM ColDiv + + ); + + expect(container.firstChild).toHaveStyleRule( + "grid-column", + "auto / span 4", + { + media: `screen and (max-width: 420px)`, + } + ); + }); + + it("should correctly apply start and span for xxs", () => { + const { container } = render( + + XXS Start and Span ColDiv + + ); + + expect(container.firstChild).toHaveStyleRule( + "grid-column", + "1 / span 2", + { + media: `screen and (max-width: 320px)`, + } + ); + }); + + it("should correctly apply start and span for lg", () => { + const { container } = render( + + LG Start and Span ColDiv + + ); + + expect(container.firstChild).toHaveStyleRule( + "grid-column", + "2 / span 4", + { + media: `screen and (max-width: 1023px)`, + } + ); + }); +});