Sidebar

Find the source code here.

Overview

Sidebar is a slide-over component which mainly has same purpose as modal.

Very often, we display menu (common usecase for mobile web application) or a form without take too much places in the main context.

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 sidebar
Copied !

Copied !


	--------------------     Components Dependencies   -------------------------

		import clsx from "clsx";

/**
 * @typedef HTMLButtonTagWithChildren
 * @type {import("@kitajs/html").PropsWithChildren<Omit<JSX.HtmlButtonTag, "className"> & import("../common/props").CLSXClassProps>}
 */

/**
 * @typedef {Object} ButtonProps
 * @type {{ text?: string } & HTMLButtonTagWithChildren & import("../common/props").BorderRadiusProps & import("../common/props").SizeProps & import("../common/props").VariantColorProps}
 */

/**
 * Button component props
 * @param {ButtonProps} props
 */
export function Button({
	children,
	class: className,
	size = "md",
	text,
	variant = "solid",
	borderRadius = "rounded",
	...restProps
}) {
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantColorMap = new Map([
		["solid", "btn"],
		["outlined", "btn-outlined"],
		["inversed", "btn-inversed"]
	]);

	/**
	 * @type {Map<import("../common/types").SizeType, string>}
	 */
	const sizeMap = new Map([
		["xs", "text-xs px-2 py-1"],
		["sm", "px-4 py-2 text-sm"],
		["md", "px-4 py-2"],
		["lg", "px-8 py-4 text-lg"],
		["xl", "px-12 py-6 text-xl"],
		["2xl", "px-16 py-8 text-4xl"]
	]);

	/**
	 * @type {Map<import("../common/types").BorderRadiusType, string>}
	 */
	const borderRadiusMap = new Map([
		["square", "rounded-none"],
		["rounded", "rounded"],
		["arc", "rounded-xl"],
		["pill", "rounded-full"],
		["curve", "rounded-lg"],
		["circle", "aspect-square rounded-full"]
	]);

	return (
		<button
			class={clsx(
				variantColorMap.get(variant),
				sizeMap.get(size),
				borderRadiusMap.get(borderRadius),
				className
			)}
			{...restProps}
		>
			{text ? <span safe>{text}</span> : children}
		</button>
	);
}

/**
 * Primary Button component props
 * @type {import("../common/props").JSXComponent<ButtonProps>}
 */
export function PrimaryButton(props) {
	const { children, class: className, variant = "solid", ...restProps } = props;
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantClassMap = new Map([
		["solid", "btn-primary"],
		["outlined", "btn-primary-outlined"],
		["inversed", "btn-primary-inversed"]
	]);

	return (
		<Button
			class={[variantClassMap.get(variant), className].join(" ")}
			{...restProps}
		>
			{children}
		</Button>
	);
}

/**
 * Secondary Button component props
 * @type {import("../common/props").JSXComponent<ButtonProps>}
 */
export function SecondaryButton(props) {
	const { children, class: className, variant = "solid", ...restProps } = props;
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantClassMap = new Map([
		["solid", "btn-secondary"],
		["outlined", "btn-secondary-outlined"],
		["inversed", "btn-secondary-inversed"]
	]);

	return (
		<Button
			class={[variantClassMap.get(variant), className].join(" ")}
			{...restProps}
		>
			{children}
		</Button>
	);
}

/**
 * Success Button component props
 * @type {import("../common/props").JSXComponent<ButtonProps>}
 */
export function SuccessButton(props) {
	const { children, class: className, variant = "solid", ...restProps } = props;
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantClassMap = new Map([
		["solid", "btn-success"],
		["outlined", "btn-success-outlined"],
		["inversed", "btn-success-inversed"]
	]);

	return (
		<Button
			class={[variantClassMap.get(variant), className].join(" ")}
			{...restProps}
		>
			{children}
		</Button>
	);
}

/**
 * Danger Button component props
 * @type {import("../common/props").JSXComponent<ButtonProps>}
 */
export function DangerButton(props) {
	const { children, class: className, variant = "solid", ...restProps } = props;
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantClassMap = new Map([
		["solid", "btn-danger"],
		["outlined", "btn-danger-outlined"],
		["inversed", "btn-danger-inversed"]
	]);

	return (
		<Button
			class={[variantClassMap.get(variant), className].join(" ")}
			{...restProps}
		>
			{children}
		</Button>
	);
}

