Checkbox
Allow users to switch between checked, unchecked, and indeterminate states.
<script lang="ts">
import { Checkbox, Label } from "bits-ui";
import Check from "phosphor-svelte/lib/Check";
import Minus from "phosphor-svelte/lib/Minus";
</script>
<div class="flex items-center space-x-3">
<Checkbox.Root
id="terms"
aria-labelledby="terms-label"
class="peer inline-flex size-[25px] items-center justify-center rounded-md border border-muted bg-foreground transition-all duration-150 ease-in-out active:scale-98 data-[state=unchecked]:border-border-input data-[state=unchecked]:bg-background data-[state=unchecked]:hover:border-dark-40"
name="hello"
indeterminate
>
{#snippet children({ checked, indeterminate })}
<div class="inline-flex items-center justify-center text-background">
{#if indeterminate}
<Minus class="size-[15px]" weight="bold" />
{:else if checked}
<Check class="size-[15px]" weight="bold" />
{/if}
</div>
{/snippet}
</Checkbox.Root>
<Label.Root
id="terms-label"
for="terms"
class="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
>
Accept terms and conditions
</Label.Root>
</div>
import typography from "@tailwindcss/typography";
import animate from "tailwindcss-animate";
import { fontFamily } from "tailwindcss/defaultTheme";
/** @type {import('tailwindcss').Config} */
export default {
darkMode: "class",
content: ["./src/**/*.{html,js,svelte,ts}"],
theme: {
container: {
center: true,
screens: {
"2xl": "1440px",
},
},
extend: {
colors: {
border: {
DEFAULT: "hsl(var(--border-card))",
input: "hsl(var(--border-input))",
"input-hover": "hsl(var(--border-input-hover))",
},
background: {
DEFAULT: "hsl(var(--background) / <alpha-value>)",
alt: "hsl(var(--background-alt) / <alpha-value>)",
},
foreground: {
DEFAULT: "hsl(var(--foreground) / <alpha-value>)",
alt: "hsl(var(--foreground-alt) / <alpha-value>)",
},
muted: {
DEFAULT: "hsl(var(--muted) / <alpha-value>)",
foreground: "hsl(var(--muted-foreground))",
},
dark: {
DEFAULT: "hsl(var(--dark) / <alpha-value>)",
4: "hsl(var(--dark-04))",
10: "hsl(var(--dark-10))",
40: "hsl(var(--dark-40))",
},
accent: {
DEFAULT: "hsl(var(--accent) / <alpha-value>)",
foreground: "hsl(var(--accent-foreground) / <alpha-value>)",
},
destructive: {
DEFAULT: "hsl(var(--destructive) / <alpha-value>)",
},
contrast: {
DEFAULT: "hsl(var(--contrast) / <alpha-value>)",
},
},
fontFamily: {
sans: ["Inter", ...fontFamily.sans],
mono: ["Source Code Pro", ...fontFamily.mono],
alt: ["Courier", ...fontFamily.sans],
},
fontSize: {
xxs: "10px",
},
borderWidth: {
6: "6px",
},
borderRadius: {
card: "16px",
"card-lg": "20px",
"card-sm": "10px",
input: "9px",
button: "5px",
"5px": "5px",
"9px": "9px",
"10px": "10px",
"15px": "15px",
},
height: {
input: "3rem",
"input-sm": "2.5rem",
},
boxShadow: {
mini: "var(--shadow-mini)",
"mini-inset": "var(--shadow-mini-inset)",
popover: "var(--shadow-popover)",
kbd: "var(--shadow-kbd)",
btn: "var(--shadow-btn)",
card: "var(--shadow-card)",
"date-field-focus": "var(--shadow-date-field-focus)",
},
opacity: {
8: "0.08",
},
scale: {
80: ".80",
98: ".98",
99: ".99",
},
},
keyframes: {
"accordion-down": {
from: { height: "0" },
to: { height: "var(--bits-accordion-content-height)" },
},
"accordion-up": {
from: { height: "var(--bits-accordion-content-height)" },
to: { height: "0" },
},
"caret-blink": {
"0%,70%,100%": { opacity: "1" },
"20%,50%": { opacity: "0" },
},
},
animation: {
"accordion-down": "accordion-down 0.2s ease-out",
"accordion-up": "accordion-up 0.2s ease-out",
"caret-blink": "caret-blink 1.25s ease-out infinite",
},
},
plugins: [typography, animate],
};
@import url("https://fonts.googleapis.com/css2?family=Source+Code+Pro:ital,wght@0,400;0,500;0,600;0,700;1,400;1,500;1,600;1,700&display=swap");
@import url("https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600&display=swap");
@tailwind base;
@tailwind components;
@tailwind utilities;
@layer base {
:root {
/* Colors */
--background: 0 0% 100%;
--background-alt: 0 0% 100%;
--foreground: 0 0% 9%;
--foreground-alt: 0 0% 32%;
--muted: 240 5% 96%;
--muted-foreground: 0 0% 9% / 0.4;
--border: 240 6% 10%;
--border-input: 240 6% 10% / 0.17;
--border-input-hover: 240 6% 10% / 0.4;
--border-card: 240 6% 10% / 0.1;
--dark: 240 6% 10%;
--dark-10: 240 6% 10% / 0.1;
--dark-40: 240 6% 10% / 0.4;
--dark-04: 240 6% 10% / 0.04;
--accent: 204 94% 94%;
--accent-foreground: 204 80% 16%;
--destructive: 347 77% 50%;
/* black */
--constrast: 0 0% 0%;
/* Shadows */
--shadow-mini: 0px 1px 0px 1px rgba(0, 0, 0, 0.04);
--shadow-mini-inset: 0px 1px 0px 0px rgba(0, 0, 0, 0.04) inset;
--shadow-popover: 0px 7px 12px 3px hsla(var(--dark-10));
--shadow-kbd: 0px 2px 0px 0px rgba(0, 0, 0, 0.07);
--shadow-btn: 0px 1px 0px 1px rgba(0, 0, 0, 0.03);
--shadow-card: 0px 2px 0px 1px rgba(0, 0, 0, 0.04);
--shadow-date-field-focus: 0px 0px 0px 3px rgba(24, 24, 27, 0.17);
}
.dark {
/* Colors */
--background: 0 0% 5%;
--background-alt: 0 0% 8%;
--foreground: 0 0% 95%;
--foreground-alt: 0 0% 70%;
--muted: 240 4% 16%;
--muted-foreground: 0 0% 100% / 0.4;
--border: 0 0% 96%;
--border-input: 0 0% 96% / 0.17;
--border-input-hover: 0 0% 96% / 0.4;
--border-card: 0 0% 96% / 0.1;
--dark: 0 0% 96%;
--dark-40: 0 0% 96% / 0.4;
--dark-10: 0 0% 96% / 0.1;
--dark-04: 0 0% 96% / 0.04;
--accent: 204 90 90%;
--accent-foreground: 204 94% 94%;
--destructive: 350 89% 60%;
/* white */
--constrast: 0 0% 100%;
/* Shadows */
--shadow-mini: 0px 1px 0px 1px rgba(0, 0, 0, 0.3);
--shadow-mini-inset: 0px 1px 0px 0px rgba(0, 0, 0, 0.5) inset;
--shadow-popover: 0px 7px 12px 3px hsla(0deg 0% 0% / 30%);
--shadow-kbd: 0px 2px 0px 0px rgba(255, 255, 255, 0.07);
--shadow-btn: 0px 1px 0px 1px rgba(0, 0, 0, 0.2);
--shadow-card: 0px 2px 0px 1px rgba(0, 0, 0, 0.4);
--shadow-date-field-focus: 0px 0px 0px 3px rgba(244, 244, 245, 0.1);
}
}
@layer base {
* {
@apply border-border;
}
html {
-webkit-text-size-adjust: 100%;
font-variation-settings: normal;
}
body {
@apply bg-background text-foreground;
font-feature-settings:
"rlig" 1,
"calt" 1;
}
/* Mobile tap highlight */
/* https://developer.mozilla.org/en-US/docs/Web/CSS/-webkit-tap-highlight-color */
html {
-webkit-tap-highlight-color: rgba(128, 128, 128, 0.5);
}
::selection {
background: #fdffa4;
color: black;
}
/* === Scrollbars === */
::-webkit-scrollbar {
@apply w-2;
@apply h-2;
}
::-webkit-scrollbar-track {
@apply !bg-transparent;
}
::-webkit-scrollbar-thumb {
@apply rounded-card-lg !bg-dark-10;
}
::-webkit-scrollbar-corner {
background: rgba(0, 0, 0, 0);
}
/* Firefox */
/* https://developer.mozilla.org/en-US/docs/Web/CSS/scrollbar-color#browser_compatibility */
html {
scrollbar-color: var(--bg-muted);
}
.antialised {
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
}
@layer utilities {
.step {
counter-increment: step;
}
.step:before {
@apply absolute inline-flex h-9 w-9 items-center justify-center rounded-full border-4 border-background bg-muted text-center -indent-px font-mono text-base font-medium;
@apply ml-[-50px] mt-[-4px];
content: counter(step);
}
}
@layer components {
*:not(body):not(.focus-override) {
outline: none !important;
&:focus-visible {
@apply focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-foreground focus-visible:ring-offset-2 focus-visible:ring-offset-background;
}
}
.link {
@apply inline-flex items-center gap-1 rounded-sm font-medium underline underline-offset-4 hover:text-foreground/80 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-foreground focus-visible:ring-offset-2 focus-visible:ring-offset-background;
}
input::-webkit-outer-spin-button,
input::-webkit-inner-spin-button {
-webkit-appearance: none;
margin: 0;
}
/* Firefox */
input[type="number"] {
-moz-appearance: textfield;
}
}
Overview
The Checkbox component provides a flexible and accessible way to create checkbox inputs in your Svelte applications. It supports three states: checked, unchecked, and indeterminate, allowing for complex form interactions and data representations.
Key Features
- Tri-State Support: Handles checked, unchecked, and indeterminate states, providing versatility in form design.
- Accessibility: Built with WAI-ARIA guidelines in mind, ensuring keyboard navigation and screen reader support.
- Flexible State Management: Supports both controlled and uncontrolled state, allowing for full control over the checkbox's checked state.
Architecture
The Checkbox component is composed of the following parts:
- Root: The main component that manages the state and behavior of the checkbox.
Structure
Here's an overview of how the Checkbox component is structured in code:
<script lang="ts">
import { Checkbox } from "bits-ui";
</script>
<Checkbox.Root>
{#snippet children({ checked, indeterminate })}
{#if indeterminate}
-
{:else if checked}
✅
{:else}
❌
{/if}
{/snippet}
</Checkbox.Root>
Reusable Components
It's recommended to use the Checkbox
primitive to create your own custom checkbox component that can be used throughout your application. In the example below, we're using the Checkbox
and Label
components to create a custom checkbox component.
<script lang="ts">
import { Checkbox, Label, useId, type WithoutChildrenOrChild } from "bits-ui";
let {
id = useId(),
checked = $bindable(false),
ref = $bindable(null),
labelRef = $bindable(null),
...restProps
}: WithoutChildrenOrChild<Checkbox.RootProps> & {
labelText: string;
labelRef?: HTMLLabelElement | null;
} = $props();
</script>
<Checkbox.Root bind:checked bind:ref {...restProps}>
{#snippet children({ checked, indeterminate })}
{#if indeterminate}
-
{:else if checked}
✅
{:else}
❌
{/if}
{/snippet}
</Checkbox.Root>
<Label.Root for={id} bind:ref={labelRef}>
{labelText}
</Label.Root>
You can then use the MyCheckbox
component in your application like so:
<script lang="ts">
import MyCheckbox from "$lib/components/MyCheckbox.svelte";
</script>
<MyCheckbox labelText="Enable notifications" />
Managing Checked State
Bits UI offers several approaches to manage and synchronize the Checkbox's checked state, catering to different levels of control and integration needs.
1. Two-Way Binding
For seamless state synchronization, use Svelte's bind:checked
directive. This method automatically keeps your local state in sync with the checkbox's internal state.
<script lang="ts">
import MyCheckbox from "$lib/components/MyCheckbox.svelte";
let myChecked = $state(false);
</script>
<button onclick={() => (myChecked = false)}> uncheck </button>
<MyCheckbox bind:checked={myChecked} />
Key Benefits
- Simplifies state management
- Automatically updates
myChecked
when the checkbox changes (e.g., via clicking on the checkbox) - Allows external control (e.g., checking via a separate button/programmatically)
2. Change Handler
For more granular control or to perform additional logic on state changes, use the onCheckedChange
prop. This approach is useful when you need to execute custom logic alongside state updates.
<script lang="ts">
import MyCheckbox from "$lib/components/MyCheckbox.svelte";
let myChecked = $state(false);
</script>
<MyCheckbox
checked={myChecked}
onCheckedChange={(checked) => {
myChecked = checked;
if (checked === "indeterminate") {
// do something different
}
// additional logic here.
}}
/>
Use Cases
- Implementing custom behaviors on checked/unchecked
- Integrating with external state management solutions
- Triggering side effects (e.g., logging, data fetching)
3. Fully Controlled
For complete control over the checkbox's checked state, use the controlledChecked
prop. This approach requires you to manually manage the checked state, giving you full control over when and how the checkbox responds to change events.
To implement controlled state:
- Set the
controlledChecked
prop totrue
on theCheckbox.Root
component. - Provide a
checked
prop toCheckbox.Root
, which should be a variable holding the current state. - Implement an
onCheckedChange
handler to update the state when the internal state changes.
<script lang="ts">
import { Checkbox } from "bits-ui";
let myChecked = $state(false);
</script>
<Checkbox.Root controlledChecked checked={myChecked} onCheckedChange={(c) => (myChecked = c)}>
<!-- ... -->
</Checkbox.Root>
When to Use
- Implementing complex checked/unchecked logic
- Coordinating multiple UI elements
- Debugging state-related issues
Note
While powerful, fully controlled state should be used judiciously as it increases complexity and can cause unexpected behaviors if not handled carefully.
For more in-depth information on controlled components and advanced state management techniques, refer to our Controlled State documentation.
Managing Indeterminate State
Bits UI offers several approaches to manage and synchronize the Checkbox's indeterminate
state, catering to different levels of control and integration needs.
1. Two-Way Binding
For seamless state synchronization, use Svelte's bind:indeterminate
directive. This method automatically keeps your local state in sync with the checkbox's internal state.
<script lang="ts">
import MyCheckbox from "$lib/components/MyCheckbox.svelte";
let myIndeterminate = $state(true);
</script>
<button onclick={() => (myIndeterminate = false)}> clear indeterminate </button>
<MyCheckbox bind:indeterminate={myIndeterminate} />
Key Benefits
- Simplifies state management
- Automatically updates
myIndeterminate
when the checkbox changes (e.g., via clicking on the checkbox) - Allows external control (e.g., checking via a separate button/programmatically)
2. Change Handler
For more granular control or to perform additional logic on state changes, use the onIndeterminateChange
prop. This approach is useful when you need to execute custom logic alongside state updates.
<script lang="ts">
import MyCheckbox from "$lib/components/MyCheckbox.svelte";
let myIndeterminate = $state(true);
</script>
<MyCheckbox
indeterminate={myIndeterminate}
onIndeterminateChange={(indeterminate) => {
myIndeterminate = indeterminate;
// additional logic here.
}}
/>
Use Cases
- Implementing custom behaviors
- Integrating with external state management solutions
- Triggering side effects (e.g., logging, data fetching)
3. Fully Controlled
For complete control over the checkbox's checked state, use the controlledIndeterminate
prop. This approach requires you to manually manage the indeterminate
state, giving you full control over when and how the checkbox responds to change events.
To implement controlled state:
- Set the
controlledIndeterminate
prop totrue
on theCheckbox.Root
component. - Provide a
indeterminate
prop toCheckbox.Root
, which should be a variable holding the current state. - Implement an
onIndeterminateChange
handler to update the state when the internal state changes.
<script lang="ts">
import { Checkbox } from "bits-ui";
let myIndeterminate = $state(true);
</script>
<Checkbox.Root
controlledIndeterminate
indeterminate={myIndeterminate}
onIndeterminateChange={(i) => (myIndeterminate = i)}
>
<!-- ... -->
</Checkbox.Root>
When to Use
- Implementing complex indeterminate logic
- Coordinating multiple UI elements
- Debugging state-related issues
Note
While powerful, fully controlled state should be used judiciously as it increases complexity and can cause unexpected behaviors if not handled carefully.
For more in-depth information on controlled components and advanced state management techniques, refer to our Controlled State documentation.
Disabled State
You can disable the checkbox by setting the disabled
prop to true
.
<MyCheckbox disabled labelText="Enable notifications" />
HTML Forms
If you set the name
prop, a hidden checkbox input will be rendered to submit the value of the checkbox with a form.
By default, the checkbox will be submitted with default checkbox value of 'on'
if the checked
prop is true
.
<MyCheckbox name="notifications" labelText="Enable notifications" />
Custom Input Value
If you'd prefer to submit a different value, you can use the value
prop to set the value of the hidden input.
For example, if you wanted to submit a string value, you could do the following:
<MyCheckbox value="hello" name="notifications" labelText="Enable notifications" />
Required
If you want to make the checkbox required, you can use the required
prop.
<Checkbox.Root required>
<!-- ... -->
</Checkbox.Root>
This will apply the required
attribute to the hidden input element, ensuring that proper form submission is enforced.
Checkbox Groups
You can use the Checkbox.Group
component to create a checkbox group.
<script lang="ts">
import { Checkbox } from "bits-ui";
</script>
<Checkbox.Group name="notifications">
<Checkbox.GroupLabel>Notifications</Checkbox.GroupLabel>
<Checkbox.Root value="marketing" />
<Checkbox.Root value="promotions" />
<Checkbox.Root value="news" />
</Checkbox.Group>
<script lang="ts">
import { Checkbox, Label, useId } from "bits-ui";
import Check from "phosphor-svelte/lib/Check";
import Minus from "phosphor-svelte/lib/Minus";
let myValue = $state<string[]>(["marketing", "news"]);
</script>
<Checkbox.Group
class="flex flex-col gap-3"
bind:value={myValue}
name="notifications"
>
<Checkbox.GroupLabel class="text-sm font-medium text-foreground-alt">
Notifications
</Checkbox.GroupLabel>
<div class="flex flex-col gap-4">
{@render MyCheckbox({ label: "Marketing", value: "marketing" })}
{@render MyCheckbox({ label: "Promotions", value: "promotions" })}
{@render MyCheckbox({ label: "News", value: "news" })}
{@render MyCheckbox({ label: "Updates", value: "updates" })}
</div>
</Checkbox.Group>
{#snippet MyCheckbox({ value, label }: { value: string; label: string })}
{@const id = useId()}
<div class="flex items-center">
<Checkbox.Root
{id}
aria-labelledby="{id}-label"
class="peer inline-flex size-[25px] items-center justify-center rounded-md border border-muted bg-foreground transition-all duration-150 ease-in-out active:scale-98 data-[state=unchecked]:border-border-input data-[state=unchecked]:bg-background data-[state=unchecked]:hover:border-dark-40"
name="hello"
{value}
>
{#snippet children({ checked, indeterminate })}
<div class="inline-flex items-center justify-center text-background">
{#if indeterminate}
<Minus class="size-[15px]" weight="bold" />
{:else if checked}
<Check class="size-[15px]" weight="bold" />
{/if}
</div>
{/snippet}
</Checkbox.Root>
<Label.Root
id="{id}-label"
for={id}
class="pl-3 text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
>
{label}
</Label.Root>
</div>
{/snippet}
import typography from "@tailwindcss/typography";
import animate from "tailwindcss-animate";
import { fontFamily } from "tailwindcss/defaultTheme";
/** @type {import('tailwindcss').Config} */
export default {
darkMode: "class",
content: ["./src/**/*.{html,js,svelte,ts}"],
theme: {
container: {
center: true,
screens: {
"2xl": "1440px",
},
},
extend: {
colors: {
border: {
DEFAULT: "hsl(var(--border-card))",
input: "hsl(var(--border-input))",
"input-hover": "hsl(var(--border-input-hover))",
},
background: {
DEFAULT: "hsl(var(--background) / <alpha-value>)",
alt: "hsl(var(--background-alt) / <alpha-value>)",
},
foreground: {
DEFAULT: "hsl(var(--foreground) / <alpha-value>)",
alt: "hsl(var(--foreground-alt) / <alpha-value>)",
},
muted: {
DEFAULT: "hsl(var(--muted) / <alpha-value>)",
foreground: "hsl(var(--muted-foreground))",
},
dark: {
DEFAULT: "hsl(var(--dark) / <alpha-value>)",
4: "hsl(var(--dark-04))",
10: "hsl(var(--dark-10))",
40: "hsl(var(--dark-40))",
},
accent: {
DEFAULT: "hsl(var(--accent) / <alpha-value>)",
foreground: "hsl(var(--accent-foreground) / <alpha-value>)",
},
destructive: {
DEFAULT: "hsl(var(--destructive) / <alpha-value>)",
},
contrast: {
DEFAULT: "hsl(var(--contrast) / <alpha-value>)",
},
},
fontFamily: {
sans: ["Inter", ...fontFamily.sans],
mono: ["Source Code Pro", ...fontFamily.mono],
alt: ["Courier", ...fontFamily.sans],
},
fontSize: {
xxs: "10px",
},
borderWidth: {
6: "6px",
},
borderRadius: {
card: "16px",
"card-lg": "20px",
"card-sm": "10px",
input: "9px",
button: "5px",
"5px": "5px",
"9px": "9px",
"10px": "10px",
"15px": "15px",
},
height: {
input: "3rem",
"input-sm": "2.5rem",
},
boxShadow: {
mini: "var(--shadow-mini)",
"mini-inset": "var(--shadow-mini-inset)",
popover: "var(--shadow-popover)",
kbd: "var(--shadow-kbd)",
btn: "var(--shadow-btn)",
card: "var(--shadow-card)",
"date-field-focus": "var(--shadow-date-field-focus)",
},
opacity: {
8: "0.08",
},
scale: {
80: ".80",
98: ".98",
99: ".99",
},
},
keyframes: {
"accordion-down": {
from: { height: "0" },
to: { height: "var(--bits-accordion-content-height)" },
},
"accordion-up": {
from: { height: "var(--bits-accordion-content-height)" },
to: { height: "0" },
},
"caret-blink": {
"0%,70%,100%": { opacity: "1" },
"20%,50%": { opacity: "0" },
},
},
animation: {
"accordion-down": "accordion-down 0.2s ease-out",
"accordion-up": "accordion-up 0.2s ease-out",
"caret-blink": "caret-blink 1.25s ease-out infinite",
},
},
plugins: [typography, animate],
};
@import url("https://fonts.googleapis.com/css2?family=Source+Code+Pro:ital,wght@0,400;0,500;0,600;0,700;1,400;1,500;1,600;1,700&display=swap");
@import url("https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600&display=swap");
@tailwind base;
@tailwind components;
@tailwind utilities;
@layer base {
:root {
/* Colors */
--background: 0 0% 100%;
--background-alt: 0 0% 100%;
--foreground: 0 0% 9%;
--foreground-alt: 0 0% 32%;
--muted: 240 5% 96%;
--muted-foreground: 0 0% 9% / 0.4;
--border: 240 6% 10%;
--border-input: 240 6% 10% / 0.17;
--border-input-hover: 240 6% 10% / 0.4;
--border-card: 240 6% 10% / 0.1;
--dark: 240 6% 10%;
--dark-10: 240 6% 10% / 0.1;
--dark-40: 240 6% 10% / 0.4;
--dark-04: 240 6% 10% / 0.04;
--accent: 204 94% 94%;
--accent-foreground: 204 80% 16%;
--destructive: 347 77% 50%;
/* black */
--constrast: 0 0% 0%;
/* Shadows */
--shadow-mini: 0px 1px 0px 1px rgba(0, 0, 0, 0.04);
--shadow-mini-inset: 0px 1px 0px 0px rgba(0, 0, 0, 0.04) inset;
--shadow-popover: 0px 7px 12px 3px hsla(var(--dark-10));
--shadow-kbd: 0px 2px 0px 0px rgba(0, 0, 0, 0.07);
--shadow-btn: 0px 1px 0px 1px rgba(0, 0, 0, 0.03);
--shadow-card: 0px 2px 0px 1px rgba(0, 0, 0, 0.04);
--shadow-date-field-focus: 0px 0px 0px 3px rgba(24, 24, 27, 0.17);
}
.dark {
/* Colors */
--background: 0 0% 5%;
--background-alt: 0 0% 8%;
--foreground: 0 0% 95%;
--foreground-alt: 0 0% 70%;
--muted: 240 4% 16%;
--muted-foreground: 0 0% 100% / 0.4;
--border: 0 0% 96%;
--border-input: 0 0% 96% / 0.17;
--border-input-hover: 0 0% 96% / 0.4;
--border-card: 0 0% 96% / 0.1;
--dark: 0 0% 96%;
--dark-40: 0 0% 96% / 0.4;
--dark-10: 0 0% 96% / 0.1;
--dark-04: 0 0% 96% / 0.04;
--accent: 204 90 90%;
--accent-foreground: 204 94% 94%;
--destructive: 350 89% 60%;
/* white */
--constrast: 0 0% 100%;
/* Shadows */
--shadow-mini: 0px 1px 0px 1px rgba(0, 0, 0, 0.3);
--shadow-mini-inset: 0px 1px 0px 0px rgba(0, 0, 0, 0.5) inset;
--shadow-popover: 0px 7px 12px 3px hsla(0deg 0% 0% / 30%);
--shadow-kbd: 0px 2px 0px 0px rgba(255, 255, 255, 0.07);
--shadow-btn: 0px 1px 0px 1px rgba(0, 0, 0, 0.2);
--shadow-card: 0px 2px 0px 1px rgba(0, 0, 0, 0.4);
--shadow-date-field-focus: 0px 0px 0px 3px rgba(244, 244, 245, 0.1);
}
}
@layer base {
* {
@apply border-border;
}
html {
-webkit-text-size-adjust: 100%;
font-variation-settings: normal;
}
body {
@apply bg-background text-foreground;
font-feature-settings:
"rlig" 1,
"calt" 1;
}
/* Mobile tap highlight */
/* https://developer.mozilla.org/en-US/docs/Web/CSS/-webkit-tap-highlight-color */
html {
-webkit-tap-highlight-color: rgba(128, 128, 128, 0.5);
}
::selection {
background: #fdffa4;
color: black;
}
/* === Scrollbars === */
::-webkit-scrollbar {
@apply w-2;
@apply h-2;
}
::-webkit-scrollbar-track {
@apply !bg-transparent;
}
::-webkit-scrollbar-thumb {
@apply rounded-card-lg !bg-dark-10;
}
::-webkit-scrollbar-corner {
background: rgba(0, 0, 0, 0);
}
/* Firefox */
/* https://developer.mozilla.org/en-US/docs/Web/CSS/scrollbar-color#browser_compatibility */
html {
scrollbar-color: var(--bg-muted);
}
.antialised {
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
}
@layer utilities {
.step {
counter-increment: step;
}
.step:before {
@apply absolute inline-flex h-9 w-9 items-center justify-center rounded-full border-4 border-background bg-muted text-center -indent-px font-mono text-base font-medium;
@apply ml-[-50px] mt-[-4px];
content: counter(step);
}
}
@layer components {
*:not(body):not(.focus-override) {
outline: none !important;
&:focus-visible {
@apply focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-foreground focus-visible:ring-offset-2 focus-visible:ring-offset-background;
}
}
.link {
@apply inline-flex items-center gap-1 rounded-sm font-medium underline underline-offset-4 hover:text-foreground/80 focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-foreground focus-visible:ring-offset-2 focus-visible:ring-offset-background;
}
input::-webkit-outer-spin-button,
input::-webkit-inner-spin-button {
-webkit-appearance: none;
margin: 0;
}
/* Firefox */
input[type="number"] {
-moz-appearance: textfield;
}
}
Managing Value State
Bits UI offers several approaches to manage and synchronize a Checkbox Group's value state, catering to different levels of control and integration needs.
1. Two-Way Binding
For seamless state synchronization, use Svelte's bind:value
directive. This method automatically keeps your local state in sync with the group's internal state.
<script lang="ts">
import { Checkbox } from "bits-ui";
let myValue = $state<string[]>([]);
</script>
<button
onclick={() => {
myValue = ["item-1", "item-2"];
}}
>
Open Items 1 and 2
</button>
<Checkbox.Group name="myItems" bind:value={myValue}>
<Checkbox.GroupLabel>Items</Checkbox.GroupLabel>
<Checkbox.Root value="item-1" />
<Checkbox.Root value="item-2" />
<Checkbox.Root value="item-3" />
</Checkbox.Group>
Key Benefits
- Simplifies state management
- Automatically updates
myValue
when the accordion changes (e.g., via clicking on an item's trigger) - Allows external control (e.g., opening an item via a separate button)
2. Change Handler
For more granular control or to perform additional logic on state changes, use the onValueChange
prop. This approach is useful when you need to execute custom logic alongside state updates.
<script lang="ts">
import { Checkbox } from "bits-ui";
let myValue = $state<string[]>([]);
</script>
<Checkbox.Group
value={myValue}
onValueChange={(value) => {
myValue = value;
// additional logic here.
}}
>
<Checkbox.GroupLabel>Items</Checkbox.GroupLabel>
<Checkbox.Root value="item-1" />
<Checkbox.Root value="item-2" />
<Checkbox.Root value="item-3" />
</Accordion.Root>
Use Cases
- Implementing custom behaviors on value change
- Integrating with external state management solutions
- Triggering side effects (e.g., logging, data fetching)
3. Fully Controlled
For complete control over the Checkbox Group's value state, use the controlledValue
prop. This approach requires you to manually manage the value state, giving you full control over when and how the group responds to value change events.
To implement controlled state:
- Set the
controlledValue
prop totrue
on theCheckbox.Group
component. - Provide a
value
prop toCheckbox.Group
, which should be a variable holding the current state. - Implement an
onValueChange
handler to update the state when the internal state changes.
<script lang="ts">
import { Checkbox } from "bits-ui";
let myValue = $state("");
</script>
<Checkbox.Group controlledValue value={myValue} onValueChange={(v) => (myValue = v)}>
<!-- ... -->
</Checkbox.Group>
When to Use
- Implementing complex logic
- Coordinating multiple UI elements
- Debugging state-related issues
Note
While powerful, fully controlled state should be used judiciously as it increases complexity and can cause unexpected behaviors if not handled carefully.
For more in-depth information on controlled components and advanced state management techniques, refer to our Controlled State documentation.
HTML Forms
To render hidden <input />
elements for the various checkboxes within a group, pass a name
to Checkbox.Group
. All descendent checkboxes will then render hidden inputs with the same name.
<Checkbox.Group name="notifications">
<!-- ... -->
</Checkbox.Group>
When a Checkbox.Group
component is used, its descendent Checkbox.Root
components will use certain properties from the group, such as the name
, required
, and disabled
.
API Reference
The button component used to toggle the state of the checkbox.
Property | Type | Description |
---|---|---|
checked $bindable | boolean | The checkbox button's checked state. This can be a boolean or the string 'indeterminate', which would typically display a dash in the checkbox. Default: false |
onCheckedChange | function | A callback that is fired when the checkbox button's checked state changes. Default: undefined |
controlledChecked | boolean | Whether or not the Default: false |
indeterminate $bindable | boolean | Whether the checkbox is an indeterminate state or not. Default: false |
onIndeterminateChange | function | A callback that is fired when the indeterminate state changes. Default: undefined |
controlledIndeterminate | boolean | Whether or not the Default: false |
disabled | boolean | Whether or not the checkbox button is disabled. This prevents the user from interacting with it. Default: false |
required | boolean | Whether or not the checkbox is required. Default: false |
name | string | The name of the checkbox. If provided a hidden input will be render to use for form submission. If not provided, the hidden input will not be rendered. Default: undefined |
value | string | The value of the checkbox. This is what is submitted with the form when the checkbox is checked. Default: undefined |
ref $bindable | HTMLButtonElement | The underlying DOM element being rendered. You can bind to this to get a reference to the element. Default: undefined |
children | Snippet | The children content to render. Default: undefined |
child | Snippet | Use render delegation to render your own element. See Child Snippet docs for more information. Default: undefined |
Data Attribute | Value | Description |
---|---|---|
data-state | enum | The checkbox's state of checked, unchecked, or indeterminate. |
data-disabled | '' | Present when the checkbox is disabled. |
data-checkbox-root | '' | Present on the root element. |
A group that synchronizes its value state with its descendant checkboxes.
Property | Type | Description |
---|---|---|
value $bindable | string[] | The value of the group. This is an array of the values of the checked checkboxes within the group. Default: [] |
onValueChange | function | A callback that is fired when the checkbox group's value state changes. Default: undefined |
controlledValue | boolean | Whether or not the Default: false |
disabled | boolean | Whether or not the checkbox group is disabled. If Default: false |
required | boolean | Whether or not the checkbox group is required for form submission. Default: false |
name | string | The name of the checkbox group. If provided a hidden input will be rendered to use for form submission. Default: undefined |
ref $bindable | HTMLDivElement | The underlying DOM element being rendered. You can bind to this to get a reference to the element. Default: undefined |
children | Snippet | The children content to render. Default: undefined |
child | Snippet | Use render delegation to render your own element. See Child Snippet docs for more information. Default: undefined |
Data Attribute | Value | Description |
---|---|---|
data-disabled | '' | Present when the checkbox group is disabled. |
data-checkbox-group | '' | Present on the group element. |
An accessible label for the checkbox group.
Property | Type | Description |
---|---|---|
ref $bindable | HTMLLabelElement | The underlying DOM element being rendered. You can bind to this to get a reference to the element. Default: undefined |
children | Snippet | The children content to render. Default: undefined |
child | Snippet | Use render delegation to render your own element. See Child Snippet docs for more information. Default: undefined |
Data Attribute | Value | Description |
---|---|---|
data-disabled | '' | Present when the checkbox group is disabled. |
data-checkbox-group-label | '' | Present on the label element. |