Inputs

Find the source code here.

Overview

Inputs are elements which allow you to let user interacts with your website.

They are essantially : text input, password, number, select, textarea, radio, checkbox. Many others will be presented in this page.

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

Copied !


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

		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>
    )
}

	
	--------------------     Input Component   -------------------------

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

/**
 * @typedef {Object} InputProps
 * @type {Omit<JSX.HtmlInputTag, "className"> & import("../common/props").CLSXClassProps}
 */

/**
 * @typedef TextInputProps
 * @type {InputProps}
 */

/**
 * @typedef PasswordInputProps
 * @type {{ hideIcon?: boolean } & InputProps}
 */

/**
 * @typedef NumberInputProps
 * @type {Omit<InputProps, "placeholder" | "min" | "max"> & { min?: number, max?: number }}
 */

/**
 * @typedef TextareaInputProps
 * @type {{ value?: string, noResize?: boolean} & Omit<JSX.HtmlTextAreaTag, "className"> & import("../common/props").CLSXClassProps}
 */

/**
 * @typedef {Object} DefaultSelectOptionType
 * @property {string} label
 * @property {string | number} [value]
 * @property {boolean} [disabled]
 */

/**
 * @typedef SelectOptionType
 * @type {DefaultSelectOptionType & Record<string, unknown>}
 */

/**
 * @typedef SelectInputProps
 * @type {{ value?: string | number | string[], placeholder?: string, items?: SelectOptionType[], hidePlaceholder?: boolean } & Omit<JSX.HtmlSelectTag, "className"> & import("../common/props").CLSXClassProps}
 */

/**
 * @typedef CheckboxInputProps
 * @type {Omit<InputProps, "placeholder">}
 */

/**
 * @typedef RadioInputProps
 * @type {Omit<InputProps, "placeholder">}
 */

/**
 * Checkbox Input props
 * @type {import("../common/props").JSXComponent<CheckboxInputProps>}
 */
export function CheckboxInput(props) {
	return <input type="checkbox" {...props} />;
}

/**
 * Date Input props
 * @type {import("../common/props").JSXComponent<TextInputProps>}
 */
export function DateInput(props) {
	const { value = "", class: className, ...restProps } = props;
	return (
		<div
			class={clsx("input flex items-center justify-center gap-x-2", className)}
			x-data={`{ value: "${value}" }`}
		>
			<input
				x-ref="dateInput"
				type="date"
				{...restProps}
				x-model="value"
				class="flex-1 px-1"
			/>
			{/* Because calendar-picker-icon doesn't change with theme toggle */}
			<button x-on:click="$refs.dateInput.showPicker()">
				<Icon name="ri.calendar-line" />
			</button>
		</div>
	);
}

/**
 * Email Input props
 * @type {import("../common/props").JSXComponent<TextInputProps>}
 */
export function EmailInput(props) {
	const { value = "", class: className, ...restProps } = props;
	return (
		<div
			class={clsx("input input-primary flex items-center gap-x-2", className)}
			x-data={`{ value: "${value}" }`}
		>
			<input type="email" {...restProps} x-model="value" class="flex-1 px-1" />
			<button
				type="button"
				x-bind:disabled="!value"
				x-on:click="value = ''"
				class="flex items-center justify-center"
				x-bind:class="value ? 'opacity-100' : 'opacity-0'"
			>
				<Icon name="ri.close-line" size={5} />
			</button>
		</div>
	);
}

/**
 * File Input props
 * @type {import("../common/props").JSXComponent<TextInputProps>}
 */
export function FileInput(props) {
	const { value = "", class: className, ...restProps } = props;
	return (
		<div
			class={clsx("input flex items-center justify-center", className)}
			x-data={`{ value: "${value}" }`}
		>
			<input type="file" {...restProps} x-model="value" class="flex-1 px-1" />
		</div>
	);
}

/**
 * Number Input props
 * @type {import("../common/props").JSXComponent<NumberInputProps>}
 */
export function NumberInput(props) {
	const { min = 1, max, value = 1, class: className, ...restProps } = props;
	return (
		<input
			type="number"
			x-data={`{ value: ${value} }`}
			class={clsx("text-center input out-of-range:bg-danger-500", className)}
			{...restProps}
			min={String(min)}
			max={String(max)}
			x-model="value"
			pattern="/\d/g"
		/>
	);
}

/**
 * Password Input props
 * @type {import("../common/props").JSXComponent<PasswordInputProps>}
 */
export function PasswordInput(props) {
	const {
		value = "",
		hideIcon = false,
		class: className,
		...restProps
	} = props;
	return (
		<div
			class={clsx("input input-primary flex items-center gap-x-2", className)}
			x-data={`{ value: "${value}", show: false }`}
		>
			<input
				x-bind:type="show ? 'text' : 'password'"
				{...restProps}
				x-model="value"
				class="flex-1 px-1"
			/>
			{!hideIcon ? (
				<button
					type="button"
					x-bind:disabled="!value"
					x-on:click="show = !show"
					class="flex items-center justify-center"
				>
					<template x-if="!show">
						<Icon name="ri.eye-line" stroke-width="0.5" size={5} />
					</template>
					<template x-if="show">
						<Icon name="ri.eye-off-line" stroke-width="0.5" size={5} />
					</template>
				</button>
			) : null}
		</div>
	);
}