/**
 * Info Button component props
 * @type {import("../common/props").JSXComponent<ButtonProps>}
 */
export function InfoButton(props) {
	const { children, class: className, variant = "solid", ...restProps } = props;
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantClassMap = new Map([
		["solid", "btn-info"],
		["outlined", "btn-info-outlined"],
		["inversed", "btn-info-inversed"]
	]);

	return (
		<Button
			class={[variantClassMap.get(variant), className].join(" ")}
			{...restProps}
		>
			{children}
		</Button>
	);
}

/**
 * Warning Button component props
 * @type {import("../common/props").JSXComponent<ButtonProps>}
 */
export function WarningButton(props) {
	const { children, class: className, variant = "solid", ...restProps } = props;
	/**
	 * @type {Map<import("../common/types").VariantColorType, string>}
	 */
	const variantClassMap = new Map([
		["solid", "btn-warning"],
		["outlined", "btn-warning-outlined"],
		["inversed", "btn-warning-inversed"]
	]);

	return (
		<Button
			class={[variantClassMap.get(variant), className].join(" ")}
			{...restProps}
		>
			{children}
		</Button>
	);
}


import { icons } from "@iconify-json/ri/icons.json";
import clsx from "clsx";
import { SVG } from "./svg.component";

/**
 * @typedef {`ri.${keyof typeof import("@iconify-json/ri/icons.json")["icons"]}`} IconName
 */

/**
 * @typedef IconProps
 * @type {{size?: number, name: IconName, color?: string, applyDefsId?: string} & import("./svg.component").SVGProps}
 */

/**
 * Icon component props
 * @param {Omit<IconProps, "viewBox">} props
 */
export function Icon({
	children,
	size = 4,
	name,
	applyDefsId,
	class: className,
	...restProps
}) {
	const iconType = name.split(".")[0];
	const iconName = /** @type {keyof typeof icons} */ (name.split(".")[1]);
	if (!icons[iconName]) {
		console.error(`"${name}" is not an icon from iconify/${iconType}`);
		return "";
	}
	const { body } = icons[iconName];

	// The purpose is to retrieve value from d attribute
	let retrieveDValue = "";
	const bodyMatch = body.match(/d=".+"/g);
	const retrieveDAttribute = bodyMatch ? bodyMatch[0] : "";
	if (retrieveDAttribute) {
		retrieveDValue = retrieveDAttribute.slice(3, -2);
	}

	return (
		<SVG
			viewBox={"0 0 24 24"}
			{...restProps}
			class={clsx(`size-${size}`, className)}
		>
			{children}
			<path
				stroke-linecap="round"
				stroke-linejoin="round"
				fill={applyDefsId ? `url(#${applyDefsId})` : "currentColor"}
				d={retrieveDValue}
			/>
		</SVG>
	);
}


/**
 * @typedef SVGProps
 * @type {{fill?: string, stroke?: string, strokeWidth?: number, viewBox: string } & Omit<JSX.HtmlTag, "className"> & import("../common/props").CLSXClassProps}
 */

/**
 * SVG component props
 * @type {import("../common/props").JSXComponent<SVGProps>}
 */
export function SVG(props) {
    const {
        children,
        class: className,
        fill = "none",
        stroke = "currentColor",
        strokeWidth = 0.5,
        ...restProps
        } = props;
    return (
        <svg
            xmlns="http://www.w3.org/2000/svg"
            fill={fill}
            stroke={stroke}
            class={className}
            style={{ strokeWidth: String(strokeWidth) }}
            {...restProps}
        >
            {children}
        </svg>
    )
}

	
	--------------------     Sidebar Component   -------------------------

		import { Button } from "./button.component";
import { Icon } from "./icon.component";
import clsx from "clsx";

/**
 * @typedef {Object} SidebarTriggerProps
 * @type {import("./button.component").ButtonProps}
 */

/**
 * @typedef {Object} SidebarContentProps
 * @type {{ selector?: string, title?: string } & import("../common/props").PositionProps & import("../common/props").HTMLTagWithChildren}
 */

/**
 * Sidebar Trigger component props
 * @type {import("../common/props").JSXComponent<SidebarTriggerProps>}
 */
