mirror of
https://github.com/coracle-social/flotilla.git
synced 2025-12-10 19:07:06 +00:00
Re-work invite links
This commit is contained in:
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
const {url} = $props()
|
const {url} = $props()
|
||||||
|
|
||||||
const display = deriveRelayDisplay(url)
|
const display = $derived(deriveRelayDisplay(url))
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{$display}
|
{$display}
|
||||||
|
|||||||
55
src/app/components/RelaySummary.svelte
Normal file
55
src/app/components/RelaySummary.svelte
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import {gt} from "@welshman/lib"
|
||||||
|
import {deriveRelay} from "@welshman/app"
|
||||||
|
import Icon from "@lib/components/Icon.svelte"
|
||||||
|
import RelayName from "@app/components/RelayName.svelte"
|
||||||
|
import RelayDescription from "@app/components/RelayDescription.svelte"
|
||||||
|
import ProfileCircles from "@app/components/ProfileCircles.svelte"
|
||||||
|
import {membersByUrl, userRoomsByUrl} from "@app/core/state"
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
url: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const {url}: Props = $props()
|
||||||
|
const relay = deriveRelay(url)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="col-4 text-left">
|
||||||
|
<div class="col-2">
|
||||||
|
<div class="relative flex gap-4">
|
||||||
|
<div class="relative">
|
||||||
|
<div class="avatar relative">
|
||||||
|
<div
|
||||||
|
class="center !flex h-12 w-12 min-w-12 rounded-full border-2 border-solid border-base-300 bg-base-300">
|
||||||
|
{#if $relay?.profile?.icon}
|
||||||
|
<img alt="" src={$relay.profile.icon} />
|
||||||
|
{:else}
|
||||||
|
<Icon icon="ghost" size={5} />
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{#if $userRoomsByUrl.has(url)}
|
||||||
|
<div
|
||||||
|
class="tooltip absolute -right-1 -top-1 h-5 w-5 rounded-full bg-primary"
|
||||||
|
data-tip="You are already a member of this space.">
|
||||||
|
<Icon icon="check-circle" class="scale-110" />
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<h2 class="ellipsize whitespace-nowrap text-xl">
|
||||||
|
<RelayName {url} />
|
||||||
|
</h2>
|
||||||
|
<p class="text-sm opacity-75">{url}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<RelayDescription {url} />
|
||||||
|
</div>
|
||||||
|
{#if gt($membersByUrl.get(url)?.size, 0)}
|
||||||
|
<div class="row-2 card2 card2-sm bg-alt">
|
||||||
|
Members:
|
||||||
|
<ProfileCircles pubkeys={Array.from($membersByUrl.get(url) || [])} />
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import {onMount} from "svelte"
|
import {onMount} from "svelte"
|
||||||
import {sleep, identity, nthEq} from "@welshman/lib"
|
import {sleep, nthEq} from "@welshman/lib"
|
||||||
import {request} from "@welshman/net"
|
import {request} from "@welshman/net"
|
||||||
import {displayRelayUrl, AUTH_INVITE} from "@welshman/util"
|
import {displayRelayUrl, AUTH_INVITE} from "@welshman/util"
|
||||||
import {slide} from "@lib/transition"
|
import {slide} from "@lib/transition"
|
||||||
@@ -11,6 +11,7 @@
|
|||||||
import ModalHeader from "@lib/components/ModalHeader.svelte"
|
import ModalHeader from "@lib/components/ModalHeader.svelte"
|
||||||
import ModalFooter from "@lib/components/ModalFooter.svelte"
|
import ModalFooter from "@lib/components/ModalFooter.svelte"
|
||||||
import {clip} from "@app/util/toast"
|
import {clip} from "@app/util/toast"
|
||||||
|
import {PLATFORM_URL} from "@app/core/state"
|
||||||
|
|
||||||
const {url} = $props()
|
const {url} = $props()
|
||||||
|
|
||||||
@@ -24,7 +25,10 @@
|
|||||||
let invite = $state("")
|
let invite = $state("")
|
||||||
|
|
||||||
$effect(() => {
|
$effect(() => {
|
||||||
invite = [displayRelayUrl(url), claim].filter(identity).join("|")
|
const relay = displayRelayUrl(url)
|
||||||
|
const params = new URLSearchParams({r: relay, c: claim}).toString()
|
||||||
|
|
||||||
|
invite = PLATFORM_URL + "/join?" + params
|
||||||
})
|
})
|
||||||
|
|
||||||
onMount(async () => {
|
onMount(async () => {
|
||||||
@@ -57,7 +61,7 @@
|
|||||||
<div>
|
<div>
|
||||||
{#if loading}
|
{#if loading}
|
||||||
<p class="center" out:slide>
|
<p class="center" out:slide>
|
||||||
<Spinner {loading}>Requesting an invite code...</Spinner>
|
<Spinner {loading}>Requesting an invite link...</Spinner>
|
||||||
</p>
|
</p>
|
||||||
{:else}
|
{:else}
|
||||||
<div in:slide>
|
<div in:slide>
|
||||||
|
|||||||
@@ -1,52 +1,44 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import {tryCatch, first, removeNil} from "@welshman/lib"
|
import type {Snippet} from "svelte"
|
||||||
|
import {tryCatch, fromPairs} from "@welshman/lib"
|
||||||
import {isRelayUrl, normalizeRelayUrl} from "@welshman/util"
|
import {isRelayUrl, normalizeRelayUrl} from "@welshman/util"
|
||||||
import {Pool, AuthStatus} from "@welshman/net"
|
import {Pool, AuthStatus} from "@welshman/net"
|
||||||
import {preventDefault} from "@lib/html"
|
import {preventDefault} from "@lib/html"
|
||||||
|
import {slideAndFade} from "@lib/transition"
|
||||||
import Spinner from "@lib/components/Spinner.svelte"
|
import Spinner from "@lib/components/Spinner.svelte"
|
||||||
import Button from "@lib/components/Button.svelte"
|
import Button from "@lib/components/Button.svelte"
|
||||||
import Field from "@lib/components/Field.svelte"
|
import Field from "@lib/components/Field.svelte"
|
||||||
import Icon from "@lib/components/Icon.svelte"
|
import Icon from "@lib/components/Icon.svelte"
|
||||||
import ModalHeader from "@lib/components/ModalHeader.svelte"
|
import ModalHeader from "@lib/components/ModalHeader.svelte"
|
||||||
import ModalFooter from "@lib/components/ModalFooter.svelte"
|
import ModalFooter from "@lib/components/ModalFooter.svelte"
|
||||||
import InfoRelay from "@app/components/InfoRelay.svelte"
|
import RelaySummary from "@app/components/RelaySummary.svelte"
|
||||||
import SpaceJoinConfirm, {confirmSpaceJoin} from "@app/components/SpaceJoinConfirm.svelte"
|
import SpaceJoinConfirm, {confirmSpaceJoin} from "@app/components/SpaceJoinConfirm.svelte"
|
||||||
import {pushToast} from "@app/util/toast"
|
import {pushToast} from "@app/util/toast"
|
||||||
import {pushModal} from "@app/util/modal"
|
import {pushModal} from "@app/util/modal"
|
||||||
import {attemptRelayAccess} from "@app/core/commands"
|
import {attemptRelayAccess} from "@app/core/commands"
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
invite: string
|
||||||
|
abortAction?: Snippet
|
||||||
|
}
|
||||||
|
|
||||||
|
let {invite = "", abortAction}: Props = $props()
|
||||||
|
|
||||||
const back = () => history.back()
|
const back = () => history.back()
|
||||||
|
|
||||||
const joinRelay = async () => {
|
const joinRelay = async () => {
|
||||||
const promises: Promise<string | undefined>[] = []
|
const {url, claim} = inviteData!
|
||||||
|
|
||||||
const [rawUrl, rawClaim] = url.split("|")
|
const error = await attemptRelayAccess(url, claim)
|
||||||
const normalizedUrl = normalizeRelayUrl(rawUrl)
|
|
||||||
|
|
||||||
if (claim) {
|
|
||||||
promises.push(attemptRelayAccess(normalizedUrl, claim))
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rawClaim) {
|
|
||||||
promises.push(attemptRelayAccess(normalizedUrl, rawClaim))
|
|
||||||
}
|
|
||||||
|
|
||||||
if (promises.length === 0) {
|
|
||||||
promises.push(attemptRelayAccess(normalizedUrl, ""))
|
|
||||||
}
|
|
||||||
|
|
||||||
const error = first(removeNil(await Promise.all(promises)))
|
|
||||||
|
|
||||||
if (error) {
|
if (error) {
|
||||||
return pushToast({theme: "error", message: error, timeout: 30_000})
|
return pushToast({theme: "error", message: error, timeout: 30_000})
|
||||||
}
|
}
|
||||||
|
|
||||||
const socket = Pool.get().get(normalizedUrl)
|
if (Pool.get().get(url).auth.status === AuthStatus.None) {
|
||||||
|
pushModal(SpaceJoinConfirm, {url}, {replaceState: true})
|
||||||
if (socket.auth.status === AuthStatus.None) {
|
|
||||||
pushModal(SpaceJoinConfirm, {url: normalizedUrl}, {replaceState: true})
|
|
||||||
} else {
|
} else {
|
||||||
await confirmSpaceJoin(normalizedUrl)
|
await confirmSpaceJoin(url)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,12 +52,25 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let url = $state("")
|
|
||||||
let claim = $state("")
|
|
||||||
let loading = $state(false)
|
let loading = $state(false)
|
||||||
|
|
||||||
const linkIsValid = $derived(
|
const inviteData = $derived.by(
|
||||||
Boolean(tryCatch(() => isRelayUrl(normalizeRelayUrl(url.split("|")[0])))),
|
() =>
|
||||||
|
tryCatch(() => {
|
||||||
|
const {r: relay = "", c: claim = ""} = fromPairs(Array.from(new URL(invite).searchParams))
|
||||||
|
const url = normalizeRelayUrl(relay)
|
||||||
|
|
||||||
|
if (isRelayUrl(url)) {
|
||||||
|
return {url, claim}
|
||||||
|
}
|
||||||
|
}) ||
|
||||||
|
tryCatch(() => {
|
||||||
|
const url = normalizeRelayUrl(invite)
|
||||||
|
|
||||||
|
if (isRelayUrl(url)) {
|
||||||
|
return {url, claim: ""}
|
||||||
|
}
|
||||||
|
}),
|
||||||
)
|
)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -75,46 +80,40 @@
|
|||||||
<div>Join a Space</div>
|
<div>Join a Space</div>
|
||||||
{/snippet}
|
{/snippet}
|
||||||
{#snippet info()}
|
{#snippet info()}
|
||||||
<div>Enter a relay URL below to join an existing space.</div>
|
<div>Enter a relay URL or invite link below to join an existing space.</div>
|
||||||
{/snippet}
|
{/snippet}
|
||||||
</ModalHeader>
|
</ModalHeader>
|
||||||
<Field>
|
<Field>
|
||||||
{#snippet label()}
|
{#snippet label()}
|
||||||
<p>Relay URL*</p>
|
<p>Invite Link*</p>
|
||||||
{/snippet}
|
{/snippet}
|
||||||
{#snippet input()}
|
{#snippet input()}
|
||||||
<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="link-round" />
|
<Icon icon="link-round" />
|
||||||
<input bind:value={url} class="grow" type="text" />
|
<input bind:value={invite} class="grow" type="text" />
|
||||||
</label>
|
</label>
|
||||||
{/snippet}
|
{/snippet}
|
||||||
{#snippet info()}
|
|
||||||
<p>
|
|
||||||
Enter the URL of the relay that hosts the space you'd like to join.
|
|
||||||
<Button class="link" onclick={() => pushModal(InfoRelay)}>What is a relay?</Button>
|
|
||||||
</p>
|
|
||||||
{/snippet}
|
|
||||||
</Field>
|
|
||||||
<Field>
|
|
||||||
{#snippet label()}
|
|
||||||
<p>Invite Code (optional)</p>
|
|
||||||
{/snippet}
|
|
||||||
{#snippet input()}
|
|
||||||
<label class="input input-bordered flex w-full items-center gap-2">
|
|
||||||
<Icon icon="ticket" />
|
|
||||||
<input bind:value={claim} class="grow" type="text" />
|
|
||||||
</label>
|
|
||||||
{/snippet}
|
|
||||||
{#snippet info()}
|
|
||||||
<p>If you have an invite code, enter it here to get access.</p>
|
|
||||||
{/snippet}
|
|
||||||
</Field>
|
</Field>
|
||||||
|
<div class="-my-4">
|
||||||
|
{#if inviteData}
|
||||||
|
<div transition:slideAndFade class="flex flex-col gap-4 py-4">
|
||||||
|
<div class="card2 bg-alt flex flex-col gap-4">
|
||||||
|
<p class="opacity-75">You're about to join:</p>
|
||||||
|
<RelaySummary url={inviteData.url} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
<ModalFooter>
|
<ModalFooter>
|
||||||
|
{#if abortAction}
|
||||||
|
{@render abortAction?.()}
|
||||||
|
{:else}
|
||||||
<Button class="btn btn-link" onclick={back}>
|
<Button class="btn btn-link" onclick={back}>
|
||||||
<Icon icon="alt-arrow-left" />
|
<Icon icon="alt-arrow-left" />
|
||||||
Go back
|
Go back
|
||||||
</Button>
|
</Button>
|
||||||
<Button type="submit" class="btn btn-primary" disabled={!linkIsValid || loading}>
|
{/if}
|
||||||
|
<Button type="submit" class="btn btn-primary" disabled={!inviteData || loading}>
|
||||||
<Spinner {loading}>Join Space</Spinner>
|
<Spinner {loading}>Join Space</Spinner>
|
||||||
<Icon icon="alt-arrow-right" />
|
<Icon icon="alt-arrow-right" />
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ import {daysBetween} from "@lib/util"
|
|||||||
import {
|
import {
|
||||||
NOTIFIER_RELAY,
|
NOTIFIER_RELAY,
|
||||||
INDEXER_RELAYS,
|
INDEXER_RELAYS,
|
||||||
getDefaultPubkeys,
|
defaultPubkeys,
|
||||||
userRoomsByUrl,
|
userRoomsByUrl,
|
||||||
getUrlsForEvent,
|
getUrlsForEvent,
|
||||||
loadMembership,
|
loadMembership,
|
||||||
@@ -416,7 +416,7 @@ export const loadUserData = async (pubkey: string, relays: string[] = []) => {
|
|||||||
// Load followed profiles slowly in the background without clogging other stuff up. Only use a single
|
// Load followed profiles slowly in the background without clogging other stuff up. Only use a single
|
||||||
// indexer relay to avoid too many redundant validations, which slow things down and eat bandwidth
|
// indexer relay to avoid too many redundant validations, which slow things down and eat bandwidth
|
||||||
promise.then(async () => {
|
promise.then(async () => {
|
||||||
for (const pubkeys of chunk(50, getDefaultPubkeys())) {
|
for (const pubkeys of chunk(50, get(defaultPubkeys))) {
|
||||||
const relays = sample(1, INDEXER_RELAYS)
|
const relays = sample(1, INDEXER_RELAYS)
|
||||||
|
|
||||||
await sleep(1000)
|
await sleep(1000)
|
||||||
|
|||||||
@@ -191,12 +191,12 @@ export const entityLink = (entity: string) => `https://coracle.social/${entity}`
|
|||||||
export const pubkeyLink = (pubkey: string, relays = Router.get().FromPubkeys([pubkey]).getUrls()) =>
|
export const pubkeyLink = (pubkey: string, relays = Router.get().FromPubkeys([pubkey]).getUrls()) =>
|
||||||
entityLink(nip19.nprofileEncode({pubkey, relays}))
|
entityLink(nip19.nprofileEncode({pubkey, relays}))
|
||||||
|
|
||||||
export const getDefaultPubkeys = () => {
|
export const defaultPubkeys = derived(userFollows, $userFollows => {
|
||||||
const appPubkeys = DEFAULT_PUBKEYS.split(",")
|
const appPubkeys = DEFAULT_PUBKEYS.split(",")
|
||||||
const userPubkeys = shuffle(getPubkeyTagValues(getListTags(get(userFollows))))
|
const userPubkeys = shuffle(getPubkeyTagValues(getListTags($userFollows)))
|
||||||
|
|
||||||
return userPubkeys.length > 5 ? userPubkeys : [...userPubkeys, ...appPubkeys]
|
return userPubkeys.length > 5 ? userPubkeys : [...userPubkeys, ...appPubkeys]
|
||||||
}
|
})
|
||||||
|
|
||||||
const failedUnwraps = new Set()
|
const failedUnwraps = new Set()
|
||||||
|
|
||||||
@@ -459,6 +459,21 @@ export const {
|
|||||||
load: makeOutboxLoader(ROOMS),
|
load: makeOutboxLoader(ROOMS),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
export const membersByUrl = derived(
|
||||||
|
[defaultPubkeys, membershipsByPubkey],
|
||||||
|
([$defaultPubkeys, $membershipsByPubkey]) => {
|
||||||
|
const $membersByUrl = new Map<string, Set<string>>()
|
||||||
|
|
||||||
|
for (const pubkey of $defaultPubkeys) {
|
||||||
|
for (const url of getMembershipUrls($membershipsByPubkey.get(pubkey))) {
|
||||||
|
addToMapKey($membersByUrl, url, pubkey)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $membersByUrl
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
// Chats
|
// Chats
|
||||||
|
|
||||||
export const chatMessages = deriveEvents(repository, {
|
export const chatMessages = deriveEvents(repository, {
|
||||||
|
|||||||
@@ -27,7 +27,7 @@
|
|||||||
</div>
|
</div>
|
||||||
{@render input?.()}
|
{@render input?.()}
|
||||||
{#if info}
|
{#if info}
|
||||||
<p class="text-sm">
|
<p class="text-sm opacity-50">
|
||||||
{@render info()}
|
{@render info()}
|
||||||
</p>
|
</p>
|
||||||
{/if}
|
{/if}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import {onMount} from "svelte"
|
import {onMount} from "svelte"
|
||||||
import {addToMapKey, dec, gt} from "@welshman/lib"
|
import {dec} from "@welshman/lib"
|
||||||
import {ROOMS} from "@welshman/util"
|
import {ROOMS} from "@welshman/util"
|
||||||
import {Router} from "@welshman/router"
|
import {Router} from "@welshman/router"
|
||||||
import {load} from "@welshman/net"
|
import {load} from "@welshman/net"
|
||||||
@@ -13,17 +13,9 @@
|
|||||||
import Spinner from "@lib/components/Spinner.svelte"
|
import Spinner from "@lib/components/Spinner.svelte"
|
||||||
import Button from "@lib/components/Button.svelte"
|
import Button from "@lib/components/Button.svelte"
|
||||||
import PageHeader from "@lib/components/PageHeader.svelte"
|
import PageHeader from "@lib/components/PageHeader.svelte"
|
||||||
import RelayName from "@app/components/RelayName.svelte"
|
import RelaySummary from "@app/components/RelaySummary.svelte"
|
||||||
import RelayDescription from "@app/components/RelayDescription.svelte"
|
|
||||||
import SpaceCheck from "@app/components/SpaceCheck.svelte"
|
import SpaceCheck from "@app/components/SpaceCheck.svelte"
|
||||||
import ProfileCircles from "@app/components/ProfileCircles.svelte"
|
import {getMembershipUrls, loadMembership, defaultPubkeys, membersByUrl} from "@app/core/state"
|
||||||
import {
|
|
||||||
membershipsByPubkey,
|
|
||||||
getMembershipUrls,
|
|
||||||
loadMembership,
|
|
||||||
userRoomsByUrl,
|
|
||||||
getDefaultPubkeys,
|
|
||||||
} from "@app/core/state"
|
|
||||||
import {pushModal} from "@app/util/modal"
|
import {pushModal} from "@app/util/modal"
|
||||||
|
|
||||||
const discoverRelays = () =>
|
const discoverRelays = () =>
|
||||||
@@ -32,7 +24,7 @@
|
|||||||
filters: [{kinds: [ROOMS]}],
|
filters: [{kinds: [ROOMS]}],
|
||||||
relays: Router.get().Index().getUrls(),
|
relays: Router.get().Index().getUrls(),
|
||||||
}),
|
}),
|
||||||
...getDefaultPubkeys().map(async pubkey => {
|
...$defaultPubkeys.map(async pubkey => {
|
||||||
await loadRelaySelections(pubkey)
|
await loadRelaySelections(pubkey)
|
||||||
|
|
||||||
const membership = await loadMembership(pubkey)
|
const membership = await loadMembership(pubkey)
|
||||||
@@ -42,27 +34,15 @@
|
|||||||
}),
|
}),
|
||||||
])
|
])
|
||||||
|
|
||||||
const wotGraph = $derived.by(() => {
|
|
||||||
const scores = new Map<string, Set<string>>()
|
|
||||||
|
|
||||||
for (const pubkey of getDefaultPubkeys()) {
|
|
||||||
for (const url of getMembershipUrls($membershipsByPubkey.get(pubkey))) {
|
|
||||||
addToMapKey(scores, url, pubkey)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return scores
|
|
||||||
})
|
|
||||||
|
|
||||||
const relaySearch = $derived(
|
const relaySearch = $derived(
|
||||||
createSearch(
|
createSearch(
|
||||||
$relays.filter(r => wotGraph.has(r.url)),
|
$relays.filter(r => $membersByUrl.has(r.url)),
|
||||||
{
|
{
|
||||||
getValue: (relay: Relay) => relay.url,
|
getValue: (relay: Relay) => relay.url,
|
||||||
sortFn: ({score, item}) => {
|
sortFn: ({score, item}) => {
|
||||||
if (score && score > 0.1) return -score!
|
if (score && score > 0.1) return -score!
|
||||||
|
|
||||||
const wotScore = wotGraph.get(item.url)?.size || 0
|
const wotScore = $membersByUrl.get(item.url)?.size || 0
|
||||||
|
|
||||||
return score ? dec(score) * wotScore : -wotScore
|
return score ? dec(score) * wotScore : -wotScore
|
||||||
},
|
},
|
||||||
@@ -110,44 +90,9 @@
|
|||||||
</label>
|
</label>
|
||||||
{#each relaySearch.searchOptions(term).slice(0, limit) as relay (relay.url)}
|
{#each relaySearch.searchOptions(term).slice(0, limit) as relay (relay.url)}
|
||||||
<Button
|
<Button
|
||||||
class="card2 bg-alt col-4 text-left shadow-xl transition-all hover:shadow-2xl hover:brightness-[1.1]"
|
class="card2 bg-alt shadow-xl transition-all hover:shadow-2xl hover:brightness-[1.1]"
|
||||||
onclick={() => openSpace(relay.url)}>
|
onclick={() => openSpace(relay.url)}>
|
||||||
<div class="col-2">
|
<RelaySummary url={relay.url} />
|
||||||
<div class="relative flex gap-4">
|
|
||||||
<div class="relative">
|
|
||||||
<div class="avatar relative">
|
|
||||||
<div
|
|
||||||
class="center !flex h-12 w-12 min-w-12 rounded-full border-2 border-solid border-base-300 bg-base-300">
|
|
||||||
{#if relay.profile?.icon}
|
|
||||||
<img alt="" src={relay.profile.icon} />
|
|
||||||
{:else}
|
|
||||||
<Icon icon="ghost" size={5} />
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{#if $userRoomsByUrl.has(relay.url)}
|
|
||||||
<div
|
|
||||||
class="tooltip absolute -right-1 -top-1 h-5 w-5 rounded-full bg-primary"
|
|
||||||
data-tip="You are already a member of this space.">
|
|
||||||
<Icon icon="check-circle" class="scale-110" />
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<h2 class="ellipsize whitespace-nowrap text-xl">
|
|
||||||
<RelayName url={relay.url} />
|
|
||||||
</h2>
|
|
||||||
<p class="text-sm opacity-75">{relay.url}</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<RelayDescription url={relay.url} />
|
|
||||||
</div>
|
|
||||||
{#if gt(wotGraph.get(relay.url)?.size, 0)}
|
|
||||||
<div class="row-2 card2 card2-sm bg-alt">
|
|
||||||
Members:
|
|
||||||
<ProfileCircles pubkeys={Array.from(wotGraph.get(relay.url) || [])} />
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
</Button>
|
</Button>
|
||||||
{/each}
|
{/each}
|
||||||
{#await discoverRelays()}
|
{#await discoverRelays()}
|
||||||
|
|||||||
19
src/routes/join/+page.svelte
Normal file
19
src/routes/join/+page.svelte
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import {page} from "$app/stores"
|
||||||
|
import {goto} from "$app/navigation"
|
||||||
|
import Icon from "@lib/components/Icon.svelte"
|
||||||
|
import Dialog from "@lib/components/Dialog.svelte"
|
||||||
|
import Button from "@lib/components/Button.svelte"
|
||||||
|
import SpaceInviteAccept from "@app/components/SpaceInviteAccept.svelte"
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<Dialog>
|
||||||
|
<SpaceInviteAccept invite={$page.url.href}>
|
||||||
|
{#snippet abortAction()}
|
||||||
|
<Button class="btn btn-link" onclick={() => goto("/home")}>
|
||||||
|
<Icon icon="alt-arrow-left" />
|
||||||
|
Go back
|
||||||
|
</Button>
|
||||||
|
{/snippet}
|
||||||
|
</SpaceInviteAccept>
|
||||||
|
</Dialog>
|
||||||
@@ -6,15 +6,13 @@
|
|||||||
import Page from "@lib/components/Page.svelte"
|
import Page from "@lib/components/Page.svelte"
|
||||||
import ContentSearch from "@lib/components/ContentSearch.svelte"
|
import ContentSearch from "@lib/components/ContentSearch.svelte"
|
||||||
import PeopleItem from "@app/components/PeopleItem.svelte"
|
import PeopleItem from "@app/components/PeopleItem.svelte"
|
||||||
import {getDefaultPubkeys} from "@app/core/state"
|
import {defaultPubkeys} from "@app/core/state"
|
||||||
|
|
||||||
const defaultPubkeys = getDefaultPubkeys()
|
|
||||||
|
|
||||||
let term = $state("")
|
let term = $state("")
|
||||||
let limit = $state(10)
|
let limit = $state(10)
|
||||||
let element: Element | undefined = $state()
|
let element: Element | undefined = $state()
|
||||||
|
|
||||||
const pubkeys = $derived(term ? $profileSearch.searchValues(term) : defaultPubkeys)
|
const pubkeys = $derived(term ? $profileSearch.searchValues(term) : $defaultPubkeys)
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
const scroller = createScroller({
|
const scroller = createScroller({
|
||||||
|
|||||||
Reference in New Issue
Block a user