| <script lang="ts"> |
| import { fade } from "svelte/transition"; |
| import IconChevron from "./icons/IconChevron.svelte"; |
|
|
| interface Props { |
| scrollNode: HTMLElement; |
| class?: string; |
| } |
|
|
| let { scrollNode, class: className = "" }: Props = $props(); |
|
|
| let visible = $state(false); |
| let observer: ResizeObserver | null = $state(null); |
|
|
| function updateVisibility() { |
| if (!scrollNode) return; |
| visible = |
| Math.ceil(scrollNode.scrollTop) + 200 < scrollNode.scrollHeight - scrollNode.clientHeight; |
| } |
|
|
| function destroy() { |
| observer?.disconnect(); |
| scrollNode?.removeEventListener("scroll", updateVisibility); |
| } |
| const cleanup = $effect.root(() => { |
| $effect(() => { |
| if (scrollNode) { |
| if (window.ResizeObserver) { |
| observer = new ResizeObserver(() => updateVisibility()); |
| observer.observe(scrollNode); |
| cleanup(); |
| } |
| scrollNode?.addEventListener("scroll", updateVisibility); |
| } |
| }); |
| return () => destroy(); |
| }); |
| </script> |
|
|
| {#if visible} |
| <button |
| transition:fade={{ duration: 150 }} |
| onclick={() => scrollNode.scrollTo({ top: scrollNode.scrollHeight, behavior: "smooth" })} |
| class="btn absolute flex h-[41px] w-[41px] rounded-full border bg-white shadow-md transition-all hover:bg-gray-100 dark:border-gray-600 dark:bg-gray-700 dark:shadow-gray-950 dark:hover:bg-gray-600 {className}" |
| ><IconChevron classNames="mt-[2px]" /></button |
| > |
| {/if} |
| |