/**
 * Radio Input props
 * @type {import("../common/props").JSXComponent<RadioInputProps>}
 */
export function RadioInput(props) {
	return <input type="radio" {...props} />;
}

/**
 * Select Input props
 * @type {import("../common/props").JSXComponent<SelectInputProps>}
 */
export function SelectInput(props) {
	const {
		items = [],
		placeholder = "Select a value",
		hidePlaceholder = false,
		value,
		class: className,
		...restProps
	} = props;
	return (
		<select
			{...restProps}
			x-data={`{ value: "${value}" }`}
			class={clsx(
				"w-full p-2 text-md border border-base-light rounded cursor-pointer focus:outline-none focus:ring-2 focus:border-primary"
			)}
		>
			<option class={clsx({ hidden: hidePlaceholder })} selected safe>
				{placeholder}
			</option>
			{items.map((item) => {
				return (
					<option value={String(item.value)} safe>
						{item.label}
					</option>
				);
			})}
		</select>
	);
}

/**
 * Text Input props
 * @type {import("../common/props").JSXComponent<TextInputProps>}
 */
export function TextInput(props) {
	const { value = "", class: className, ...restProps } = props;
	return (
		<div
			class={clsx("input input-primary flex items-center gap-x-2", className)}
			x-data={`{ value: "${value}" }`}
		>
			<input type="text" {...restProps} x-model="value" class="flex-1 px-1" />
			<button
				type="button"
				x-bind:disabled="!value"
				x-on:click="value = ''"
				class="flex items-center justify-center"
				x-bind:class="value ? 'opacity-100' : 'opacity-0'"
			>
				<Icon name="ri.close-line" size={5} />
			</button>
		</div>
	);
}

/**
 * Textarea Input props
 * @type {import("../common/props").JSXComponent<TextareaInputProps>}
 */
export function TextareaInput(props) {
	const {
		value = "",
		noResize = false,
		class: className,
		...restProps
	} = props;
	return (
		<textarea
			x-data={`{ value: "${value}" }`}
			x-model="value"
			class={clsx("border rounded-lg h-full p-2", className)}
			{...restProps}
		></textarea>
	);
}

/**
 * Text Input props
 * @type {import("../common/props").JSXComponent<TextInputProps>}
 */
export function TimeInput(props) {
	const { value = "", class: className, ...restProps } = props;
	return (
		<div
			class={clsx("input flex items-center justify-center gap-x-2", className)}
			x-data={`{ value: "${value}" }`}
		>
			<input
				x-ref="timeInput"
				type="time"
				{...restProps}
				x-model="value"
				class="flex-1 px-1"
			/>
			{/* Because calendar-picker-icon doesn't change with theme toggle */}
			<button x-on:click="$refs.timeInput.showPicker()">
				<Icon name="ri.time-line" />
			</button>
		</div>
	);
}

	
	

Main Inputs

Here below are main inputs that are mostly use on a website.

Copied !

import { EmailField, DateField, FileField, NumberField, PasswordField, TextareaField, TextField, TimeField } from "$components/field.component";

export function MainInputExample() {
	return (
		<div class="grid grid-cols-2 gap-x-2 gap-y-4 items-center">
			<TextField
				label="Text Field"
				id="username"
				name="username"
				placeholder="type your username here"
				x-on:input="console.log($event.target.value)"
			/>
			<PasswordField
				label="Password Field"
				id="password"
				name="password"
				placeholder="type your password here"
			/>
			<EmailField
				label="Email Field"
				id="email"
				name="email"
				placeholder="example@email.com"
				x-on:input="console.log($event.target.value)"
			/>
			<div class="flex items-center justify-center">
				<NumberField
					label="Number Field"
					id="quantity"
					name="quantity"
					min={1}
					max={5}
					class="w-24"
					x-on:input="console.log($event.target.value)"
				/>
			</div>
			<DateField
				label="Select a date"
				id="datefield"
				name="datefield"
				x-on:input="console.log($event.target.value)"
			/>
			<TimeField
				label="Select a time"
				id="timefield"
				name="timefield"
				x-on:input="console.log($event.target.value)"
			/>
			<FileField
				label="Select a file"
				x-on:input="console.log($event.target.value)"
			/>
			<FileField
				label="Select files"
				x-on:input="console.log($event.target.value)"
				multiple
			/>
			<div class="col-span-2 border">
				<TextareaField
					placeholder="This is textarea component"
					label="Textarea Field"
					x-on:input="console.log($event.target.value)"
					class="flex flex-col h-48 w-full"
				/>
			</div>
		</div>
	);
}

Selected Input

Select is an element listing a number of value to choose.

Copied !

import { SelectField } from "$components/field.component";

