Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

update checkbox style #1706

Open
wants to merge 14 commits into
base: master
Choose a base branch
from
5 changes: 5 additions & 0 deletions .changeset/fair-cherries-breathe.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"ingred-ui": patch
---

update svg icons
5 changes: 5 additions & 0 deletions .changeset/smooth-bees-tickle.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"ingred-ui": minor
---

update checkbox style
70 changes: 56 additions & 14 deletions src/components/Checkbox/Checkbox.stories.tsx
Original file line number Diff line number Diff line change
@@ -1,12 +1,18 @@
import React from "react";
import { StoryObj } from "@storybook/react";
import Checkbox, { CheckBoxProps } from "./Checkbox";
import Checkbox, { CheckBoxProps, CheckboxSize } from "./Checkbox";
import { Flex } from "..";

export default {
title: "Components/Inputs/Checkbox",
component: Checkbox,
argTypes: {
size: {
type: "select",
options: ["small", "medium"],
},
checked: { control: "boolean" },
disabled: { control: "boolean" },
onChange: { action: "changed" },
},
};
Expand All @@ -17,18 +23,54 @@ export const Example: StoryObj<CheckBoxProps> = {

export const DesignSamples: StoryObj = {
render: () => (
<Flex display="flex" flexDirection="column">
<Checkbox checked={false}>Not checked</Checkbox>
<Checkbox checked={true}>Checked</Checkbox>
<Checkbox indeterminate={true} checked={true}>
Checked(indeterminate)
</Checkbox>
<Checkbox disabled checked={false}>
Not checked(disabled)
</Checkbox>
<Checkbox disabled checked={true}>
Checked(disabled)
</Checkbox>
</Flex>
<>
<Flex display="flex" gap={3}>
<Flex display="flex" flexDirection="column" gap={1}>
<Checkbox checked={false} size={CheckboxSize.SMALL}>
Not checked
</Checkbox>
<Checkbox checked={true} size={CheckboxSize.SMALL}>
Checked
</Checkbox>
<Checkbox
indeterminate={true}
checked={true}
size={CheckboxSize.SMALL}
>
Checked(indeterminate)
</Checkbox>
<Checkbox disabled checked={false} size={CheckboxSize.SMALL}>
Not checked(disabled)
</Checkbox>
<Checkbox disabled checked={true} size={CheckboxSize.SMALL}>
Checked(disabled)
</Checkbox>
<Checkbox
disabled
checked={true}
indeterminate={true}
size={CheckboxSize.SMALL}
>
Checked(indeterminate, disabled)
</Checkbox>
</Flex>
<Flex display="flex" flexDirection="column" gap={1}>
<Checkbox checked={false}>Not checked</Checkbox>
<Checkbox checked={true}>Checked</Checkbox>
<Checkbox indeterminate={true} checked={true}>
Checked(indeterminate)
</Checkbox>
<Checkbox disabled checked={false}>
Not checked(disabled)
</Checkbox>
<Checkbox disabled checked={true}>
Checked(disabled)
</Checkbox>
<Checkbox disabled indeterminate={true} checked={true}>
Checked(indeterminate, disabled)
</Checkbox>
</Flex>
</Flex>
</>
),
};
17 changes: 16 additions & 1 deletion src/components/Checkbox/Checkbox.tsx
Original file line number Diff line number Diff line change
@@ -1,10 +1,21 @@
import * as React from "react";
import * as Styled from "./styled";

export type CheckBoxProps = React.ComponentPropsWithoutRef<"input"> & {
export enum CheckboxSize {
SMALL = "small",
MEDIUM = "medium",
}

// sizeは大きさを示すプロパティとして使いたいため
// size属性を除いたものを定義
type CheckBoxPropsBase = Omit<React.ComponentPropsWithoutRef<"input">, "size">;

export type CheckBoxProps = CheckBoxPropsBase & {
indeterminate?: boolean;
error?: boolean;
inputRef?: React.Ref<HTMLInputElement>;
disabled?: boolean;
size?: CheckboxSize;
};

const Checkbox = React.forwardRef<HTMLLabelElement, CheckBoxProps>(
Expand All @@ -15,6 +26,7 @@ const Checkbox = React.forwardRef<HTMLLabelElement, CheckBoxProps>(
error = false,
disabled = false,
inputRef,
size = CheckboxSize.MEDIUM,
...rest
},
ref,
Expand All @@ -28,12 +40,15 @@ const Checkbox = React.forwardRef<HTMLLabelElement, CheckBoxProps>(
type="checkbox"
indeterminate={indeterminate}
disabled={disabled}
_size={size}
{...rest}
/>
<Styled.Span
size={size}
error={error}
indeterminate={indeterminate}
hasChild={!!children}
disabled={disabled}
>
{children}
</Styled.Span>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,15 +3,15 @@
exports[`Checkbox component testing Checkbox 1`] = `
<DocumentFragment>
<label
class="sc-gsnTZi UNiNj"
class="sc-dkzDqf bwjwyU"
>
<input
class="sc-bczRLJ dyzxTr"
class="sc-gsnTZi bTjvFv"
readonly=""
type="checkbox"
/>
<span
class="sc-dkzDqf iyVHFI"
class="sc-hKMtZM hGuzJQ"
/>
</label>
</DocumentFragment>
Expand Down
88 changes: 56 additions & 32 deletions src/components/Checkbox/styled.ts
Original file line number Diff line number Diff line change
@@ -1,26 +1,63 @@
import styled from "styled-components";
import { getShadow } from "../../utils/getShadow";
import { fontSize } from "../Typography/Typography";
import { CheckboxSize } from "./Checkbox";

export const Checkbox = styled.input<{
indeterminate: boolean;
error: boolean;
disabled: boolean;
_size: CheckboxSize;
}>`
opacity: 0;
-webkit-appearance: none;
appearance: none;
position: absolute;

// inputの状態に合わせたチェックアイコンのスタイルを
// ここに書いているのは、スタイルの継承の都合が悪かったため
&:checked + span::before {
/* flex: 1 0 auto; */
/* background */
background-color: ${({ theme }) => theme.palette.primary.main};
box-shadow: ${({ theme }) => theme.shadow["3dShadowPrimary"]};
border-color: ${({ theme }) => theme.palette.primary.dark};

/* check icon */
background-image: ${({ indeterminate }) =>
background-color: ${({ theme }) => theme.palette.primary.main};
background-image: ${({ indeterminate, _size }) =>
// eslint-disable-next-line no-nested-ternary
indeterminate
? `url('data:image/svg+xml;charset=utf8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2210%22%20height%3D%223%22%20viewBox%3D%220%200%2010%203%22%3E%3Cdefs%3E%3Cstyle%3E.a%7Bfill%3A%23fff%3B%7D%3C%2Fstyle%3E%3C%2Fdefs%3E%3Crect%20class%3D%22a%22%20width%3D%2210%22%20height%3D%223%22%20rx%3D%220.596%22%2F%3E%3C%2Fsvg%3E')`
: `url('data:image/svg+xml;charset=utf8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2212.001%22%20height%3D%228.997%22%20viewBox%3D%220%200%2012.001%208.997%22%3E%3Cdefs%3E%3Cstyle%3E.a%7Bfill%3A%23fff%3B%7D%3C%2Fstyle%3E%3C%2Fdefs%3E%3Cpath%20class%3D%22a%22%20d%3D%22M405.681%2C343.44l-5.154%2C5.12-1.332%2C1.323a.622.622%2C0%2C0%2C1-.877%2C0l-1.333-1.323h0l-2.943-2.924a.612.612%2C0%2C0%2C1%2C0-.871l1.332-1.324a.624.624%2C0%2C0%2C1%2C.877%2C0l2.506%2C2.49%2C4.714-4.685a.624.624%2C0%2C0%2C1%2C.877%2C0l1.333%2C1.323A.614.614%2C0%2C0%2C1%2C405.681%2C343.44Z%22%20transform%3D%22translate(-393.861%20-341.066)%22%2F%3E%3C%2Fsvg%3E')`};
? // マイナスアイコン
_size === CheckboxSize.SMALL
? // チェックアイコン(小)
`url('')`
: // チェックアイコン(大)
`url('')`
: // チェックアイコン
_size === CheckboxSize.SMALL
? // チェックアイコン(小)
`url('')`
: // チェックアイコン(大)
`url('')`};
background-repeat: no-repeat;
background-position: center center;
}
&:disabled + span::before {
border: 1px solid ${({ theme }) => theme.palette.divider};
background-color: ${({ theme }) => theme.palette.gray.light};
}
&:disabled:checked + span::before {
box-shadow: ${({ theme }) => theme.shadow["3dShadowBasic"]};
background-image: ${({ indeterminate, _size }) =>
// eslint-disable-next-line no-nested-ternary
indeterminate
? // マイナスアイコン(disabled)
_size === CheckboxSize.SMALL
? // マイナスアイコン(disabled, 小)
`url('')`
: // マイナスアイコン(disabled, 大)
`url('')`
: // チェックアイコン(disabled)
_size === CheckboxSize.SMALL
? // チェックアイコン(disabled, 小)
`url('')`
: // チェックアイコン(disabled, 大)
`url('')`};
background-repeat: no-repeat;
background-position: center center;
}
Expand All @@ -33,49 +70,36 @@ export const Container = styled.label<{ disabled: boolean }>`
`;

export const Span = styled.span<{
size: CheckboxSize;
hasChild: boolean;
indeterminate: boolean;
error: boolean;
disabled: boolean;
}>`
display: inline-flex;
align-items: center;
color: ${({ error }) =>
error
? ({ theme }) => theme.palette.danger.main
: ({ theme }) => theme.palette.black};
font-size: ${({ size }) =>
size === CheckboxSize.SMALL ? `${fontSize.sm}px` : `${fontSize.md}px`};
&:disabled {
color: ${({ theme }) => theme.palette.text.disabled};
}
&::before {
flex-shrink: 0;
content: "";
width: 18px;
height: 18px;
aspect-ratio: 1;
width: ${({ size }) => (size === CheckboxSize.SMALL ? "16px" : "18px")};
border: 1px solid
${({ error, theme }) =>
error ? theme.palette.danger.main : theme.palette.divider};
border-radius: ${({ theme }) => theme.radius * 0.5}px;
box-shadow: ${({ theme }) =>
getShadow(
3,
theme.palette.action.shadowOpacity,
theme.palette.action.shadowBase,
)};
box-shadow: ${({ theme }) => theme.shadow["3dShadowBasic"]};
background-color: ${({ theme }) => theme.palette.background.default};
margin-right: ${({ hasChild, theme }) =>
hasChild ? `${theme.spacing / 2}px` : "auto"};
transition: background-color 0.3s ease;
}
${Checkbox}:disabled + & {
color: ${({ theme }) => theme.palette.text.disabled};
}
${Checkbox}:disabled + &::before {
border: 1px solid ${({ theme }) => theme.palette.divider};
background-color: ${({ theme }) => theme.palette.gray.light};
}
${Checkbox}:disabled:checked + &::before {
background-image: ${({ indeterminate }) =>
indeterminate
? `url('data:image/svg+xml;charset=utf8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2210%22%20height%3D%223%22%20viewBox%3D%220%200%2010%203%22%3E%3Cdefs%3E%3Cstyle%3E.a%7Bfill%3A%23b3bac1%3B%7D%3C%2Fstyle%3E%3C%2Fdefs%3E%3Crect%20class%3D%22a%22%20width%3D%2210%22%20height%3D%223%22%20rx%3D%220.596%22%2F%3E%3C%2Fsvg%3E')`
: `url('data:image/svg+xml;charset=utf8,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2212.001%22%20height%3D%228.997%22%20viewBox%3D%220%200%2012.001%208.997%22%3E%3Cdefs%3E%3Cstyle%3E.a%7Bfill%3A%23b3bac1%3B%7D%3C%2Fstyle%3E%3C%2Fdefs%3E%3Cpath%20class%3D%22a%22%20d%3D%22M405.681%2C343.44l-5.154%2C5.12-1.332%2C1.323a.622.622%2C0%2C0%2C1-.877%2C0l-1.333-1.323h0l-2.943-2.924a.612.612%2C0%2C0%2C1%2C0-.871l1.332-1.324a.624.624%2C0%2C0%2C1%2C.877%2C0l2.506%2C2.49%2C4.714-4.685a.624.624%2C0%2C0%2C1%2C.877%2C0l1.333%2C1.323A.614.614%2C0%2C0%2C1%2C405.681%2C343.44Z%22%20transform%3D%22translate(-393.861%20-341.066)%22%2F%3E%3C%2Fsvg%3E')`};
background-repeat: no-repeat;
background-position: center center;
}
`;
Loading