Files
flotilla/src/app/components/ProfileMultiSelect.svelte
T
2024-10-11 16:18:05 -07:00

90 lines
2.4 KiB
Svelte

<script lang="ts">
import type {SvelteComponent} from "svelte"
import {type Instance} from "tippy.js"
import {append, remove, uniq} from "@welshman/lib"
import {profileSearch} from "@welshman/app"
import Icon from "@lib/components/Icon.svelte"
import Tippy from "@lib/components/Tippy.svelte"
import Button from "@lib/components/Button.svelte"
import Suggestions from "@lib/editor/Suggestions.svelte"
import SuggestionProfile from "@lib/editor/SuggestionProfile.svelte"
import ProfileName from "@app/components/ProfileName.svelte"
import ProfileDetail from "@app/components/ProfileDetail.svelte"
import {pushDrawer} from "@app/modal"
export let value: string[]
let term = ""
let input: Element
let popover: Instance
let instance: SvelteComponent
const onClick = (pubkey: string) => pushDrawer(ProfileDetail, {pubkey})
const selectPubkey = (pubkey: string) => {
term = ""
popover.hide()
value = uniq(append(pubkey, value))
}
const removePubkey = (pubkey: string) => {
value = remove(pubkey, value)
}
const onKeyDown = (e: Event) => {
if (instance.onKeyDown(e)) {
e.preventDefault()
}
}
$: {
if (term) {
popover?.show()
} else {
popover?.hide()
}
}
</script>
<div class="flex flex-col gap-2">
<div>
{#each value as pubkey (pubkey)}
<div class="flex-inline badge badge-neutral mr-1 gap-1">
<Button class="flex items-center" on:click={() => removePubkey(pubkey)}>
<Icon icon="close-circle" size={4} class="-ml-1 mt-px" />
</Button>
<Button on:click={() => onClick(pubkey)}>
<ProfileName {pubkey} />
</Button>
</div>
{/each}
</div>
<label class="input input-bordered flex w-full items-center gap-2" bind:this={input}>
<Icon icon="magnifer" />
<input
class="grow"
type="text"
placeholder="Search for profiles..."
bind:value={term}
on:keydown={onKeyDown} />
</label>
<Tippy
bind:popover
bind:instance
component={Suggestions}
props={{
term,
select: selectPubkey,
search: profileSearch,
component: SuggestionProfile,
class: "rounded-box",
style: `left: 4px; width: ${input?.clientWidth + 12}px`,
}}
params={{
trigger: "manual",
interactive: true,
maxWidth: "none",
getReferenceClientRect: () => input.getBoundingClientRect(),
}} />
</div>