export function SidebarTrigger(props) {
	const { ...restProps } = props;

	return (
		<div class="inline-flex items-center justify-center">
			<Button x-bind="trigger" {...restProps} />
		</div>
	);
}

/**
 * Sidebar Content component props
 * @type {import("../common/props").JSXComponent<SidebarContentProps>}
 */
export function SidebarContent(props) {
	const {
		children,
		selector = "body",
		title,
		position = "right",
		class: className
	} = props;

	/**
	 * @type {Map<import("../common/types").PositionType, string>}
	 */
	const positionClassMap = new Map([
		["right", "right-0"],
		["left", "left-0"],
		["top", "top-0"],
		["bottom", "bottom-0"]
	]);

	/**
	 * @type {Map<import("../common/types").PositionType, import("../common/types").TransitionStateType>}
	 */
	const transitionClassMap = new Map([
		[
			"right",
			{
				enter: {
					start: "translate-x-full opacity-0",
					end: "translate-x-0 opacity-100"
				},
				leave: {
					start: "translate-x-0 opacity-100",
					end: "translate-x-full opacity-0"
				}
			}
		],
		[
			"left",
			{
				enter: {
					start: "-translate-x-full opacity-0",
					end: "translate-x-0 opacity-100"
				},
				leave: {
					start: "translate-x-0 opacity-100",
					end: "-translate-x-full opacity-0"
				}
			}
		],
		[
			"top",
			{
				enter: {
					start: "-translate-y-full opacity-0",
					end: "translate-y-0 opacity-100"
				},
				leave: {
					start: "translate-y-0 opacity-100",
					end: "-translate-y-full opacity-0"
				}
			}
		],
		[
			"bottom",
			{
				enter: {
					start: "translate-y-full opacity-0",
					end: "translate-y-0 opacity-100"
				},
				leave: {
					start: "translate-y-0 opacity-100",
					end: "translate-y-full opacity-0"
				}
			}
		]
	]);

	return (
		<div
			x-bind="shower"
			class={clsx(
				"top-0 left-0 flex items-center justify-center",
				selector !== "body"
					? "absolute w-full h-full"
					: "fixed w-screen h-screen z-[99]"
			)}
		>
			<div
				x-bind="shower"
				x-transition:enter="ease-out duration-300"
				x-transition:enter-start="opacity-0"
				x-transition:enter-end="opacity-100"
				x-transition:leave="ease-in duration-300"
				x-transition:leave-start="opacity-100"
				x-transition:leave-end="opacity-0"
				x-on:click="close()"
				class="absolute inset-0 w-full h-full bg-overlay-dark/75"
			/>
			<div
				x-bind="shower"
				x-transition:enter="transition ease-in-out duration-500 sm:duration-700"
				x-transition:enter-start={transitionClassMap.get(position)?.enter.start}
				x-transition:enter-end={transitionClassMap.get(position)?.enter.end}
				x-transition:leave="transition ease-in-out duration-500 sm:duration-700"
				x-transition:leave-start={transitionClassMap.get(position)?.leave.start}
				x-transition:leave-end={transitionClassMap.get(position)?.leave.end}
				class={clsx(
					"absolute inset-O flex flex-col gap-y-2 h-full overflow-y-scroll bg-background border-base-dark/70",
					positionClassMap.get(position),
					className
				)}
			>
				<div class="flex items-center justify-between w-full p-2 sm:p-4">
					{title ? (
						<h2
							class="text-base font-semibold leading-6 w-full"
							id="slide-over-title"
							safe
						>
							{title}
						</h2>
					) : null}
					<div class="flex items-center justify-end w-full">
						<button
							x-bind="closerClick"
							class="flex items-center justify-between px-3 py-2 gap-x-2 text-xs font-medium uppercase bg-background text-muted-foreground rounded-md hover:text-foreground"
						>
							<Icon name="ri.close-line" />
							<span>Close</span>
						</button>
					</div>
				</div>
				{children}
			</div>
		</div>
	);
}

/**
 * Sidebar Content component props
 * @type {import("../common/props").JSXComponent<import("../common/props").HTMLTagWithChildren>}
 */
