Switch
Overview
Switch component has same behaviour than checkbox. You can notice that the difference is only the design.
The switch listener is x-on:change. It will give you the checked value and the field's name. A pretty good behaviour is the ability to update label.
See examples below to know how to customize it.
Copied !
import { Switch } from "$components/switch.component";
export function DefaultSwitchExample() {
return (
<div class="grid grid-cols-2 gap-4">
<div class="p-4 border rounded-lg w-52">
<Switch
label="Enable Feature"
name="switch-example"
id="switch-example"
x-on:change="console.log($event.detail)"
class="[&>button]:h-6"
/>
</div>
<div class="p-4 border rounded-lg w-52">
<Switch
label="Primary Label"
name="switch-example-primary"
id="switch-example-primary"
x-on:change="console.log($event.detail)"
type="primary"
class="[&>button]:h-4 [&>button>span]:h-8 [&>button>span]:w-8"
/>
</div>
<div class="p-4 border rounded-lg w-52">
<Switch
label="Secondary Label"
name="switch-example-secondary"
id="switch-example-secondary"
x-on:change="console.log($event.detail)"
type="secondary"
class="[&>button]:h-6"
/>
</div>
<div class="p-4 border rounded-lg w-52">
<Switch
label="Label on Left"
name="switch-example-success"
id="switch-example-success"
x-on:change="console.log($event.detail)"
type="success"
class="[&>button]:h-6 flex-row-reverse"
/>
</div>
<div class="p-4 border rounded-lg w-52">
<Switch
label="Danger Label"
name="switch-example-danger"
id="switch-example-danger"
x-on:change="console.log($event.detail)"
type="danger"
class="[&>button]:h-6"
/>
</div>
<div class="p-4 border rounded-lg w-52">
<Switch
label="Info Label"
name="switch-example-info"
id="switch-example-info"
x-on:change="console.log($event.detail)"
type="info"
class="[&>button]:h-6"
/>
</div>
<div class="p-4 border rounded-lg w-52">
<Switch
label="Warning Label"
name="switch-example-warning"
id="switch-example-warning"
x-on:change="console.log($event.detail)"
type="warning"
class="[&>button]:h-6"
/>
</div>
<div class="p-4 border rounded-lg w-32">
<Switch
label="Off"
name="switch-example-warning"
id="switch-example-warning"
x-on:change={`
label = $event.detail.checked ? 'On' : 'Off';
console.log($event.detail)
`}
type="success"
class="[&>button]:h-4 [&>button]:rounded [&>button>span]:w-4 [&>button>span]:h-8"
/>
</div>
</div>
);
}
Installation
Notes
To use this component, you need to initialize your project first. If not done yet, run one of the following command:
npx jsxpine init or yarn jsxpine init or pnpm jsxpine init or bunx jsxpine init.
Go to the installation and usage page to learn more.
jsxpine add switch
Copied !
Copied !
-------------------- Switch Component -------------------------
import clsx from "clsx";
/**
* @typedef SwitchProps
* @type {{ label: string } & import("./input.component").InputProps & import("../common/props").ThemeColorProps}
*/
/**
* Switch component props
* @type {import("../common/props").JSXComponent<SwitchProps>}
*/
export function Switch(props) {
const {
class: className,
name,
id,
label,
type = "primary",
...restProps
} = props;
/**
* @type {Map<import("../common/types").ThemeColorType, { button: string; label: string }>}
*/
const colorType = new Map([
[
"primary",
{
button: "bg-primary",
label: "text-primary-dark-1 dark:text-primary-light-1"
}
],
[
"secondary",
{
button: "bg-secondary",
label: "text-secondary dark:text-secondary-light"
}
],
[
"success",
{
button: "bg-success",
label: "text-success-dark dark:text-success-light"
}
],
[
"danger",
{ button: "bg-danger", label: "text-danger-dark" }
],
[
"info",
{ button: "bg-info", label: "text-info-dark dark:text-info-light" }
],
[
"warning",
{
button: "bg-warning",
label: "text-warning-dark dark:text-warning-light"
}
]
]);
return (
<div
x-data={`switchCheckbox("${label}")`}
class={clsx("flex items-center justify-center gap-x-2", className)}
{...restProps}
>
<input
id={id}
type="checkbox"
name={name}
x-ref="checkbox"
class="hidden"
x-bind:checked="switchOn"
/>
<button
x-ref="switchButton"
type="button"
x-on:click="toggle"
x-bind:class={`switchOn ? '${
colorType.get(type)?.button
}' : 'bg-base-light'`}
class="relative transition-colors duration-200 inline-flex focus:outline-none rounded-full grow"
x-cloak="true"
>
<span
x-bind:class="switchOn ? 'left-full -translate-x-[110%]' : 'translate-x-0.5'"
class="absolute top-1/2 -translate-y-1/2 left-0 w-5 h-5 duration-200 ease-in-out bg-background rounded-full shadow-md"
/>
</button>
<template x-if="label">
<label
x-on:click="$refs.switchButton.click(); $refs.switchButton.focus()"
x-bind:class={`{ '${
colorType.get(type)?.label
}' : switchOn, 'text-foreground': !switchOn }`}
x-text="label"
class="text-sm select-none"
x-cloak="true"
/>
</template>
</div>
);
}
-------------------- Alpine Dependencies -------------------------
/**
* @typedef {Object} SwitchCheckboxDataOutput
* @property {boolean} switchOn
* @property {string} label
* @property {Function} toggle
*/
/**
* Switch Checkbox alpine data
* @param {string} label
* @returns {import("alpinejs").AlpineComponent<SwitchCheckboxDataOutput>}
*/
export function switchCheckboxData(label = "") {
return {
init() {
this.$watch("switchOn", (value) => {
this.$dispatch("change", {
checked: value,
name: this.$refs.checkbox.name,
});
});
},
switchOn: false,
label,
toggle() {
this.switchOn = !this.switchOn;
},
};
}