src/lib/components/ScrollToBottomBtn.svelte (41 lines of code) (raw):

<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}