export function SelectInputExample() {
	/**
	 * @type {Array<import("$components/input.component").SelectOptionType>}
	 */
	const items = [
		{
			label: "Milk",
			value: "milk",
			disabled: false
		},
		{
			label: "Eggs",
			value: "eggs",
			disabled: false
		},
		{
			label: "Cheese",
			value: "cheese",
			disabled: false
		},
		{
			label: "Bread",
			value: "bread",
			disabled: false
		},
		{
			label: "Apples",
			value: "apples",
			disabled: false
		},
		{
			label: "Bananas",
			value: "bananas",
			disabled: false
		},
		{
			label: "Yogurt",
			value: "yogurt",
			disabled: false
		},
		{
			label: "Sugar",
			value: "sugar",
			disabled: false
		},
		{
			label: "Salt",
			value: "salt",
			disabled: false
		},
		{
			label: "Coffee",
			value: "coffee",
			disabled: false
		},
		{
			label: "Tea",
			value: "tea",
			disabled: false
		}
	];

	return (
		<SelectField
			label="Select Field"
			id="select-field"
			name="select-field"
			placeholder="Select an item"
			x-on:change="console.log($event.target.value)"
			items={items}
		/>
	);
}

multiple-select-input

Multiple Selected Input

A select input component with multiple props set to true.

Copied !

import { SelectField } from "$components/field.component";

export function MultipleSelectInputExample() {
	/**
	 * @type {Array<import("$components/input.component").SelectOptionType>}
	 */
	const items = [
		{
			label: "Milk",
			value: "milk",
			disabled: false
		},
		{
			label: "Eggs",
			value: "eggs",
			disabled: false
		},
		{
			label: "Cheese",
			value: "cheese",
			disabled: false
		},
		{
			label: "Bread",
			value: "bread",
			disabled: false
		},
		{
			label: "Apples",
			value: "apples",
			disabled: false
		},
		{
			label: "Bananas",
			value: "bananas",
			disabled: false
		},
		{
			label: "Yogurt",
			value: "yogurt",
			disabled: false
		},
		{
			label: "Sugar",
			value: "sugar",
			disabled: false
		},
		{
			label: "Salt",
			value: "salt",
			disabled: false
		},
		{
			label: "Coffee",
			value: "coffee",
			disabled: false
		},
		{
			label: "Tea",
			value: "tea",
			disabled: false
		}
	];

	return (
		<SelectField
			label="Select Field"
			id="select-field"
			name="select-field"
			placeholder="Select an item"
			x-on:change="console.log($event.target.value)"
			items={items}
			multiple="true"
		/>
	);
}

radio-input

Radio Input

Radio input allows a single choice between many propositions.

Associated with Alpine interop, you can intercept the radio valuewith x-on:change / x-on:input listener.

Copied !

import { RadioInput } from "$components/input.component";

export function RadioInputExample() {
	return (
		<div class="flex gap-x-4">
			<div class="flex items-center gap-x-2">
				<label for="radio-1">Radio 1</label>
				<RadioInput
					x-on:change={`console.log($event.target.id, $event.target.checked)`}
					name="radio"
					value="radio-1"
					id="radio-1"
				/>
			</div>
			<div class="flex items-center gap-x-2">
				<label for="radio-2">Radio 2</label>
				<RadioInput
					x-on:change={`console.log($event.target.id, $event.target.checked)`}
					name="radio"
					value="radio-2"
					id="radio-2"
				/>
			</div>
			<div class="flex items-center gap-x-2">
				<label for="radio-3">Radio 3</label>
				<RadioInput
					x-on:change={`console.log($event.target.id, $event.target.checked)`}
					name="radio"
					value="radio-3"
					id="radio-3"
				/>
			</div>
		</div>
	);
}

checkbox-input

Checkbox Input

Checkbox input in the other hands, allows you to select as many choices between propositions.

Associated with Alpine interop, you can intercept the value with x-on:change / x-on:input listener.

Copied !

import { CheckboxInput } from "$components/input.component";

export function CheckboxInputExample() {
	return (
		<div class="flex gap-x-4">
			<div class="flex items-center gap-x-2">
				<label for="checkbox-1">Checkbox 1</label>
				<CheckboxInput
					x-on:change={`console.log($event.target.id, $event.target.checked)`}
					name="checkbox-1"
					value="checkbox-1"
					id="checkbox-1"
				/>
			</div>
			<div class="flex items-center gap-x-2">
				<label for="checkbox-2">Checkbox 2</label>
				<CheckboxInput
					x-on:change={`console.log($event.target.id, $event.target.checked)`}
					name="checkbox-2"
					value="checkbox-2"
					id="checkbox-2"
				/>
			</div>
			<div class="flex items-center gap-x-2">
				<label for="checkbox-3">Checkbox 3</label>
				<CheckboxInput
					x-on:change={`console.log($event.target.id, $event.target.checked)`}
					name="checkbox-3"
					value="checkbox-3"
					id="checkbox-3"
				/>
			</div>
		</div>
	);
}
ComponentsImagePicker
ComponentsModals