diff --git a/packages/orbit-components/src/Checkbox/Checkbox.stories.tsx b/packages/orbit-components/src/Checkbox/Checkbox.stories.tsx index ec51344ab7..bddad0d041 100644 --- a/packages/orbit-components/src/Checkbox/Checkbox.stories.tsx +++ b/packages/orbit-components/src/Checkbox/Checkbox.stories.tsx @@ -16,7 +16,7 @@ const meta: Meta = { parameters: { info: "Additionally you can add tooltip to this component.", controls: { - exclude: ["onChange"], + exclude: ["onChange", "defaultChecked"], }, }, diff --git a/packages/orbit-components/src/Checkbox/README.md b/packages/orbit-components/src/Checkbox/README.md index 0d67b5d13b..a5b33d9928 100644 --- a/packages/orbit-components/src/Checkbox/README.md +++ b/packages/orbit-components/src/Checkbox/README.md @@ -16,27 +16,28 @@ After adding import into your project you can use it simply like: Table below contains all types of the props available in Checkbox component. -| Name | Type | Default | Description | -| :------- | :------------------------- | :------ | :----------------------------------------------------------------------------------------------------------- | -| checked | `boolean` | `false` | If `true`, the Checkbox will be checked. | -| disabled | `boolean` | `false` | If `true`, the Checkbox will be set up as disabled. | -| dataTest | `string` | | Optional prop for testing purposes. | -| id | `string` | | Set `id` for `Checkbox` | -| hasError | `boolean` | `false` | If `true`, the border of the Checkbox will turn red. [See Functional specs](#functional-specs) | -| info | `React.Node` | | The additional info about the Checkbox. | -| label | `string` | | The label of the Checkbox. | -| name | `string` | | The name for the Checkbox. | -| onChange | `event => void \| Promise` | | Function for handling onChange event. | -| ref | `func` | | Prop for forwarded ref of the Checkbox. [See Functional specs](#functional-specs) | -| tabIndex | `string \| number` | | Specifies the tab order of an element | -| tooltip | `Element` | | Optional property when you need to attach Tooltip to the Checkbox. [See Functional specs](#functional-specs) | -| value | `string` | | The value of the Checkbox. | +| Name | Type | Default | Description | +| :------------- | :------------------------- | :------ | :----------------------------------------------------------------------------------------------------------- | +| checked | `boolean` | | If `true`, the Checkbox will be checked. | +| defaultChecked | `boolean` | | If `true`, the Checkbox will be checked by default. Only to be used in uncontrolled. | +| disabled | `boolean` | `false` | If `true`, the Checkbox will be set up as disabled. | +| dataTest | `string` | | Optional prop for testing purposes. | +| id | `string` | | Set `id` for `Checkbox` | +| hasError | `boolean` | `false` | If `true`, the border of the Checkbox will turn red. [See Functional specs](#functional-specs) | +| info | `React.Node` | | The additional info about the Checkbox. | +| label | `string` | | The label of the Checkbox. | +| name | `string` | | The name for the Checkbox. | +| onChange | `event => void \| Promise` | | Function for handling onChange event. | +| ref | `func` | | Prop for forwarded ref of the Checkbox. [See Functional specs](#functional-specs) | +| tabIndex | `string \| number` | | Specifies the tab order of an element | +| tooltip | `Element` | | Optional property when you need to attach Tooltip to the Checkbox. [See Functional specs](#functional-specs) | +| value | `string` | | The value of the Checkbox. | ## Functional specs -- The `hasError` prop will be visible only when the Checkbox has `checked` or `disabled` prop set on **false**. +- The `hasError` prop will be visible only when the Checkbox is not checked nor disabled. -- `ref` can be used for example auto-focus the elements immediately after render. +- `ref` can be used, for example, to control focus or to get the status (checked) of the element. ```jsx import * as React from "react"; diff --git a/packages/orbit-components/src/Checkbox/__tests__/index.test.tsx b/packages/orbit-components/src/Checkbox/__tests__/index.test.tsx index 3c88835878..855d08f393 100644 --- a/packages/orbit-components/src/Checkbox/__tests__/index.test.tsx +++ b/packages/orbit-components/src/Checkbox/__tests__/index.test.tsx @@ -7,7 +7,7 @@ import CheckBox from ".."; describe("CheckBox", () => { const user = userEvent.setup(); - it("default", async () => { + it("can be controlled", async () => { const onChange = jest.fn(); render( { expect(checkbox).toHaveAttribute("value", "option"); expect(checkbox).toHaveAttribute("name", "name"); expect(checkbox).toHaveAttribute("tabIndex", "-1"); + expect(checkbox).not.toHaveAttribute("checked"); await user.click(checkbox); expect(onChange).toHaveBeenCalled(); }); + + it("can be uncontrolled", async () => { + const onChange = jest.fn(); + render( + , + ); + expect(screen.getByTestId("test")).toBeInTheDocument(); + const checkbox = screen.getByRole("checkbox", { name: "Checkbox" }) as HTMLInputElement; + expect(checkbox.checked).toBeTruthy(); + await user.click(checkbox); + expect(onChange).toHaveBeenCalled(); + expect(checkbox.checked).toBeFalsy(); + }); }); diff --git a/packages/orbit-components/src/Checkbox/index.tsx b/packages/orbit-components/src/Checkbox/index.tsx index c7827eca22..22fcffb780 100644 --- a/packages/orbit-components/src/Checkbox/index.tsx +++ b/packages/orbit-components/src/Checkbox/index.tsx @@ -14,7 +14,8 @@ const Checkbox = React.forwardRef((props, ref) => { value, hasError = false, disabled = false, - checked = false, + checked, + defaultChecked, name, onChange, dataTest, @@ -33,19 +34,24 @@ const Checkbox = React.forwardRef((props, ref) => { "flex flex-row", "relative w-full", "[&_.orbit-checkbox-icon-container]:hover:shadow-none", + "[&_.orbit-checkbox-icon-container]:has-[:checked]:bg-blue-normal [&_.orbit-checkbox-icon-container]:has-[:checked]:hover:bg-blue-dark", disabled - ? "cursor-not-allowed" + ? [ + "cursor-not-allowed", + "[&_.orbit-checkbox-icon-container]:bg-form-element-disabled-background", + "[&_.orbit-checkbox-icon-container]:has-[:checked]:bg-cloud-dark", + checked && "[&_.orbit-checkbox-icon-container]:bg-cloud-dark", + ] : [ "cursor-pointer", + "[&_.orbit-checkbox-icon-container]:has-[:checked]:border-blue-normal [&_.orbit-checkbox-icon-container]:has-[:checked]:hover:border-blue-dark", checked && + !hasError && "[&_.orbit-checkbox-icon-container]:bg-blue-normal [&_.orbit-checkbox-icon-container]:border-blue-normal [&_.orbit-checkbox-icon-container]:hover:bg-blue-dark [&_.orbit-checkbox-icon-container]:hover:border-blue-dark", !checked && "[&_.orbit-checkbox-icon-container]:bg-form-element-background", - !checked && - hasError && - "[&_.orbit-checkbox-icon-container]:border-form-element-error", - !checked && - !hasError && - "[&_.orbit-checkbox-icon-container]:border-form-element-border-color [&_.orbit-checkbox-icon-container]:hover:border-blue-light-active [&_.orbit-checkbox-icon-container]:active:border-form-element-focus", + !checked && hasError + ? "[&_.orbit-checkbox-icon-container]:border-form-element-error" + : "[&_.orbit-checkbox-icon-container]:border-form-element-border-color [&_.orbit-checkbox-icon-container]:hover:border-blue-light-active [&_.orbit-checkbox-icon-container]:active:border-form-element-focus", ], )} > @@ -59,7 +65,8 @@ const Checkbox = React.forwardRef((props, ref) => { disabled={disabled} name={name} tabIndex={tabIndex ? Number(tabIndex) : undefined} - checked={checked} + checked={checked || undefined} + defaultChecked={defaultChecked} onChange={!readOnly ? onChange : undefined} onClick={e => readOnly && e.stopPropagation()} ref={ref} @@ -77,15 +84,12 @@ const Checkbox = React.forwardRef((props, ref) => { "rounded-150 de:rounded-100", "duration-fast transition-all ease-in-out", "peer-focus:outline-blue-normal peer-focus:outline peer-focus:outline-2", + "peer-checked:[&>svg]:visible", "[&>svg]:size-icon-small", "[&>svg]:flex [&>svg]:items-center [&>svg]:justify-center", "active:scale-95", checked ? "[&>svg]:visible" : "[&>svg]:invisible", - disabled && [ - "border-cloud-dark", - checked && "bg-cloud-dark", - !checked && "bg-form-element-disabled-background", - ], + disabled && ["border-cloud-dark"], )} > diff --git a/packages/orbit-components/src/Checkbox/types.d.ts b/packages/orbit-components/src/Checkbox/types.d.ts index 93d06d51fe..963a0cf822 100644 --- a/packages/orbit-components/src/Checkbox/types.d.ts +++ b/packages/orbit-components/src/Checkbox/types.d.ts @@ -11,6 +11,7 @@ export interface Props extends Common.Globals { readonly hasError?: boolean; readonly disabled?: boolean; readonly checked?: boolean; + readonly defaultChecked?: boolean; readonly name?: string; readonly info?: React.ReactNode; readonly tabIndex?: string | number;