mirror of
https://github.com/khoaliber/dockhand.git
synced 2026-03-03 13:18:56 +00:00
Initial commit
This commit is contained in:
93
lib/components/MultiSelectFilter.svelte
Normal file
93
lib/components/MultiSelectFilter.svelte
Normal file
@@ -0,0 +1,93 @@
|
||||
<script lang="ts">
|
||||
import * as Select from '$lib/components/ui/select';
|
||||
import type { Component } from 'svelte';
|
||||
|
||||
interface FilterOption {
|
||||
value: string;
|
||||
label: string;
|
||||
icon?: Component;
|
||||
color?: string;
|
||||
}
|
||||
|
||||
interface Props {
|
||||
value: string[];
|
||||
options: FilterOption[];
|
||||
placeholder: string;
|
||||
pluralLabel?: string;
|
||||
width?: string;
|
||||
defaultIcon?: Component;
|
||||
}
|
||||
|
||||
let {
|
||||
value = $bindable([]),
|
||||
options,
|
||||
placeholder,
|
||||
pluralLabel,
|
||||
width = 'w-36',
|
||||
defaultIcon
|
||||
}: Props = $props();
|
||||
|
||||
// Control dropdown open state
|
||||
let open = $state(false);
|
||||
|
||||
// Check if any options have icons
|
||||
const hasIcons = $derived(options.some(o => o.icon));
|
||||
|
||||
// Get the icon for single selection
|
||||
const singleOption = $derived(() => {
|
||||
if (value.length === 1) {
|
||||
return options.find(o => o.value === value[0]);
|
||||
}
|
||||
return null;
|
||||
});
|
||||
|
||||
const displayLabel = $derived(() => {
|
||||
if (value.length === 0) {
|
||||
return placeholder;
|
||||
} else if (value.length === 1) {
|
||||
const opt = options.find(o => o.value === value[0]);
|
||||
return opt?.label || value[0];
|
||||
} else {
|
||||
return `${value.length} ${pluralLabel || placeholder.toLowerCase()}`;
|
||||
}
|
||||
});
|
||||
|
||||
function clearAndClose() {
|
||||
value = [];
|
||||
open = false;
|
||||
}
|
||||
</script>
|
||||
|
||||
<Select.Root type="multiple" bind:value bind:open>
|
||||
<Select.Trigger size="sm" class="{width} text-sm">
|
||||
{#if hasIcons || defaultIcon}
|
||||
{@const opt = singleOption()}
|
||||
{@const IconComponent = opt?.icon || defaultIcon}
|
||||
{#if IconComponent}
|
||||
<svelte:component this={IconComponent} class="w-3.5 h-3.5 mr-1.5 {opt?.color || 'text-muted-foreground'} shrink-0" />
|
||||
{/if}
|
||||
{/if}
|
||||
<span class="{value.length === 0 ? 'text-muted-foreground' : ''}">
|
||||
{displayLabel()}
|
||||
</span>
|
||||
</Select.Trigger>
|
||||
<Select.Content>
|
||||
{#if value.length > 0}
|
||||
<button
|
||||
type="button"
|
||||
class="w-full px-2 py-1 text-xs text-left text-muted-foreground/60 hover:text-muted-foreground"
|
||||
onclick={clearAndClose}
|
||||
>
|
||||
Clear
|
||||
</button>
|
||||
{/if}
|
||||
{#each options as option}
|
||||
<Select.Item value={option.value}>
|
||||
{#if option.icon}
|
||||
<svelte:component this={option.icon} class="w-4 h-4 mr-2 {option.color || ''}" />
|
||||
{/if}
|
||||
{option.label}
|
||||
</Select.Item>
|
||||
{/each}
|
||||
</Select.Content>
|
||||
</Select.Root>
|
||||
Reference in New Issue
Block a user