Inputs
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 !
-------------------- 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>
);
}