Switch

Find the source code here.

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;
    },
  };
}


	
ComponentsSVG
ComponentsTable