Files
profilarr/src/lib/client/ui/form/NumberInput.svelte

149 lines
4.6 KiB
Svelte

<script lang="ts">
import { onMount, onDestroy, createEventDispatcher } from 'svelte';
import { ChevronUp, ChevronDown } from 'lucide-svelte';
const dispatch = createEventDispatcher<{ change: number | undefined }>();
// Props
export let name: string;
export let id: string = name;
export let value: number | undefined = undefined;
export let min: number | undefined = undefined;
export let max: number | undefined = undefined;
export let step: number = 1;
export let required: boolean = false;
export let disabled: boolean = false;
export let placeholder: string = '';
export let font: 'mono' | 'sans' | undefined = undefined;
export let compact: boolean = false;
// Responsive: auto-switch to compact on smaller screens (< 1280px)
export let responsive: boolean = false;
export let onchange: ((value: number) => void) | undefined = undefined;
export let onMinBlocked: (() => void) | undefined = undefined;
export let onMaxBlocked: (() => void) | undefined = undefined;
let isSmallScreen = false;
let mediaQuery: MediaQueryList | null = null;
onMount(() => {
if (responsive && typeof window !== 'undefined') {
mediaQuery = window.matchMedia('(max-width: 1279px)');
isSmallScreen = mediaQuery.matches;
mediaQuery.addEventListener('change', handleMediaChange);
}
});
onDestroy(() => {
if (mediaQuery) {
mediaQuery.removeEventListener('change', handleMediaChange);
}
});
function handleMediaChange(e: MediaQueryListEvent) {
isSmallScreen = e.matches;
}
$: isCompact = compact || (responsive && isSmallScreen);
$: hideButtons = responsive && isSmallScreen;
$: fontClass = font === 'mono' ? 'font-mono' : font === 'sans' ? 'font-sans' : '';
$: inputSizeClasses = isCompact
? hideButtons
? 'rounded px-2 py-1 text-xs'
: 'rounded px-2 py-1 pr-7 text-xs'
: 'rounded-lg px-3 py-2 pr-10';
$: buttonSizeClasses = isCompact ? 'h-2.5 w-4' : 'h-4 w-6';
$: iconSize = isCompact ? 10 : 12;
function updateValue(newValue: number) {
value = newValue;
onchange?.(newValue);
dispatch('change', newValue);
}
// Increment/decrement handlers
function increment() {
const currentValue = value ?? min ?? 0;
if (max !== undefined && currentValue >= max) {
onMaxBlocked?.();
return;
}
updateValue(currentValue + step);
}
function decrement() {
const currentValue = value ?? min ?? 0;
if (min !== undefined && currentValue <= min) {
onMinBlocked?.();
return;
}
updateValue(currentValue - step);
}
// Validate on input
function handleInput(event: Event) {
const target = event.target as HTMLInputElement;
// Allow clearing the input (show placeholder)
if (target.value === '') {
value = undefined;
dispatch('change', undefined);
return;
}
let newValue = parseFloat(target.value);
if (isNaN(newValue)) {
return; // Don't update for invalid input
}
if (min !== undefined && newValue < min) {
newValue = min;
}
if (max !== undefined && newValue > max) {
newValue = max;
}
updateValue(newValue);
}
</script>
<div class="relative">
<input
type="number"
{id}
{name}
bind:value
on:input={handleInput}
{min}
{max}
{step}
{required}
{disabled}
{placeholder}
class="block w-full [appearance:textfield] border border-neutral-300 bg-white text-neutral-900 placeholder-neutral-400 focus:outline-none disabled:bg-neutral-100 disabled:text-neutral-500 dark:border-neutral-700 dark:bg-neutral-800 dark:text-neutral-50 dark:placeholder-neutral-500 dark:disabled:bg-neutral-900 dark:disabled:text-neutral-600 [&::-webkit-inner-spin-button]:appearance-none [&::-webkit-outer-spin-button]:appearance-none {inputSizeClasses} {fontClass}"
/>
<!-- Custom increment/decrement buttons (hidden on mobile when responsive) -->
{#if !hideButtons}
<div class="absolute top-1/2 right-1 flex -translate-y-1/2 flex-col">
<button
type="button"
on:click={increment}
{disabled}
class="flex {buttonSizeClasses} items-center justify-center rounded-t border border-neutral-300 bg-white text-neutral-600 transition-colors hover:bg-neutral-50 disabled:cursor-not-allowed disabled:opacity-40 dark:border-neutral-600 dark:bg-neutral-700 dark:text-neutral-300 dark:hover:bg-neutral-600"
>
<ChevronUp size={iconSize} />
</button>
<button
type="button"
on:click={decrement}
{disabled}
class="flex {buttonSizeClasses} items-center justify-center rounded-b border border-t-0 border-neutral-300 bg-white text-neutral-600 transition-colors hover:bg-neutral-50 disabled:cursor-not-allowed disabled:opacity-40 dark:border-neutral-600 dark:bg-neutral-700 dark:text-neutral-300 dark:hover:bg-neutral-600"
>
<ChevronDown size={iconSize} />
</button>
</div>
{/if}
</div>