diff --git a/.prettierignore b/.prettierignore index b035b468a2c0468add11ae02ec434ef6e026ff90..710d003053e9b31e9746dba7ab738d6f8da56e42 100644 --- a/.prettierignore +++ b/.prettierignore @@ -5,6 +5,7 @@ nginx *.yml *.svg *.ttf +*.drawio # Any dotfiles/dirs \.* diff --git a/README.md b/README.md index c570acd9118249b33d09e5a6c149bc95ef8f564c..f2e11f98724a01e45302d0dedb1dbea1a86fa352 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,25 @@ This is the code implementation of the design system as according to the [Figma](https://www.figma.com/file/WRWUWArhAkmxpA5QJVFw1i/Rainbow-Design-System) designs. -Installation of the package in client application is done by creating a `.npmrc` file in the repo with the following content: +The project is based on React, Typescript and Vite. Components are extended from [NextUI](https://nextui.org/) as much as possible. +[Tailwind](https://tailwindcss.com/docs) is used for styling, and [Storybook](https://storybook.js.org/docs/react/get-started/why-storybook) for documentation. + +It is also published as a component library for other Astron apps to use: + +<img src="src/assets/design-system.svg" /> + +## Deployment + +Deployment is automated via Gitlab CI/CD. + +Test environment: https://sdc-dev.astron.nl/design-system/ +Production environment: https://sdc.astron.nl/design-system/ + +Release notes: https://git.astron.nl/astron-sdc/design-system/-/releases + +## Installation + +Installation of the package in client application is done by creating a `.npmrc` file in the repo with the following: ``` @astron-sdc:registry=https://git.astron.nl/api/v4/packages/npm/ @@ -15,22 +33,6 @@ and running: npm i @astron-sdc/design-system ``` -The project is based on React, Typescript and Vite. It uses [Tailwind](https://tailwindcss.com/docs) -for styling, and [Storybook](https://storybook.js.org/docs/react/get-started/why-storybook) -as documentation. - -Aside from hosting the documentation at Astron, there is also an idea to publish this -repository as a component library for other Astron apps to use: - -<img src="src/assets/workflow.png" height=500 /> - -## Deployment - -Deployment is automated via Gitlab CI/CD. - -Test environment: https://sdc-dev.astron.nl/design-system/ -Production environment: https://sdc.astron.nl/design-system/ - ## Contact This repository is maintained by Team Rainbow. Feel free to contact us on @@ -47,16 +49,22 @@ npm run storybook to see the Storybook at http://localhost:6006/. +### Docs + General documentation that is not related to any component in particular are written as `.mdx` files in the `src/docs/` directory, and can be viewed at http://localhost:6006/?path=/docs/. +### Components + +For consistency, please extend from the NextUI component library as much as possible. + Each component is paired with a story file, which is responsible for showcasing said component on the Storybook. The story file for `src/components/ComponentName` shall be found at `src/stories/ComponentName.stories.ts`. -### How to +### Icons -#### Icons +This is the current workflow: Download or copy the SVG code from Figma. Save it into `src/assets/icons/your-file-name.svg`. In the file, change the stroke color to `currentColor`. This is to enable dynamic stroke colors. @@ -67,11 +75,13 @@ Add `"your-file-name"` to `iconNameOptions` array and the `iconMap` dict of `Ico <Icon name="your-file-name" color="secondary" /> ``` -#### Colors +### Classnames + +We use `clsx` to concatenate classnames. Since Tailwind [does not support](https://tailwindcss.com/docs/content-configuration#dynamic-class-names) -dynamic class names, there are some functions in `src/components/colors.ts` to help reduce boilerplate -code. They also convert the semantic color names in NextUI to the color names that we use in Figma. +dynamic class names, there are some functions in `src/components/utils/classes.ts` to help reduce boilerplate +code. They also convert the color names that we use in Figma into the semantic color names in NextUI. Please use them (or add new ones there) as much as possible instead of supplying colors via another manner. So we can keep the styling code consistent and organized. diff --git a/package-lock.json b/package-lock.json index ea77b209af3117ed0e0fe0ce2abf2936ee5ece07..bc5e613abcbdddc5a485a75d234b41620e2a34c5 100644 --- a/package-lock.json +++ b/package-lock.json @@ -33,6 +33,7 @@ "@typescript-eslint/parser": "^6.0.0", "@vitejs/plugin-react": "^4.0.3", "autoprefixer": "^10.4.16", + "clsx": "^2.1.0", "concurrently": "^8.2.2", "date-fns": "^2.30.0", "eslint": "^8.45.0", @@ -52,7 +53,7 @@ "react-slider": "^2.0.6", "react-svg": "^16.1.29", "semantic-release": "^22.0.8", - "storybook": "^7.5.3", + "storybook": "^7.6.10", "tailwindcss": "^3.3.3", "typescript": "^5.0.2", "vite": "^4.4.5", @@ -4932,6 +4933,15 @@ "tailwind-variants": ">=0.1.13" } }, + "node_modules/@nextui-org/system-rsc/node_modules/clsx": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz", + "integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, "node_modules/@nextui-org/table": { "version": "2.0.27", "resolved": "https://registry.npmjs.org/@nextui-org/table/-/table-2.0.27.tgz", @@ -6320,6 +6330,15 @@ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0" } }, + "node_modules/@react-aria/focus/node_modules/clsx": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz", + "integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, "node_modules/@react-aria/form": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/@react-aria/form/-/form-3.0.0.tgz", @@ -6731,6 +6750,15 @@ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0" } }, + "node_modules/@react-aria/utils/node_modules/clsx": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz", + "integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, "node_modules/@react-aria/visually-hidden": { "version": "3.8.7", "resolved": "https://registry.npmjs.org/@react-aria/visually-hidden/-/visually-hidden-3.8.7.tgz", @@ -13105,9 +13133,9 @@ } }, "node_modules/clsx": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/clsx/-/clsx-1.2.1.tgz", - "integrity": "sha512-EcR6r5a8bj6pu3ycsa/E/cKVGuTgZJZdsyUYHOksG/UHIiKfjxzRxYJpyVBwYaQeOvghal9fcc4PidlgzugAQg==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.0.tgz", + "integrity": "sha512-m3iNNWpd9rl3jvvcBnu70ylMdrXt8Vlq4HYadnU5fwcOtvkSQWPmj7amUcDT2qYI7risszBjI5AUIUox9D16pg==", "dev": true, "engines": { "node": ">=6" diff --git a/package.json b/package.json index 7b54f7dc349fe36739be1b0233f272f46114cf31..40ead48abffd544923cd29ec894e42b4c14d1b90 100644 --- a/package.json +++ b/package.json @@ -95,6 +95,7 @@ "@typescript-eslint/parser": "^6.0.0", "@vitejs/plugin-react": "^4.0.3", "autoprefixer": "^10.4.16", + "clsx": "^2.1.0", "concurrently": "^8.2.2", "date-fns": "^2.30.0", "eslint": "^8.45.0", diff --git a/src/assets/design-system.drawio b/src/assets/design-system.drawio new file mode 100644 index 0000000000000000000000000000000000000000..888834b37dd696b443687344d440b7df587d132d --- /dev/null +++ b/src/assets/design-system.drawio @@ -0,0 +1,104 @@ +<mxfile host="app.diagrams.net" modified="2024-01-25T14:37:52.745Z" agent="Mozilla/5.0 (X11; Linux x86_64; rv:122.0) Gecko/20100101 Firefox/122.0" etag="oWlL_CtQoktVZJd4I5d9" version="23.0.1" type="device"> + <diagram name="Page-1" id="wt0zWwvxc1aIomN0vQJN"> + <mxGraphModel dx="1434" dy="870" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1169" pageHeight="827" math="0" shadow="0"> + <root> + <mxCell id="0" /> + <mxCell id="1" parent="0" /> + <mxCell id="wMKxM34ov-EKaF612LfR-51" value="" style="rounded=0;whiteSpace=wrap;html=1;strokeColor=none;" vertex="1" parent="1"> + <mxGeometry x="20" y="30" width="940" height="445" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-1" value="<a href="https://www.figma.com/file/WRWUWArhAkmxpA5QJVFw1i/Astron-Design-System"><font style="font-size: 15px;">Figma</font></a>" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="60" y="170" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-4" value="<div style="font-size: 15px;"><a href="https://git.astron.nl/astron-sdc/design-system/"><font style="font-size: 15px;">Design System</font></a></div>" style="shape=process;whiteSpace=wrap;html=1;backgroundOutline=1;" vertex="1" parent="1"> + <mxGeometry x="210" y="270" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-47" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;entryX=1;entryY=0.75;entryDx=0;entryDy=0;exitX=0.5;exitY=0;exitDx=0;exitDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-5" target="wMKxM34ov-EKaF612LfR-43"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-5" value="<a href="https://sdc.astron.nl/design-system/"><font style="font-size: 15px;">Storybook</font></a>" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="495" y="230" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-23" value="" style="edgeStyle=orthogonalEdgeStyle;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-6" target="wMKxM34ov-EKaF612LfR-17"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-24" value="<font style="font-size: 15px;">Import</font>" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="wMKxM34ov-EKaF612LfR-23"> + <mxGeometry x="-0.0286" relative="1" as="geometry"> + <mxPoint as="offset" /> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-6" value="<a href="https://git.astron.nl/astron-sdc/design-system/-/packages"><font style="font-size: 15px;">NPM registry</font></a>" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="495" y="310" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-12" value="" style="edgeStyle=elbowEdgeStyle;elbow=horizontal;html=1;rounded=1;curved=0;sourcePerimeterSpacing=0;targetPerimeterSpacing=0;startSize=6;endSize=6;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-4" target="wMKxM34ov-EKaF612LfR-5"> + <mxGeometry relative="1" as="geometry"> + <mxPoint x="420" y="300" as="sourcePoint" /> + <mxPoint x="470" y="260" as="targetPoint" /> + <Array as="points"> + <mxPoint x="430" y="280" /> + </Array> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-14" value="" style="edgeStyle=elbowEdgeStyle;elbow=horizontal;html=1;rounded=1;curved=0;sourcePerimeterSpacing=0;targetPerimeterSpacing=0;startSize=6;endSize=6;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-4" target="wMKxM34ov-EKaF612LfR-6"> + <mxGeometry relative="1" as="geometry"> + <mxPoint x="420" y="300" as="sourcePoint" /> + <mxPoint x="470" y="340" as="targetPoint" /> + <Array as="points"> + <mxPoint x="430" y="320" /> + </Array> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-19" value="<font style="font-size: 15px;">CI/CD</font>" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="wMKxM34ov-EKaF612LfR-14"> + <mxGeometry x="-0.5415" relative="1" as="geometry"> + <mxPoint as="offset" /> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-45" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-17" target="wMKxM34ov-EKaF612LfR-27"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-17" value="" style="rounded=0;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="715" y="270" width="190" height="140" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-49" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;entryX=1;entryY=0.75;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-27" target="wMKxM34ov-EKaF612LfR-43"> + <mxGeometry relative="1" as="geometry"> + <Array as="points"> + <mxPoint x="800" y="125" /> + </Array> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-27" value="<div><br></div><div><br></div><div>User</div>" style="shape=actor;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="790" y="180" width="40" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-28" value="" style="edgeStyle=elbowEdgeStyle;elbow=horizontal;html=1;rounded=1;curved=0;sourcePerimeterSpacing=0;targetPerimeterSpacing=0;startSize=6;endSize=6;exitX=0.5;exitY=1;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-1" target="wMKxM34ov-EKaF612LfR-4"> + <mxGeometry relative="1" as="geometry"> + <mxPoint x="50" y="510" as="sourcePoint" /> + <mxPoint x="215" y="470" as="targetPoint" /> + <Array as="points"> + <mxPoint x="120" y="310" /> + </Array> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-30" value="<font style="font-size: 15px;">Code</font>" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="wMKxM34ov-EKaF612LfR-28"> + <mxGeometry x="0.175" y="-2" relative="1" as="geometry"> + <mxPoint x="16" y="-2" as="offset" /> + </mxGeometry> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-31" value="" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="738" y="298" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-32" value="" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="748" y="308" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-33" value="Astron apps" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="758" y="318" width="120" height="60" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-44" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-43" target="wMKxM34ov-EKaF612LfR-1"> + <mxGeometry relative="1" as="geometry" /> + </mxCell> + <mxCell id="wMKxM34ov-EKaF612LfR-43" value="<div><br></div><div><br></div><div>De-signer</div>" style="shape=actor;whiteSpace=wrap;html=1;" vertex="1" parent="1"> + <mxGeometry x="100" y="80" width="40" height="60" as="geometry" /> + </mxCell> + </root> + </mxGraphModel> + </diagram> +</mxfile> diff --git a/src/assets/design-system.svg b/src/assets/design-system.svg new file mode 100644 index 0000000000000000000000000000000000000000..96631b70ac773a93996a24b22a04f97bb2984034 --- /dev/null +++ b/src/assets/design-system.svg @@ -0,0 +1,4 @@ +<?xml version="1.0" encoding="UTF-8"?> +<!-- Do not edit this file with editors other than draw.io --> +<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> +<svg xmlns="http://www.w3.org/2000/svg" style="background-color: rgb(255, 255, 255);" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="940px" height="445px" viewBox="-0.5 -0.5 940 445" content="<mxfile host="app.diagrams.net" modified="2024-01-25T14:37:31.998Z" agent="Mozilla/5.0 (X11; Linux x86_64; rv:122.0) Gecko/20100101 Firefox/122.0" etag="8iPCeNckhT3gJChXQIGw" version="23.0.1" scale="1" border="0">
 <diagram name="Page-1" id="wt0zWwvxc1aIomN0vQJN">
 <mxGraphModel dx="1434" dy="870" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="1169" pageHeight="827" math="0" shadow="0">
 <root>
 <mxCell id="0" />
 <mxCell id="1" parent="0" />
 <mxCell id="wMKxM34ov-EKaF612LfR-51" value="" style="rounded=0;whiteSpace=wrap;html=1;strokeColor=none;" vertex="1" parent="1">
 <mxGeometry x="20" y="30" width="940" height="445" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-1" value="&lt;a href=&quot;https://www.figma.com/file/WRWUWArhAkmxpA5QJVFw1i/Astron-Design-System&quot;&gt;&lt;font style=&quot;font-size: 15px;&quot;&gt;Figma&lt;/font&gt;&lt;/a&gt;" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="60" y="170" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-4" value="&lt;div style=&quot;font-size: 15px;&quot;&gt;&lt;a href=&quot;https://git.astron.nl/astron-sdc/design-system/&quot;&gt;&lt;font style=&quot;font-size: 15px;&quot;&gt;Design System&lt;/font&gt;&lt;/a&gt;&lt;/div&gt;" style="shape=process;whiteSpace=wrap;html=1;backgroundOutline=1;" vertex="1" parent="1">
 <mxGeometry x="210" y="270" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-47" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;entryX=1;entryY=0.75;entryDx=0;entryDy=0;exitX=0.5;exitY=0;exitDx=0;exitDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-5" target="wMKxM34ov-EKaF612LfR-43">
 <mxGeometry relative="1" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-5" value="&lt;a href=&quot;https://sdc.astron.nl/design-system/&quot;&gt;&lt;font style=&quot;font-size: 15px;&quot;&gt;Storybook&lt;/font&gt;&lt;/a&gt;" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="495" y="230" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-23" value="" style="edgeStyle=orthogonalEdgeStyle;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-6" target="wMKxM34ov-EKaF612LfR-17">
 <mxGeometry relative="1" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-24" value="&lt;font style=&quot;font-size: 15px;&quot;&gt;Import&lt;/font&gt;" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="wMKxM34ov-EKaF612LfR-23">
 <mxGeometry x="-0.0286" relative="1" as="geometry">
 <mxPoint as="offset" />
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-6" value="&lt;a href=&quot;https://git.astron.nl/astron-sdc/design-system/-/packages&quot;&gt;&lt;font style=&quot;font-size: 15px;&quot;&gt;NPM registry&lt;/font&gt;&lt;/a&gt;" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="495" y="310" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-12" value="" style="edgeStyle=elbowEdgeStyle;elbow=horizontal;html=1;rounded=1;curved=0;sourcePerimeterSpacing=0;targetPerimeterSpacing=0;startSize=6;endSize=6;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-4" target="wMKxM34ov-EKaF612LfR-5">
 <mxGeometry relative="1" as="geometry">
 <mxPoint x="420" y="300" as="sourcePoint" />
 <mxPoint x="470" y="260" as="targetPoint" />
 <Array as="points">
 <mxPoint x="430" y="280" />
 </Array>
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-14" value="" style="edgeStyle=elbowEdgeStyle;elbow=horizontal;html=1;rounded=1;curved=0;sourcePerimeterSpacing=0;targetPerimeterSpacing=0;startSize=6;endSize=6;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-4" target="wMKxM34ov-EKaF612LfR-6">
 <mxGeometry relative="1" as="geometry">
 <mxPoint x="420" y="300" as="sourcePoint" />
 <mxPoint x="470" y="340" as="targetPoint" />
 <Array as="points">
 <mxPoint x="430" y="320" />
 </Array>
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-19" value="&lt;font style=&quot;font-size: 15px;&quot;&gt;CI/CD&lt;/font&gt;" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="wMKxM34ov-EKaF612LfR-14">
 <mxGeometry x="-0.5415" relative="1" as="geometry">
 <mxPoint as="offset" />
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-45" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-17" target="wMKxM34ov-EKaF612LfR-27">
 <mxGeometry relative="1" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-17" value="" style="rounded=0;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="715" y="270" width="190" height="140" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-49" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;entryX=1;entryY=0.75;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-27" target="wMKxM34ov-EKaF612LfR-43">
 <mxGeometry relative="1" as="geometry">
 <Array as="points">
 <mxPoint x="800" y="125" />
 </Array>
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-27" value="&lt;div&gt;&lt;br&gt;&lt;/div&gt;&lt;div&gt;&lt;br&gt;&lt;/div&gt;&lt;div&gt;User&lt;/div&gt;" style="shape=actor;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="790" y="180" width="40" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-28" value="" style="edgeStyle=elbowEdgeStyle;elbow=horizontal;html=1;rounded=1;curved=0;sourcePerimeterSpacing=0;targetPerimeterSpacing=0;startSize=6;endSize=6;exitX=0.5;exitY=1;exitDx=0;exitDy=0;entryX=0;entryY=0.5;entryDx=0;entryDy=0;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-1" target="wMKxM34ov-EKaF612LfR-4">
 <mxGeometry relative="1" as="geometry">
 <mxPoint x="50" y="510" as="sourcePoint" />
 <mxPoint x="215" y="470" as="targetPoint" />
 <Array as="points">
 <mxPoint x="120" y="310" />
 </Array>
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-30" value="&lt;font style=&quot;font-size: 15px;&quot;&gt;Code&lt;/font&gt;" style="edgeLabel;html=1;align=center;verticalAlign=middle;resizable=0;points=[];" vertex="1" connectable="0" parent="wMKxM34ov-EKaF612LfR-28">
 <mxGeometry x="0.175" y="-2" relative="1" as="geometry">
 <mxPoint x="16" y="-2" as="offset" />
 </mxGeometry>
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-31" value="" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="738" y="298" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-32" value="" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="748" y="308" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-33" value="Astron apps" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="758" y="318" width="120" height="60" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-44" value="" style="edgeStyle=orthogonalEdgeStyle;curved=1;rounded=0;orthogonalLoop=1;jettySize=auto;html=1;" edge="1" parent="1" source="wMKxM34ov-EKaF612LfR-43" target="wMKxM34ov-EKaF612LfR-1">
 <mxGeometry relative="1" as="geometry" />
 </mxCell>
 <mxCell id="wMKxM34ov-EKaF612LfR-43" value="&lt;div&gt;&lt;br&gt;&lt;/div&gt;&lt;div&gt;&lt;br&gt;&lt;/div&gt;&lt;div&gt;De-signer&lt;/div&gt;" style="shape=actor;whiteSpace=wrap;html=1;" vertex="1" parent="1">
 <mxGeometry x="100" y="80" width="40" height="60" as="geometry" />
 </mxCell>
 </root>
 </mxGraphModel>
 </diagram>