export function Sidebar(props) {
	const { children, ...restProps } = props;
	return (
		<div x-data="sidebar" x-bind="closerEscape" {...restProps}>
			{children}
		</div>
	);
}

	
	
	--------------------     Alpine Dependencies   -------------------------

		/**
 * @typedef {Object} SidebarDataOutput
 * @property {boolean} show
 * @property {Function} close
 * @property {Function} open
 * @property {Record<string, () => boolean>} shower
 * @property {Record<string, Function>} trigger
 * @property {Record<string, Function>} closerClick
 * @property {Record<string, Function>} closerEscape
 */

/**
 * Sidebar alpine data
 * @returns {import("alpinejs").AlpineComponent<SidebarDataOutput>}
 */
export function sidebarData() {
  return {
		destroy() {
			this.show = false;
		},
		show: false,
		close() {
			this.show = false;
		},
		open() {
			this.show = true;
		},
		shower: {
			["x-show"]() {
				return this.show;
			}
		},
		trigger: {
			["@click"]() {
				this.open();
			}
		},
		closerClick: {
			["@click"]() {
				this.close();
			}
		},
		closerEscape: {
			["@keydown.escape.window"]() {
				this.close();
			}
		}
	};
}


	

Default Sidebar

This example shows where to put your content when using Sidebar.

Sidebar Title

Copied !

import {
	Sidebar,
	SidebarTrigger,
	SidebarContent
} from "$components/sidebar.component";

export function DefaultSidebarExample() {
	return (
		<Sidebar>
			<SidebarTrigger text="Open Sidebar" />
			<SidebarContent
				title="Sidebar Title"
				class="h-full w-3/4 max-w-lg"
			>
				<div class="mx-2 mb-2 h-full border-dashed border border-muted-foreground"></div>
			</SidebarContent>
		</Sidebar>
	);
}

Full Example

Here an example with a pratical case, display a form in sidebar.

Create an account

Enter your email below to create your account

Or continue with

Already have an account?Login here

By continuing, you agree to our Terms and Policy

Copied !

import { PrimaryButton } from "$components/button.component";
import { Sidebar, SidebarContent } from "$components/sidebar.component";

