forked from coracle/flotilla
111 lines
3.5 KiB
Svelte
111 lines
3.5 KiB
Svelte
<script lang="ts">
|
|
import {onMount} from "svelte"
|
|
import {page} from "$app/stores"
|
|
import {sleep} from "@welshman/lib"
|
|
import type {MakeNonOptional} from "@welshman/lib"
|
|
import {COMMENT, getTagValue} from "@welshman/util"
|
|
import {repository} from "@welshman/app"
|
|
import {request} from "@welshman/net"
|
|
import {deriveEventsById, deriveEventsAsc} from "@welshman/store"
|
|
import SortVertical from "@assets/icons/sort-vertical.svg?dataurl"
|
|
import Reply from "@assets/icons/reply-2.svg?dataurl"
|
|
import Icon from "@lib/components/Icon.svelte"
|
|
import PageContent from "@lib/components/PageContent.svelte"
|
|
import Spinner from "@lib/components/Spinner.svelte"
|
|
import Button from "@lib/components/Button.svelte"
|
|
import SpaceBar from "@app/components/SpaceBar.svelte"
|
|
import NoteContent from "@app/components/NoteContent.svelte"
|
|
import NoteCard from "@app/components/NoteCard.svelte"
|
|
import ThreadActions from "@app/components/ThreadActions.svelte"
|
|
import CommentActions from "@app/components/CommentActions.svelte"
|
|
import EventReply from "@app/components/EventReply.svelte"
|
|
import {deriveEvent} from "@app/repository"
|
|
import {decodeRelay} from "@app/relays"
|
|
|
|
const {relay, id} = $page.params as MakeNonOptional<typeof $page.params>
|
|
const url = decodeRelay(relay)
|
|
const event = deriveEvent(id, [url])
|
|
const filters = [{kinds: [COMMENT], "#E": [id]}]
|
|
const replies = deriveEventsAsc(deriveEventsById({filters, repository}))
|
|
|
|
const back = () => history.back()
|
|
|
|
const openReply = () => {
|
|
showReply = true
|
|
}
|
|
|
|
const closeReply = () => {
|
|
showReply = false
|
|
}
|
|
|
|
const expand = () => {
|
|
showAll = true
|
|
}
|
|
|
|
let showAll = $state(false)
|
|
let showReply = $state(false)
|
|
|
|
onMount(() => {
|
|
const controller = new AbortController()
|
|
|
|
request({relays: [url], filters, signal: controller.signal})
|
|
|
|
return () => {
|
|
controller.abort()
|
|
}
|
|
})
|
|
</script>
|
|
|
|
<SpaceBar {back}>
|
|
{#snippet title()}
|
|
<h1 class="text-xl">{getTagValue("title", $event?.tags || []) || ""}</h1>
|
|
{/snippet}
|
|
</SpaceBar>
|
|
|
|
<PageContent class="flex flex-col gap-2 p-2">
|
|
{#if $event}
|
|
<div class="flex flex-col gap-3">
|
|
<NoteCard event={$event} {url} class="card2 bg-alt z-feature w-full">
|
|
<div class="col-3 ml-12">
|
|
<NoteContent showEntire event={$event} {url} />
|
|
<ThreadActions showRoom event={$event} {url} />
|
|
</div>
|
|
</NoteCard>
|
|
{#if !showAll && $replies.length > 4}
|
|
<div class="flex justify-center">
|
|
<Button class="btn btn-link" onclick={expand}>
|
|
<Icon icon={SortVertical} />
|
|
Show all {$replies.length} replies
|
|
</Button>
|
|
</div>
|
|
{/if}
|
|
{#each $replies.slice(0, showAll ? undefined : 4) as reply (reply.id)}
|
|
<NoteCard event={reply} {url} class="card2 bg-alt z-feature w-full">
|
|
<div class="col-3 ml-12">
|
|
<NoteContent showEntire event={reply} {url} />
|
|
<CommentActions segment="threads" event={reply} {url} />
|
|
</div>
|
|
</NoteCard>
|
|
{/each}
|
|
</div>
|
|
{#if showReply}
|
|
<EventReply {url} event={$event} onClose={closeReply} onSubmit={closeReply} />
|
|
{:else}
|
|
<div class="flex justify-end">
|
|
<Button class="btn btn-primary" onclick={openReply}>
|
|
<Icon icon={Reply} />
|
|
Reply to thread
|
|
</Button>
|
|
</div>
|
|
{/if}
|
|
{:else}
|
|
<div class="flex justify-center py-20">
|
|
{#await sleep(5000)}
|
|
<Spinner loading>Loading thread...</Spinner>
|
|
{:then}
|
|
<p>Failed to load thread.</p>
|
|
{/await}
|
|
</div>
|
|
{/if}
|
|
</PageContent>
|