</mxfile>
"><defs/><rect fill="#ffffff" width="100%" height="100%" x="0" y="0"/><g><rect x="0" y="0" width="940" height="445" fill="rgb(255, 255, 255)" stroke="none" pointer-events="all"/><rect x="40" y="140" width="120" height="60" rx="9" ry="9" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 170px; margin-left: 41px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><a href="https://www.figma.com/file/WRWUWArhAkmxpA5QJVFw1i/Astron-Design-System" target="_blank"><font style="font-size: 15px;">Figma</font></a></div></div></div></foreignObject><image x="41" y="161.5" width="118" height="21" xlink:href=""/></switch></g><rect x="190" y="240" width="120" height="60" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><path d="M 202 240 L 202 300 M 298 240 L 298 300" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 94px; height: 1px; padding-top: 270px; margin-left: 203px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><div style="font-size: 15px;"><a href="https://git.astron.nl/astron-sdc/design-system/" target="_blank"><font style="font-size: 15px;">Design System</font></a></div></div></div></div></foreignObject><image x="203" y="252.5" width="94" height="39" xlink:href=""/></switch></g><path d="M 535 200 Q 535 95 126.37 95" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 121.12 95 L 128.12 91.5 L 126.37 95 L 128.12 98.5 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><rect x="475" y="200" width="120" height="60" rx="9" ry="9" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 230px; margin-left: 476px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><a href="https://sdc.astron.nl/design-system/" target="_blank"><font style="font-size: 15px;">Storybook</font></a></div></div></div></foreignObject><image x="476" y="221.5" width="118" height="21" xlink:href=""/></switch></g><path d="M 595 310 L 688.63 310" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 693.88 310 L 686.88 313.5 L 688.63 310 L 686.88 306.5 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 311px; margin-left: 645px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); background-color: rgb(255, 255, 255); "><div style="display: inline-block; font-size: 11px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; background-color: rgb(255, 255, 255); white-space: nowrap;"><font style="font-size: 15px;">Import</font></div></div></div></foreignObject><image x="623.5" y="302.5" width="43" height="20.75" xlink:href=""/></switch></g><rect x="475" y="280" width="120" height="60" rx="9" ry="9" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 310px; margin-left: 476px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><a href="https://git.astron.nl/astron-sdc/design-system/-/packages" target="_blank"><font style="font-size: 15px;">NPM registry</font></a></div></div></div></foreignObject><image x="476" y="301.5" width="118" height="21" xlink:href=""/></switch></g><path d="M 310 270 L 400 270 Q 410 270 410 260 L 410 240 Q 410 230 420 230 L 468.63 230" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 473.88 230 L 466.88 233.5 L 468.63 230 L 466.88 226.5 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><path d="M 310 270 L 400 270 Q 410 270 410 280 L 410 300 Q 410 310 420 310 L 468.63 310" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 473.88 310 L 466.88 313.5 L 468.63 310 L 466.88 306.5 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 271px; margin-left: 358px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); background-color: rgb(255, 255, 255); "><div style="display: inline-block; font-size: 11px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; background-color: rgb(255, 255, 255); white-space: nowrap;"><font style="font-size: 15px;">CI/CD</font></div></div></div></foreignObject><image x="337.5" y="262.5" width="41" height="20.75" xlink:href=""/></switch></g><path d="M 790 240 Q 790 220 790 225 Q 790 230 790 216.37" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 790 211.12 L 793.5 218.12 L 790 216.37 L 786.5 218.12 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><rect x="695" y="240" width="190" height="140" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><path d="M 780 150 Q 780 95 126.37 95" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 121.12 95 L 128.12 91.5 L 126.37 95 L 128.12 98.5 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><path d="M 770 210 C 770 186 770 174 790 174 C 776.67 174 776.67 150 790 150 C 803.33 150 803.33 174 790 174 C 810 174 810 186 810 210 Z" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 38px; height: 1px; padding-top: 180px; margin-left: 771px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><div><br /></div><div><br /></div><div>User</div></div></div></div></foreignObject><image x="771" y="159" width="38" height="46" xlink:href=""/></switch></g><path d="M 100 200 L 100 260 Q 100 270 110 270 L 183.63 270" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 188.88 270 L 181.88 273.5 L 183.63 270 L 181.88 266.5 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 271px; margin-left: 141px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); background-color: rgb(255, 255, 255); "><div style="display: inline-block; font-size: 11px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; background-color: rgb(255, 255, 255); white-space: nowrap;"><font style="font-size: 15px;">Code</font></div></div></div></foreignObject><image x="123" y="262.5" width="36" height="20.75" xlink:href=""/></switch></g><rect x="718" y="268" width="120" height="60" rx="9" ry="9" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><rect x="728" y="278" width="120" height="60" rx="9" ry="9" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><rect x="738" y="288" width="120" height="60" rx="9" ry="9" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 118px; height: 1px; padding-top: 318px; margin-left: 739px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;">Astron apps</div></div></div></foreignObject><image x="739" y="311.5" width="118" height="17" xlink:href=""/></switch></g><path d="M 100 110 Q 100 130 100 125 Q 100 120 100 133.63" fill="none" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="stroke"/><path d="M 100 138.88 L 96.5 131.88 L 100 133.63 L 103.5 131.88 Z" fill="rgb(0, 0, 0)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><path d="M 80 110 C 80 86 80 74 100 74 C 86.67 74 86.67 50 100 50 C 113.33 50 113.33 74 100 74 C 120 74 120 86 120 110 Z" fill="rgb(255, 255, 255)" stroke="rgb(0, 0, 0)" stroke-miterlimit="10" pointer-events="all"/><g transform="translate(-0.5 -0.5)"><switch><foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility"><div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 38px; height: 1px; padding-top: 80px; margin-left: 81px;"><div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: rgb(0, 0, 0); "><div style="display: inline-block; font-size: 12px; font-family: Helvetica; color: rgb(0, 0, 0); line-height: 1.2; pointer-events: all; white-space: normal; overflow-wrap: normal;"><div><br /></div><div><br /></div><div>De-signer</div></div></div></div></foreignObject><image x="81" y="51.5" width="38" height="61" xlink:href=""/></switch></g></g></svg> \ No newline at end of file diff --git a/src/assets/workflow.png b/src/assets/workflow.png deleted file mode 100644 index 5d343de7b53b137c46eefeb7c3deca1c25c2d88a..0000000000000000000000000000000000000000 Binary files a/src/assets/workflow.png and /dev/null differ diff --git a/src/components/Alert.tsx b/src/components/Alert.tsx index c4c995fc7fd037ac61f6274643173fc525833316..b5a3bc1fe2026a2c39cec6b8aeccb99096852045 100644 --- a/src/components/Alert.tsx +++ b/src/components/Alert.tsx @@ -1,11 +1,13 @@ +import { clsx } from "clsx"; import Icon, { IconName } from "./Icon.tsx"; import Typography from "./Typography.tsx"; -import { baseColor, borderColor, textColor } from "./utils/colors.ts"; +import { baseColorType } from "./utils/colors.ts"; +import { borderColor, textColor } from "./utils/classes.ts"; type AlertProps = { title: string; message: string; - alertType?: baseColor; + alertType?: baseColorType; }; /** @@ -23,12 +25,10 @@ const Alert = ({ title, message, alertType = "primary" }: AlertProps) => { positive: "checkmark-circle", negative: "cross-circle", }; + const baseClass = + "flex flex-row items-start py-[13px] px-[15px] rounded-[8px] w-min border-l-8 border-y-0 border-r-0 bg-baseWhite dark:bg-mediumGrey shadow-2xl"; return ( - <div - className={`${borderColor( - alertType - )} flex flex-row items-start py-[13px] px-[15px] rounded-[8px] w-min border-l-8 border-y-0 border-r-0 bg-baseWhite dark:bg-mediumGrey shadow-2xl`} - > + <div className={clsx(baseClass, borderColor(alertType))}> <Icon name={icon[alertType] as IconName} color={alertType} /> <div className="flex flex-col w-[400px] ml-[13px]"> <Typography diff --git a/src/components/Badge.tsx b/src/components/Badge.tsx index 8523d646d3d6426d92195d963c7e516794f34060..0d9240b8822ae471c660fcd41b137a856197e35b 100644 --- a/src/components/Badge.tsx +++ b/src/components/Badge.tsx @@ -1,10 +1,12 @@ import { Chip } from "@nextui-org/react"; +import { clsx } from "clsx"; import Typography from "./Typography.tsx"; -import { bgColor, borderColor, Color, textColor } from "./utils/colors.ts"; +import { ColorType } from "./utils/colors.ts"; +import { bgColor, borderColor, textColor } from "./utils/classes.ts"; type BadgeProps = { text: string; - color?: Color; + color?: ColorType; inverted?: boolean; }; @@ -16,16 +18,20 @@ type BadgeProps = { * @param inverted: boolean */ const Badge = ({ text, color = "primary", inverted = false }: BadgeProps) => { - const bgClass = inverted ? "bg-transparent" : bgColor(color); - const textClass = inverted ? textColor(color) : "text-baseWhite"; + const bgClass = (inverted: boolean) => + inverted ? "bg-transparent" : bgColor(color); + const textClass = (inverted: boolean) => + inverted ? textColor(color) : "text-baseWhite"; return ( <Chip variant={inverted ? "bordered" : "solid"} classNames={{ - base: `${borderColor( - color - )} ${bgClass} rounded-[4px] h-[20px] px-[10px]`, - content: `${textClass} px-0`, + base: clsx( + borderColor(color), + bgClass(inverted), + "rounded-[4px] h-[20px] px-[10px]" + ), + content: clsx(textClass(inverted), "px-0"), }} > <Typography diff --git a/src/components/Button.tsx b/src/components/Button.tsx index 4c400237e332c2a4e7c88eb8506e549775e03e8a..df51ccb5c16c8a0450d5f07db83e2d16bf25d07e 100644 --- a/src/components/Button.tsx +++ b/src/components/Button.tsx @@ -1,12 +1,13 @@ import { Button as NextButton } from "@nextui-org/react"; +import { clsx } from "clsx"; import Icon, { IconName } from "./Icon.tsx"; import Typography from "./Typography.tsx"; -import { bgColor, Color } from "./utils/colors.ts"; -import { disabledCursor } from "./utils/classes.ts"; +import { ColorType } from "./utils/colors.ts"; +import { bgColor, cursorPointer } from "./utils/classes.ts"; export type ButtonProps = { label?: string | null; - color?: Color; + color?: ColorType; icon?: IconName | null; iconClass?: string; isDisabled?: boolean; @@ -33,25 +34,23 @@ const Button = ({ }: ButtonProps) => { if (!label && !icon) throw Error("Button cannot be empty"); - const px = () => { - if (icon && label) return "px-[8px] "; - else if (icon) return "px-[9px] "; - else if (label) return "px-[10px] "; - }; - const bgClass = (isDisabled: boolean) => { - if (!isDisabled) return bgColor(color) + " "; - return "bg-lightGrey "; + const pxClass = (icon: boolean, label: boolean) => { + if (icon && label) return "px-[8px]"; + else if (icon) return "px-[9px]"; + else if (label) return "px-[10px]"; }; + const bgClass = (isDisabled: boolean) => + isDisabled ? "bg-lightGrey !opacity-100" : bgColor(color); return ( <NextButton startContent={icon && <Icon name={icon} customClass={iconClass} />} - className={ - px() + - bgClass(isDisabled) + - disabledCursor(isDisabled) + - " h-[32px] min-w-fit max-w-fit py-[5px] text-baseWhite rounded-[4px] flex focus:!outline-0" - } + className={clsx( + pxClass(!!icon, !!label), + bgClass(isDisabled), + cursorPointer(isDisabled), + "h-[32px] min-w-fit max-w-fit py-[5px] text-baseWhite rounded-[4px] flex focus:!outline-0" + )} isDisabled={isDisabled} type={type} onClick={onClick} diff --git a/src/components/Checkbox.tsx b/src/components/Checkbox.tsx index 4128933494bb8968736f6738c0eb397e5ba515b8..a743ecfbf5debb3a310ed8b56424811afd4d3d37 100644 --- a/src/components/Checkbox.tsx +++ b/src/components/Checkbox.tsx @@ -1,7 +1,8 @@ import { Checkbox as NextCheckbox } from "@nextui-org/react"; import { Dispatch, SetStateAction } from "react"; +import { clsx } from "clsx"; import Icon from "./Icon.tsx"; -import { disabledBg, disabledCursor } from "./utils/classes.ts"; +import { formFieldBg, cursorPointer } from "./utils/classes.ts"; type CheckboxProps = { isDisabled?: boolean; @@ -30,11 +31,12 @@ const Checkbox = ({ color="default" disableAnimation classNames={{ - base: disabledCursor(isDisabled), - wrapper: - "w-[24px] h-[24px] rounded-[4px] " + - "border-[1px] border-grey dark:border-none " + - disabledBg(isDisabled), + base: cursorPointer(isDisabled), + wrapper: clsx( + "w-[24px] h-[24px] rounded-[4px]", + "border-[1px] border-grey dark:border-none", + formFieldBg(isDisabled) + ), }} > {label} diff --git a/src/components/DateInput.tsx b/src/components/DateInput.tsx index 67764dc660f9c092e212d0319b0b6a77c7a93e0b..0e7ee1efbbd10ac8ac398719cbd2d7a20aa52abb 100644 --- a/src/components/DateInput.tsx +++ b/src/components/DateInput.tsx @@ -5,7 +5,8 @@ import { format } from "date-fns"; import { BaseInput } from "./utils/baseComponents.tsx"; import Icon from "./Icon.tsx"; import Typography from "./Typography.tsx"; -import { textColor } from "./utils/colors.ts"; + +import { textColor } from "./utils/classes.ts"; type DateInputProps = { isRequired?: boolean; diff --git a/src/components/Dropdown.tsx b/src/components/Dropdown.tsx index b2ed37b6b3a1ba16385112d4083e4a3e15913011..0179b428da025dda7de5e484a237fd7ac8d025ea 100644 --- a/src/components/Dropdown.tsx +++ b/src/components/Dropdown.tsx @@ -1,5 +1,6 @@ import { Autocomplete, AutocompleteItem } from "@nextui-org/react"; import { Dispatch, SetStateAction } from "react"; +import { clsx } from "clsx"; import { baseInputClassNames } from "./utils/baseComponents.tsx"; import Icon from "./Icon.tsx"; @@ -18,6 +19,10 @@ const Dropdown = ({ value, onValueChange, }: DropdownProps) => { + const focusClass = + "data-[focus=true]:bg-lightGrey/50 dark:data-[focus=true]:bg-grey/50"; + const hoverCLass = + "data-[selected=true]:!bg-primary hover:!bg-lightGrey dark:hover:!bg-grey"; return ( <Autocomplete isRequired={isRequired} @@ -42,16 +47,11 @@ const Dropdown = ({ }} > {valueOptions.map((option) => { - const focusClass = - "data-[focus=true]:bg-lightGrey/50 dark:data-[focus=true]:bg-grey/50"; - const hoverCLass = - "data-[selected=true]:!bg-primary hover:!bg-lightGrey dark:hover:!bg-grey"; return ( <AutocompleteItem key={option.value} value={option.value} - // todo a concat classname util because I always forget the space in between - className={focusClass + " " + hoverCLass} + className={clsx(focusClass, hoverCLass)} > {option.label} </AutocompleteItem> diff --git a/src/components/Icon.tsx b/src/components/Icon.tsx index 05d316f7b61b7993eaf2dcfc76c25ec8648809cb..4b7953f1227f8398b1187813a7266da11dc656d9 100644 --- a/src/components/Icon.tsx +++ b/src/components/Icon.tsx @@ -1,6 +1,8 @@ import { ReactSVG } from "react-svg"; +import { clsx } from "clsx"; import Icons from "src/assets/icons"; -import { TextColor, textColor } from "./utils/colors.ts"; +import { TextColorType } from "./utils/colors.ts"; +import { textColor } from "./utils/classes.ts"; export const iconNameOptions = [ "attention-circle", @@ -46,7 +48,7 @@ const iconMap = { type IconProps = { name: IconName; - color?: TextColor | null; + color?: TextColorType | null; customClass?: string; }; @@ -63,9 +65,14 @@ const Icon = ({ name, color = null, customClass = "" }: IconProps) => { if (!color && !customClass) { return <ReactSVG src={src} style={{ color: "currentColor" }} />; } - const colorClass = color ? textColor(color) : ""; return ( - <ReactSVG src={src} className={`${colorClass} ${customClass}`.trim()} /> + <ReactSVG + src={src} + className={clsx( + { [textColor(color as TextColorType)]: !!color }, + customClass + )} + /> ); }; diff --git a/src/components/Modal.tsx b/src/components/Modal.tsx index 960cefbfc672fc762f14daf2d279fc4830f48e94..f2b47079d36558d2483651086c6cb09e37e129f7 100644 --- a/src/components/Modal.tsx +++ b/src/components/Modal.tsx @@ -50,7 +50,7 @@ const Modal = ({ body: "p-0 pb-[10px] pr-[20px]", closeButton: "text-foreground-heading mt-[17px] mr-[17px] p-0", footer: "flex flex-row justify-start gap-x-5 p-0 pt-[10px]", - header: "flex flex-col gap-1 px-0 py-[10px] ", + header: "flex flex-col gap-1 px-0 py-[10px]", }} > <ModalContent> diff --git a/src/components/Radio.tsx b/src/components/Radio.tsx index ad9faedf63d9e2845bd4f86455188ded274b3413..25ad779d6a6bb37988cb352b5f3a43d64d7ecb1a 100644 --- a/src/components/Radio.tsx +++ b/src/components/Radio.tsx @@ -1,6 +1,7 @@ import { RadioGroup, Radio as NextRadio } from "@nextui-org/react"; import { Dispatch, SetStateAction } from "react"; -import { disabledBg, disabledCursor } from "./utils/classes.ts"; +import { clsx } from "clsx"; +import { formFieldBg, cursorPointer } from "./utils/classes.ts"; type RadioProps = { isRequired?: boolean; @@ -31,10 +32,11 @@ const Radio = ({ value={option.value} disableAnimation classNames={{ - base: disabledCursor(option.isDisabled || false), - wrapper: - "w-[16px] h-[16px] border-[1px] !border-grey dark:border-none " + - disabledBg(option.isDisabled || false), + base: cursorPointer(option.isDisabled || false), + wrapper: clsx( + "w-[16px] h-[16px] border-[1px] !border-grey dark:border-none", + formFieldBg(option.isDisabled || false) + ), control: "w-[10px] h-[10px]", }} > diff --git a/src/components/Toggle.tsx b/src/components/Toggle.tsx index ab7d3c0d49f14e68eb47cff2eacd74c62a239638..577361e5c0c244b0a5d70d1b4ad4313e87ba2bd5 100644 --- a/src/components/Toggle.tsx +++ b/src/components/Toggle.tsx @@ -1,6 +1,7 @@ import { Dispatch, SetStateAction } from "react"; import { Switch } from "@nextui-org/react"; -import { disabledCursor } from "./utils/classes.ts"; +import { clsx } from "clsx"; +import { cursorPointer } from "./utils/classes.ts"; type ToggleProps = { isDisabled?: boolean; @@ -15,9 +16,8 @@ const Toggle = ({ setIsSelected, label, }: ToggleProps) => { - const selectedClass = isSelected - ? "group-data-[selected=true]:ml-[11px] bg-primary" - : "bg-grey"; + const selectedClass = (isSelected: boolean) => + isSelected ? "group-data-[selected=true]:ml-[11px] bg-primary" : "bg-grey"; return ( <Switch isDisabled={isDisabled} @@ -26,8 +26,8 @@ const Toggle = ({ onValueChange={setIsSelected} color="secondary" classNames={{ - base: disabledCursor(isDisabled), - thumb: `h-[14px] w-[14px] ${selectedClass}`, + base: cursorPointer(isDisabled), + thumb: clsx("h-[14px] w-[14px]", selectedClass(isSelected)), wrapper: "h-[8px] w-[25px] p-0 overflow-visible bg-lightGrey dark:bg-mediumGrey", }} diff --git a/src/components/Typography.tsx b/src/components/Typography.tsx index 86dd2cb0ec88c0c38d4cbd6fdac1822274804fd5..40a911617698c47d193f98afd26b42f31ceda119 100644 --- a/src/components/Typography.tsx +++ b/src/components/Typography.tsx @@ -1,4 +1,5 @@ -import { textColor } from "./utils/colors.ts"; +import { clsx } from "clsx"; +import { textColor } from "./utils/classes.ts"; type TypographyProps = { text: string; @@ -55,10 +56,16 @@ const Typography = ({ paragraph: textColor("body"), note: textColor("heading"), }; - const style = `${font[variant]} ${customClass} ${ - !customColor && color[variant] - }`; - return <span className={style}>{text}</span>; + + return ( + <span + className={clsx(font[variant], customClass, { + [color[variant]]: !customColor, + })} + > + {text} + </span> + ); }; export default Typography; diff --git a/src/components/utils/baseComponents.tsx b/src/components/utils/baseComponents.tsx index 0baa9f1235109b57ad573fc07b57afae61b5a21b..7530a1455185768f782c325ac2d4ee78d94bfb83 100644 --- a/src/components/utils/baseComponents.tsx +++ b/src/components/utils/baseComponents.tsx @@ -1,7 +1,8 @@ import { Input } from "@nextui-org/react"; import { Dispatch, ReactNode, SetStateAction } from "react"; import { InputSlots } from "@nextui-org/theme"; -import { disabledBg, disabledCursor } from "./classes.ts"; +import { clsx } from "clsx"; +import { formFieldBg, cursorPointer } from "./classes.ts"; type BaseInputProps = { // basic info @@ -21,34 +22,33 @@ type BaseInputProps = { type baseInputClassNamesObj = { [key in InputSlots]: string }; -export const baseInputClassNames = (isDisabled: boolean) => { - // No we cannot do "[&>svg]:" + className because Tailwind does not support dynamic class names - const clearButtonSvgClass = - "[&>svg]:w-[42px] [&>svg]:h-[42px] [&>svg]:mt-[-6px] [&>svg]:ml-[-5px]"; +export const baseInputClassNames = ( + isDisabled: boolean +): baseInputClassNamesObj => { + // this is the border color (use text-*) + const mainWrapperClass = (isDisabled: boolean) => + isDisabled + ? "text-mediumGrey dark:text-darkModeBlue" + : "text-grey dark:text-mediumGrey hover:text-baseBlack focus-within:!text-primary"; - const baseClass = { - base: disabledCursor(isDisabled), - label: "!text-foreground-body", - input: "!text-foreground-body bg-default", - inputWrapper: "px-2 rounded-[4px] " + disabledBg(isDisabled), - }; + const clearButtonClass = (isDisabled: boolean) => { + // we cannot do "[&>svg]:" + className because Tailwind does not support dynamic class names + const clearButtonSvgClass: string = + "[&>svg]:w-[42px] [&>svg]:h-[42px] [&>svg]:mt-[-6px] [&>svg]:ml-[-5px]"; + const clearButtonBaseClass: string = + "bg-mediumGrey text-baseWhite dark:bg-baseWhite dark:text-mediumGrey w-[30px] h-[30px] p-0 !opacity-100"; - const defaultClass = { - ...baseClass, - mainWrapper: - "text-grey dark:text-mediumGrey hover:text-baseBlack focus-within:!text-primary", // this is the border color (use text-*) - clearButton: - clearButtonSvgClass + - " " + - "bg-mediumGrey text-baseWhite dark:bg-baseWhite dark:text-mediumGrey w-[30px] h-[30px] p-0 !opacity-100", - } as baseInputClassNamesObj; + return isDisabled ? "" : clsx(clearButtonSvgClass, clearButtonBaseClass); + }; - const disabledClass = { - ...baseClass, - mainWrapper: "text-mediumGrey dark:text-darkModeBlue", + return { + base: cursorPointer(isDisabled), + label: "!text-foreground-body", + input: "!text-foreground-body bg-default", + inputWrapper: clsx("px-2 rounded-[4px]", formFieldBg(isDisabled)), + mainWrapper: mainWrapperClass(isDisabled), + clearButton: clearButtonClass(isDisabled), } as baseInputClassNamesObj; - - return isDisabled ? disabledClass : defaultClass; }; export const BaseInput = ({ diff --git a/src/components/utils/classes.ts b/src/components/utils/classes.ts index 7ab8b54562f54ee7adce8ab53fcde921c3aa0db7..91f97f3d9a189734d05fd032d91b6411110e56eb 100644 --- a/src/components/utils/classes.ts +++ b/src/components/utils/classes.ts @@ -1,9 +1,66 @@ -export const disabledCursor = (isDisabled: boolean) => { - if (!isDisabled) return ""; - return "pointer-events-auto cursor-not-allowed opacity-50"; -}; +import { ColorType, TextColorType } from "./colors.ts"; + +export const cursorPointer = (isDisabled: boolean) => + isDisabled ? "pointer-events-auto cursor-not-allowed" : ""; + +export const formFieldBg = (isDisabled: boolean) => + isDisabled + ? "bg-lightGrey dark:bg-darkModeBlue" + : "bg-baseWhite dark:bg-mediumGrey"; -export const disabledBg = (isDisabled: boolean) => { - if (!isDisabled) return "bg-baseWhite dark:bg-mediumGrey"; - return "bg-lightGrey dark:bg-darkModeBlue"; +export const bgColor = (color: ColorType) => { + switch (color) { + case "primary": + return "bg-primary"; + case "secondary": + return "bg-secondary"; + case "positive": + return "bg-success"; + case "negative": + return "bg-danger"; + case "warning": + return "bg-warning"; + case "neutral": + return "bg-neutral"; + } +}; +export const borderColor = (color: ColorType) => { + switch (color) { + case "primary": + return "border-primary"; + case "secondary": + return "border-secondary"; + case "positive": + return "border-success"; + case "negative": + return "border-danger"; + case "warning": + return "border-warning"; + case "neutral": + return "border-neutral"; + } +}; +export const textColor = (color: TextColorType) => { + switch (color) { + case "heading": + return "text-foreground-heading"; + case "heading2": + return "text-foreground-heading2"; + case "body": + return "text-foreground-body"; + case "transparent": + return "text-default"; + case "primary": + return "text-primary"; + case "secondary": + return "text-secondary"; + case "positive": + return "text-success"; + case "negative": + return "text-danger"; + case "warning": + return "text-warning"; + case "neutral": + return "text-neutral"; + } }; diff --git a/src/components/utils/colors.ts b/src/components/utils/colors.ts index 7bb0e1408efd87f8288250c91f2e7aa1c9834371..6c485182dd77869ec85ee88053e46f5621017679 100644 --- a/src/components/utils/colors.ts +++ b/src/components/utils/colors.ts @@ -5,7 +5,7 @@ export const baseColorOptions = [ "warning", ] as const; -export type baseColor = (typeof baseColorOptions)[number]; +export type baseColorType = (typeof baseColorOptions)[number]; export const colorOptions = [ ...baseColorOptions, @@ -13,7 +13,7 @@ export const colorOptions = [ "neutral", ] as const; -export type Color = (typeof colorOptions)[number]; +export type ColorType = (typeof colorOptions)[number]; export const textColorOptions = [ ...colorOptions, @@ -23,63 +23,4 @@ export const textColorOptions = [ "transparent", ] as const; -export type TextColor = (typeof textColorOptions)[number]; - -export const bgColor = (color: Color) => { - switch (color) { - case "primary": - return "bg-primary"; - case "secondary": - return "bg-secondary"; - case "positive": - return "bg-success"; - case "negative": - return "bg-danger"; - case "warning": - return "bg-warning"; - case "neutral": - return "bg-neutral"; - } -}; - -export const borderColor = (color: Color) => { - switch (color) { - case "primary": - return "border-primary"; - case "secondary": - return "border-secondary"; - case "positive": - return "border-success"; - case "negative": - return "border-danger"; - case "warning": - return "border-warning"; - case "neutral": - return "border-neutral"; - } -}; - -export const textColor = (color: TextColor) => { - switch (color) { - case "heading": - return "text-foreground-heading"; - case "heading2": - return "text-foreground-heading2"; - case "body": - return "text-foreground-body"; - case "transparent": - return "text-default"; - case "primary": - return "text-primary"; - case "secondary": - return "text-secondary"; - case "positive": - return "text-success"; - case "negative": - return "text-danger"; - case "warning": - return "text-warning"; - case "neutral": - return "text-neutral"; - } -}; +export type TextColorType = (typeof textColorOptions)[number];