export function FormSidebarExample() {
	return (
		<Sidebar>
			<PrimaryButton
				variant="outlined"
				borderRadius="square"
				text="Primary Button as Trigger"
				x-bind="trigger"
			/>
			<SidebarContent position="left" class="w-full md:w-auto">
				<div class="relative flex flex-wrap items-center w-full h-full p-2 md:p-8">
					<div class="relative w-full max-w-sm mx-auto lg:mb-0">
						<div class="relative text-center">
							<div class="flex flex-col mb-6 space-y-2">
								<h2 class="text-2xl font-semibold tracking-tight">
									Create an account
								</h2>
								<p class="text-sm text-neutral-500">
									Enter your email below to create your account
								</p>
							</div>
							<form x-on:submit="$event.preventDefault();" class="space-y-2">
								<input
									type="text"
									placeholder="name@example.com"
									class="flex w-full h-10 px-3 py-2 text-sm bg-white border rounded-md border-neutral-300 ring-offset-background placeholder:text-neutral-500 focus:border-neutral-300 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-neutral-400 disabled:cursor-not-allowed disabled:opacity-50"
								/>
								<button
									type="button"
									class="inline-flex items-center justify-center w-full h-10 px-4 py-2 text-sm font-medium tracking-wide text-white transition-colors duration-200 rounded-md bg-neutral-950 hover:bg-neutral-900 focus:ring-2 focus:ring-offset-2 focus:ring-neutral-900 focus:shadow-outline focus:outline-none"
								>
									Sign up with Email
								</button>
								<div class="relative py-6">
									<div class="absolute inset-0 flex items-center">
										<span class="w-full border-t"></span>
									</div>
									<div class="relative flex justify-center text-xs uppercase">
										<span class="px-2 bg-white text-neutral-500">
											Or continue with
										</span>
									</div>
								</div>
								<button
									class="inline-flex items-center justify-center w-full h-10 px-4 py-2 text-sm font-medium border rounded-md focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:pointer-events-none border-input hover:bg-neutral-100"
									type="button"
								>
									<svg viewBox="0 0 438.549 438.549" class="w-4 h-4 mr-2">
										<path
											fill="currentColor"
											d="M409.132 114.573c-19.608-33.596-46.205-60.194-79.798-79.8-33.598-19.607-70.277-29.408-110.063-29.408-39.781 0-76.472 9.804-110.063 29.408-33.596 19.605-60.192 46.204-79.8 79.8C9.803 148.168 0 184.854 0 224.63c0 47.78 13.94 90.745 41.827 128.906 27.884 38.164 63.906 64.572 108.063 79.227 5.14.954 8.945.283 11.419-1.996 2.475-2.282 3.711-5.14 3.711-8.562 0-.571-.049-5.708-.144-15.417a2549.81 2549.81 0 01-.144-25.406l-6.567 1.136c-4.187.767-9.469 1.092-15.846 1-6.374-.089-12.991-.757-19.842-1.999-6.854-1.231-13.229-4.086-19.13-8.559-5.898-4.473-10.085-10.328-12.56-17.556l-2.855-6.57c-1.903-4.374-4.899-9.233-8.992-14.559-4.093-5.331-8.232-8.945-12.419-10.848l-1.999-1.431c-1.332-.951-2.568-2.098-3.711-3.429-1.142-1.331-1.997-2.663-2.568-3.997-.572-1.335-.098-2.43 1.427-3.289 1.525-.859 4.281-1.276 8.28-1.276l5.708.853c3.807.763 8.516 3.042 14.133 6.851 5.614 3.806 10.229 8.754 13.846 14.842 4.38 7.806 9.657 13.754 15.846 17.847 6.184 4.093 12.419 6.136 18.699 6.136 6.28 0 11.704-.476 16.274-1.423 4.565-.952 8.848-2.383 12.847-4.285 1.713-12.758 6.377-22.559 13.988-29.41-10.848-1.14-20.601-2.857-29.264-5.14-8.658-2.286-17.605-5.996-26.835-11.14-9.235-5.137-16.896-11.516-22.985-19.126-6.09-7.614-11.088-17.61-14.987-29.979-3.901-12.374-5.852-26.648-5.852-42.826 0-23.035 7.52-42.637 22.557-58.817-7.044-17.318-6.379-36.732 1.997-58.24 5.52-1.715 13.706-.428 24.554 3.853 10.85 4.283 18.794 7.952 23.84 10.994 5.046 3.041 9.089 5.618 12.135 7.708 17.705-4.947 35.976-7.421 54.818-7.421s37.117 2.474 54.823 7.421l10.849-6.849c7.419-4.57 16.18-8.758 26.262-12.565 10.088-3.805 17.802-4.853 23.134-3.138 8.562 21.509 9.325 40.922 2.279 58.24 15.036 16.18 22.559 35.787 22.559 58.817 0 16.178-1.958 30.497-5.853 42.966-3.9 12.471-8.941 22.457-15.125 29.979-6.191 7.521-13.901 13.85-23.131 18.986-9.232 5.14-18.182 8.85-26.84 11.136-8.662 2.286-18.415 4.004-29.263 5.146 9.894 8.562 14.842 22.077 14.842 40.539v60.237c0 3.422 1.19 6.279 3.572 8.562 2.379 2.279 6.136 2.95 11.276 1.995 44.163-14.653 80.185-41.062 108.068-79.226 27.88-38.161 41.825-81.126 41.825-128.906-.01-39.771-9.818-76.454-29.414-110.049z"
										></path>
									</svg>
									<span>Github</span>
								</button>
							</form>
						</div>
						<p class="mt-6 text-sm text-center text-neutral-500">
							<span>Already have an account?</span>
							<a href="#_" class="relative font-medium text-blue-600 group">
								<span>Login here</span>
								<span class="absolute bottom-0 left-0 w-0 group-hover:w-full ease-out duration-300 h-0.5 bg-blue-600"></span>
							</a>
						</p>
						<p class="px-8 mt-1 text-sm text-center text-neutral-500">
							By continuing, you agree to our{" "}
							<a
								class="underline underline-offset-4 hover:text-primary"
								href="/terms"
							>
								Terms
							</a>{" "}
							and{" "}
							<a
								class="underline underline-offset-4 hover:text-primary"
								href="/privacy"
							>
								Policy
							</a>
						</p>
					</div>
				</div>
			</SidebarContent>
		</Sidebar>
	);
}
ComponentsSelect
ComponentsSVG