Prevent icon picker from going off screen

This commit is contained in:
Jon Staab
2026-01-29 11:11:08 -08:00
parent e74f922e8d
commit a3c1a5c731
5 changed files with 57 additions and 20 deletions
+3 -4
View File
@@ -43,7 +43,6 @@
} }
</script> </script>
<div class="w-96 rounded-box bg-base-100 p-4 shadow-2xl">
<label class="input input-bordered flex w-full items-center gap-2"> <label class="input input-bordered flex w-full items-center gap-2">
<Icon icon={Magnifier} /> <Icon icon={Magnifier} />
<input bind:value={searchTerm} class="grow" type="text" placeholder="Search icons..." /> <input bind:value={searchTerm} class="grow" type="text" placeholder="Search icons..." />
@@ -52,12 +51,12 @@
<div class="grid grid-cols-8 gap-2 p-2"> <div class="grid grid-cols-8 gap-2 p-2">
{#each filteredIcons as icon} {#each filteredIcons as icon}
<button <button
type="button"
title={icon.name}
class="flex aspect-square items-center justify-center rounded-box transition-colors hover:bg-primary hover:text-primary-content" class="flex aspect-square items-center justify-center rounded-box transition-colors hover:bg-primary hover:text-primary-content"
onclick={() => handleSelect(icon.url)} onclick={() => handleSelect(icon.url)}>
title={icon.name}>
<Icon icon={icon.url} class="h-6 w-6" /> <Icon icon={icon.url} class="h-6 w-6" />
</button> </button>
{/each} {/each}
</div> </div>
</div> </div>
</div>
@@ -0,0 +1,13 @@
<script lang="ts">
import IconPicker from "@app/components/IconPicker.svelte"
type Props = {
onSelect: (iconUrl: string) => void
}
const {onSelect}: Props = $props()
</script>
<div class="w-96 rounded-box bg-base-100 p-4 shadow-2xl">
<IconPicker {onSelect} />
</div>
+1
View File
@@ -31,6 +31,7 @@
<div class="center fixed inset-0 z-modal"> <div class="center fixed inset-0 z-modal">
<button <button
type="button"
aria-label="Close dialog" aria-label="Close dialog"
class="absolute inset-0 cursor-pointer bg-[#ccc] opacity-75 dark:bg-black" class="absolute inset-0 cursor-pointer bg-[#ccc] opacity-75 dark:bg-black"
transition:fade={{duration: 300}} transition:fade={{duration: 300}}
+1 -1
View File
@@ -9,7 +9,7 @@
const {...props}: Props = $props() const {...props}: Props = $props()
</script> </script>
<div class="grid grid-cols-1 gap-6 lg:grid-cols-3 {props.class}"> <div class="grid grid-cols-1 gap-2 lg:gap-6 lg:grid-cols-3 {props.class}">
<label class="flex items-center gap-2 font-bold"> <label class="flex items-center gap-2 font-bold">
{@render props.label?.()} {@render props.label?.()}
</label> </label>
+27 -3
View File
@@ -1,17 +1,34 @@
<script lang="ts"> <script lang="ts">
import {type Instance} from "tippy.js" import {type Instance} from "tippy.js"
import {between, throttle} from "@welshman/lib" import {between, throttle} from "@welshman/lib"
import {isMobile} from "@lib/html"
import Button from "@lib/components/Button.svelte" import Button from "@lib/components/Button.svelte"
import Dialog from "@lib/components/Dialog.svelte"
import Tippy from "@lib/components/Tippy.svelte" import Tippy from "@lib/components/Tippy.svelte"
import IconPicker from "@app/components/IconPicker.svelte" import IconPicker from "@app/components/IconPicker.svelte"
import IconPickerDialog from "@app/components/IconPickerDialog.svelte"
const {...props} = $props() const {...props} = $props()
const open = () => popover?.show() const open = () => {
if (isMobile) {
showIconPicker = true
} else {
popover?.show()
}
}
const close = () => {
if (isMobile) {
showIconPicker = false
} else {
popover?.hide()
}
}
const onClick = (iconUrl: string) => { const onClick = (iconUrl: string) => {
props.onSelect(iconUrl) props.onSelect(iconUrl)
popover?.hide() close()
} }
const onMouseMove = throttle(300, ({clientX, clientY}: any) => { const onMouseMove = throttle(300, ({clientX, clientY}: any) => {
@@ -24,6 +41,7 @@
} }
}) })
let showIconPicker = $state(false)
let popover: Instance | undefined = $state() let popover: Instance | undefined = $state()
</script> </script>
@@ -31,10 +49,16 @@
<Tippy <Tippy
bind:popover bind:popover
component={IconPicker} component={IconPickerDialog}
props={{onSelect: onClick}} props={{onSelect: onClick}}
params={{trigger: "manual", interactive: true}}> params={{trigger: "manual", interactive: true}}>
<Button onclick={open} class={props.class}> <Button onclick={open} class={props.class}>
{@render props.children?.()} {@render props.children?.()}
</Button> </Button>
</Tippy> </Tippy>
{#if showIconPicker}
<Dialog onClose={close}>
<IconPicker onSelect={onClick} />
</Dialog>
{/if}