forked from coracle/flotilla
Show active links in compose
This commit is contained in:
Generated
+14
@@ -12,6 +12,7 @@
|
|||||||
"@noble/hashes": "^1.4.0",
|
"@noble/hashes": "^1.4.0",
|
||||||
"@poppanator/sveltekit-svg": "^4.2.1",
|
"@poppanator/sveltekit-svg": "^4.2.1",
|
||||||
"@tiptap/starter-kit": "^2.6.4",
|
"@tiptap/starter-kit": "^2.6.4",
|
||||||
|
"@tiptap/suggestion": "^2.6.4",
|
||||||
"@types/throttle-debounce": "^5.0.2",
|
"@types/throttle-debounce": "^5.0.2",
|
||||||
"@welshman/lib": "^0.0.14",
|
"@welshman/lib": "^0.0.14",
|
||||||
"@welshman/net": "^0.0.18",
|
"@welshman/net": "^0.0.18",
|
||||||
@@ -1490,6 +1491,19 @@
|
|||||||
"url": "https://github.com/sponsors/ueberdosis"
|
"url": "https://github.com/sponsors/ueberdosis"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@tiptap/suggestion": {
|
||||||
|
"version": "2.6.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/@tiptap/suggestion/-/suggestion-2.6.4.tgz",
|
||||||
|
"integrity": "sha512-t4GOEcsVSCwTlugHjZdK5Swe6or/tBej5E3ZWYOFHxkNLDod76Q7hvAeBPYrLeDo6m3sPnxrazfdqSeVclk72g==",
|
||||||
|
"funding": {
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/ueberdosis"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"@tiptap/core": "^2.6.4",
|
||||||
|
"@tiptap/pm": "^2.6.4"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@trysound/sax": {
|
"node_modules/@trysound/sax": {
|
||||||
"version": "0.2.0",
|
"version": "0.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.2.0.tgz",
|
||||||
|
|||||||
@@ -38,6 +38,7 @@
|
|||||||
"@noble/hashes": "^1.4.0",
|
"@noble/hashes": "^1.4.0",
|
||||||
"@poppanator/sveltekit-svg": "^4.2.1",
|
"@poppanator/sveltekit-svg": "^4.2.1",
|
||||||
"@tiptap/starter-kit": "^2.6.4",
|
"@tiptap/starter-kit": "^2.6.4",
|
||||||
|
"@tiptap/suggestion": "^2.6.4",
|
||||||
"@types/throttle-debounce": "^5.0.2",
|
"@types/throttle-debounce": "^5.0.2",
|
||||||
"@welshman/lib": "^0.0.14",
|
"@welshman/lib": "^0.0.14",
|
||||||
"@welshman/net": "^0.0.18",
|
"@welshman/net": "^0.0.18",
|
||||||
|
|||||||
@@ -91,3 +91,7 @@
|
|||||||
.link-content {
|
.link-content {
|
||||||
@apply text-sm rounded px-1 bg-neutral text-neutral-content no-underline;
|
@apply text-sm rounded px-1 bg-neutral text-neutral-content no-underline;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.link-content.link-content-selected {
|
||||||
|
@apply bg-primary text-primary-content;
|
||||||
|
}
|
||||||
|
|||||||
@@ -5,19 +5,23 @@
|
|||||||
import StarterKit from '@tiptap/starter-kit'
|
import StarterKit from '@tiptap/starter-kit'
|
||||||
import {NostrExtension} from 'nostr-editor'
|
import {NostrExtension} from 'nostr-editor'
|
||||||
import type {StampedEvent} from '@welshman/util'
|
import type {StampedEvent} from '@welshman/util'
|
||||||
import {LinkExtension} from '@lib/tiptap'
|
import {LinkExtension, Mention, Topic} from '@lib/tiptap'
|
||||||
import GroupComposeMention from '@app/components/GroupComposeMention.svelte'
|
import GroupComposeMention from '@app/components/GroupComposeMention.svelte'
|
||||||
import GroupComposeEvent from '@app/components/GroupComposeEvent.svelte'
|
import GroupComposeEvent from '@app/components/GroupComposeEvent.svelte'
|
||||||
import GroupComposeImage from '@app/components/GroupComposeImage.svelte'
|
import GroupComposeImage from '@app/components/GroupComposeImage.svelte'
|
||||||
import GroupComposeBolt11 from '@app/components/GroupComposeBolt11.svelte'
|
import GroupComposeBolt11 from '@app/components/GroupComposeBolt11.svelte'
|
||||||
import GroupComposeVideo from '@app/components/GroupComposeVideo.svelte'
|
import GroupComposeVideo from '@app/components/GroupComposeVideo.svelte'
|
||||||
import GroupComposeLink from '@app/components/GroupComposeLink.svelte'
|
import GroupComposeLink from '@app/components/GroupComposeLink.svelte'
|
||||||
|
import GroupComposeSuggestions from '@app/components/GroupComposeSuggestions.svelte'
|
||||||
|
import GroupComposeTopicSuggestion from '@app/components/GroupComposeTopicSuggestion.svelte'
|
||||||
|
import GroupComposeProfileSuggestion from '@app/components/GroupComposeProfileSuggestion.svelte'
|
||||||
import {signer} from '@app/base'
|
import {signer} from '@app/base'
|
||||||
|
import {searchProfiles, searchTopics, displayProfileByPubkey} from '@app/state'
|
||||||
|
|
||||||
let editor: Readable<Editor>
|
let editor: Readable<Editor>
|
||||||
|
|
||||||
const asInline = (extend: Record<string, any>) =>
|
const asInline = (extend: Record<string, any>) =>
|
||||||
({inline: true, group: 'inline', draggable: false, ...extend})
|
({inline: true, group: 'inline', ...extend})
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
editor = createEditor({
|
editor = createEditor({
|
||||||
@@ -56,12 +60,57 @@
|
|||||||
onDrop() {
|
onDrop() {
|
||||||
// setPending(true)
|
// setPending(true)
|
||||||
},
|
},
|
||||||
onComplete(currentEditor: Editor) {
|
onComplete(currentEditor: any) {
|
||||||
console.log('Upload Completed', currentEditor.getText())
|
console.log('Upload Completed', currentEditor.getText())
|
||||||
// setPending(false)
|
// setPending(false)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
Mention.configure(
|
||||||
|
(() => {
|
||||||
|
let suggestions: GroupComposeSuggestions
|
||||||
|
|
||||||
|
const mapProps = (props: any) => ({
|
||||||
|
term: props.query,
|
||||||
|
select: (id: string) => props.command({id}),
|
||||||
|
search: searchProfiles,
|
||||||
|
component: GroupComposeProfileSuggestion,
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
getLabel: displayProfileByPubkey,
|
||||||
|
tippyOptions: {arrow: false, theme: "transparent"},
|
||||||
|
onStart: ({target, props}) => {
|
||||||
|
suggestions = new GroupComposeSuggestions({target, props: mapProps(props)})
|
||||||
|
},
|
||||||
|
onUpdate: ({props}) => suggestions.$set(mapProps(props)),
|
||||||
|
onKeyDown: ({event}) => suggestions.onKeyDown(event),
|
||||||
|
onExit: () => suggestions.$destroy(),
|
||||||
|
}
|
||||||
|
})(),
|
||||||
|
),
|
||||||
|
Topic.configure(
|
||||||
|
(() => {
|
||||||
|
let suggestions: GroupComposeSuggestions
|
||||||
|
|
||||||
|
const mapProps = (props: any) => ({
|
||||||
|
term: props.query,
|
||||||
|
select: (id: string) => props.command({id}),
|
||||||
|
search: searchTopics,
|
||||||
|
component: GroupComposeTopicSuggestion,
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
tippyOptions: {arrow: false, theme: "transparent"},
|
||||||
|
onStart: ({target, props}) => {
|
||||||
|
suggestions = new GroupComposeSuggestions({target, props: mapProps(props)})
|
||||||
|
},
|
||||||
|
onUpdate: ({props}) => suggestions.$set(mapProps(props)),
|
||||||
|
onKeyDown: ({event}) => suggestions.onKeyDown(event),
|
||||||
|
onExit: () => suggestions.$destroy(),
|
||||||
|
}
|
||||||
|
})(),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
content: '',
|
content: '',
|
||||||
onUpdate: () => {
|
onUpdate: () => {
|
||||||
|
|||||||
@@ -2,17 +2,18 @@
|
|||||||
import cx from 'classnames'
|
import cx from 'classnames'
|
||||||
import type {NodeViewProps} from '@tiptap/core'
|
import type {NodeViewProps} from '@tiptap/core'
|
||||||
import {NodeViewWrapper} from 'svelte-tiptap'
|
import {NodeViewWrapper} from 'svelte-tiptap'
|
||||||
import {stripProtocol} from '@welshman/lib'
|
import {displayUrl} from '@welshman/lib'
|
||||||
import Icon from '@lib/components/Icon.svelte'
|
import Icon from '@lib/components/Icon.svelte'
|
||||||
import Link from '@lib/components/Link.svelte'
|
import Link from '@lib/components/Link.svelte'
|
||||||
|
|
||||||
export let node: NodeViewProps['node']
|
export let node: NodeViewProps['node']
|
||||||
|
export let selected: NodeViewProps['selected']
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<NodeViewWrapper class="inline">
|
<NodeViewWrapper class="inline">
|
||||||
<Link external href={node.attrs.url} class="link-content">
|
<Link external href={node.attrs.url} class={cx("link-content", {'link-content-selected': selected})}>
|
||||||
<Icon icon="link-round" size={3} class="inline-block translate-y-px" />
|
<Icon icon="link-round" size={3} class="inline-block translate-y-px" />
|
||||||
{stripProtocol(node.attrs.url)}
|
{displayUrl(node.attrs.url)}
|
||||||
</Link>
|
</Link>
|
||||||
</NodeViewWrapper>
|
</NodeViewWrapper>
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
export let value
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
@{value}
|
||||||
|
</div>
|
||||||
@@ -0,0 +1,80 @@
|
|||||||
|
<svelte:options accessors />
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import {throttle} from "throttle-debounce"
|
||||||
|
import {slide} from "svelte/transition"
|
||||||
|
import {clamp} from "@welshman/lib"
|
||||||
|
|
||||||
|
export let term
|
||||||
|
export let search
|
||||||
|
export let select
|
||||||
|
export let component
|
||||||
|
export let loading = false
|
||||||
|
|
||||||
|
let index = 0
|
||||||
|
let element: Element
|
||||||
|
let items: string[] = []
|
||||||
|
|
||||||
|
$: populateItems(term)
|
||||||
|
|
||||||
|
const populateItems = throttle(300, term => {
|
||||||
|
items = $search.searchValues(term).slice(0, 30)
|
||||||
|
})
|
||||||
|
|
||||||
|
const setIndex = (newIndex: number, block: ScrollLogicalPosition) => {
|
||||||
|
index = clamp([0, items.length - 1], newIndex)
|
||||||
|
element.querySelector(`button:nth-child(${index})`)?.scrollIntoView({block})
|
||||||
|
}
|
||||||
|
|
||||||
|
export const onKeyDown = (e: any) => {
|
||||||
|
if (e.code === "Enter") {
|
||||||
|
const value = items[index]
|
||||||
|
|
||||||
|
if (value) {
|
||||||
|
select(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.code === "ArrowUp") {
|
||||||
|
setIndex(index - 1, "start")
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (e.code === "ArrowDown") {
|
||||||
|
setIndex(index + 1, "start")
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if items.length > 0}
|
||||||
|
<div
|
||||||
|
bind:this={element}
|
||||||
|
transition:slide|local={{duration: 100}}
|
||||||
|
class="mt-2 flex max-h-[350px] flex-col overflow-y-auto overflow-x-hidden border border-solid border-neutral-600">
|
||||||
|
{#each items as value, i (value)}
|
||||||
|
<button
|
||||||
|
class="cursor-pointer border-l-2 border-solid px-4 py-2 text-left text-neutral-100 hover:border-accent hover:bg-tinted-700"
|
||||||
|
class:bg-neutral-800={index !== i}
|
||||||
|
class:bg-tinted-700={index === i}
|
||||||
|
class:border-transparent={index !== i}
|
||||||
|
class:border-accent={index === i}
|
||||||
|
on:mousedown|preventDefault
|
||||||
|
on:click|preventDefault={() => select(value)}>
|
||||||
|
<svelte:component this={component} {value} />
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{#if loading}
|
||||||
|
<div transition:slide|local class="flex gap-2 bg-tinted-700 px-4 py-2 text-neutral-200">
|
||||||
|
<div>
|
||||||
|
<i class="fa fa-circle-notch fa-spin" />
|
||||||
|
</div>
|
||||||
|
Loading more options...
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
{/if}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
export let value
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
#{value}
|
||||||
|
</div>
|
||||||
@@ -3,11 +3,11 @@
|
|||||||
import {readable} from "svelte/store"
|
import {readable} from "svelte/store"
|
||||||
import {hash} from "@welshman/lib"
|
import {hash} from "@welshman/lib"
|
||||||
import type {TrustedEvent} from "@welshman/util"
|
import type {TrustedEvent} from "@welshman/util"
|
||||||
import {GROUP_REPLY, getAncestorTags, displayProfile, displayPubkey} from "@welshman/util"
|
import {GROUP_REPLY, getAncestorTags, displayPubkey} from "@welshman/util"
|
||||||
import {fly} from "@lib/transition"
|
import {fly} from "@lib/transition"
|
||||||
import Icon from "@lib/components/Icon.svelte"
|
import Icon from "@lib/components/Icon.svelte"
|
||||||
import Avatar from "@lib/components/Avatar.svelte"
|
import Avatar from "@lib/components/Avatar.svelte"
|
||||||
import {deriveProfile, deriveEvent} from "@app/state"
|
import {deriveProfile, deriveProfileDisplay, deriveEvent} from "@app/state"
|
||||||
|
|
||||||
export let event: TrustedEvent
|
export let event: TrustedEvent
|
||||||
export let showPubkey: boolean
|
export let showPubkey: boolean
|
||||||
@@ -35,6 +35,7 @@
|
|||||||
]
|
]
|
||||||
|
|
||||||
const profile = deriveProfile(event.pubkey)
|
const profile = deriveProfile(event.pubkey)
|
||||||
|
const profileDisplay = deriveProfileDisplay(event.pubkey)
|
||||||
const {replies} = getAncestorTags(event.tags)
|
const {replies} = getAncestorTags(event.tags)
|
||||||
const parentId = replies[0]?.[1]
|
const parentId = replies[0]?.[1]
|
||||||
const parentHints = [replies[0]?.[2]].filter(Boolean)
|
const parentHints = [replies[0]?.[2]].filter(Boolean)
|
||||||
@@ -43,6 +44,7 @@
|
|||||||
|
|
||||||
$: parentPubkey = $parentEvent?.pubkey || replies[0]?.[4]
|
$: parentPubkey = $parentEvent?.pubkey || replies[0]?.[4]
|
||||||
$: parentProfile = deriveProfile(parentPubkey)
|
$: parentProfile = deriveProfile(parentPubkey)
|
||||||
|
$: parentProfileDisplay = deriveProfileDisplay(parentPubkey)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div in:fly class="group relative flex flex-col gap-1 p-2 transition-colors hover:bg-base-300">
|
<div in:fly class="group relative flex flex-col gap-1 p-2 transition-colors hover:bg-base-300">
|
||||||
@@ -50,7 +52,7 @@
|
|||||||
<div class="flex items-center gap-1 pl-12 text-xs">
|
<div class="flex items-center gap-1 pl-12 text-xs">
|
||||||
<Icon icon="arrow-right" />
|
<Icon icon="arrow-right" />
|
||||||
<Avatar src={$parentProfile?.picture} size={4} />
|
<Avatar src={$parentProfile?.picture} size={4} />
|
||||||
<p class="text-primary">{displayProfile($parentProfile, displayPubkey(parentPubkey))}</p>
|
<p class="text-primary">{$parentProfileDisplay}</p>
|
||||||
<p></p>
|
<p></p>
|
||||||
<p
|
<p
|
||||||
class="flex cursor-pointer items-center gap-1 overflow-hidden text-ellipsis whitespace-nowrap opacity-75 hover:underline">
|
class="flex cursor-pointer items-center gap-1 overflow-hidden text-ellipsis whitespace-nowrap opacity-75 hover:underline">
|
||||||
@@ -67,8 +69,7 @@
|
|||||||
{/if}
|
{/if}
|
||||||
<div class="-mt-1">
|
<div class="-mt-1">
|
||||||
{#if showPubkey}
|
{#if showPubkey}
|
||||||
<strong class="text-sm" style="color: {colorValue}" data-color={colorName}
|
<strong class="text-sm" style="color: {colorValue}" data-color={colorName}>{$profileDisplay}</strong>
|
||||||
>{displayProfile($profile, displayPubkey(event.pubkey))}</strong>
|
|
||||||
{/if}
|
{/if}
|
||||||
<p class="text-sm">{event.content}</p>
|
<p class="text-sm">{event.content}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
+57
-1
@@ -1,3 +1,4 @@
|
|||||||
|
import {throttle} from 'throttle-debounce'
|
||||||
import type {Readable} from "svelte/store"
|
import type {Readable} from "svelte/store"
|
||||||
import type {FuseResult} from "fuse.js"
|
import type {FuseResult} from "fuse.js"
|
||||||
import {get, writable, readable, derived} from "svelte/store"
|
import {get, writable, readable, derived} from "svelte/store"
|
||||||
@@ -16,6 +17,7 @@ import {
|
|||||||
indexBy,
|
indexBy,
|
||||||
now,
|
now,
|
||||||
Worker,
|
Worker,
|
||||||
|
inc,
|
||||||
} from "@welshman/lib"
|
} from "@welshman/lib"
|
||||||
import {
|
import {
|
||||||
getIdFilters,
|
getIdFilters,
|
||||||
@@ -32,6 +34,8 @@ import {
|
|||||||
readProfile,
|
readProfile,
|
||||||
readList,
|
readList,
|
||||||
asDecryptedEvent,
|
asDecryptedEvent,
|
||||||
|
displayProfile,
|
||||||
|
displayPubkey,
|
||||||
GROUP_JOIN,
|
GROUP_JOIN,
|
||||||
GROUP_ADD_USER,
|
GROUP_ADD_USER,
|
||||||
} from "@welshman/util"
|
} from "@welshman/util"
|
||||||
@@ -39,7 +43,7 @@ import type {SignedEvent, HashedEvent, EventTemplate, TrustedEvent, PublishedPro
|
|||||||
import type {SubscribeRequest, PublishRequest} from "@welshman/net"
|
import type {SubscribeRequest, PublishRequest} from "@welshman/net"
|
||||||
import {publish as basePublish, subscribe as baseSubscribe, PublishStatus} from "@welshman/net"
|
import {publish as basePublish, subscribe as baseSubscribe, PublishStatus} from "@welshman/net"
|
||||||
import {decrypt, stamp, own, hash} from "@welshman/signer"
|
import {decrypt, stamp, own, hash} from "@welshman/signer"
|
||||||
import {deriveEvents, deriveEventsMapped, getter, withGetter} from "@welshman/store"
|
import {custom, deriveEvents, deriveEventsMapped, getter, withGetter} from "@welshman/store"
|
||||||
import {createSearch} from "@lib/util"
|
import {createSearch} from "@lib/util"
|
||||||
import type {Handle, Relay} from "@app/types"
|
import type {Handle, Relay} from "@app/types"
|
||||||
import {INDEXER_RELAYS, DUFFLEPUD_URL, repository, pk, getSession, getSigner} from "@app/base"
|
import {INDEXER_RELAYS, DUFFLEPUD_URL, repository, pk, getSession, getSigner} from "@app/base"
|
||||||
@@ -244,6 +248,43 @@ export const ensurePlaintext = async (e: TrustedEvent) => {
|
|||||||
return getPlaintext(e)
|
return getPlaintext(e)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Topics
|
||||||
|
|
||||||
|
export type Topic = {
|
||||||
|
name: string
|
||||||
|
count: number
|
||||||
|
}
|
||||||
|
|
||||||
|
export const topics = custom<Topic[]>(setter => {
|
||||||
|
const getTopics = () => {
|
||||||
|
const topics = new Map<string, number>()
|
||||||
|
for (const tagString of repository.eventsByTag.keys()) {
|
||||||
|
if (tagString.startsWith('t:')) {
|
||||||
|
const topic = tagString.slice(2).toLowerCase()
|
||||||
|
|
||||||
|
topics.set(topic, inc(topics.get(topic)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Array.from(topics.entries()).map(([name, count]) => ({name, count}))
|
||||||
|
}
|
||||||
|
|
||||||
|
setter(getTopics())
|
||||||
|
|
||||||
|
const onUpdate = throttle(3000, () => setter(getTopics()))
|
||||||
|
|
||||||
|
repository.on("update", onUpdate)
|
||||||
|
|
||||||
|
return () => repository.off("update", onUpdate)
|
||||||
|
})
|
||||||
|
|
||||||
|
export const searchTopics = derived(topics, $topics =>
|
||||||
|
createSearch($topics, {
|
||||||
|
getValue: (topic: Topic) => topic.name,
|
||||||
|
fuseOptions: {keys: ["name"]},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
// Relay info
|
// Relay info
|
||||||
|
|
||||||
export const relays = writable<Relay[]>([])
|
export const relays = writable<Relay[]>([])
|
||||||
@@ -331,6 +372,21 @@ export const {
|
|||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
export const searchProfiles = derived(profiles, $profiles =>
|
||||||
|
createSearch($profiles, {
|
||||||
|
getValue: (profile: PublishedProfile) => profile.event.pubkey,
|
||||||
|
fuseOptions: {
|
||||||
|
keys: ["name", "display_name", {name: "about", weight: 0.3}],
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
export const displayProfileByPubkey = (pubkey: string, profile?: PublishedProfile) =>
|
||||||
|
displayProfile(profile, pubkey ? displayPubkey(pubkey) : undefined)
|
||||||
|
|
||||||
|
export const deriveProfileDisplay = (pubkey: string) =>
|
||||||
|
derived(deriveProfile(pubkey), $profile => displayProfileByPubkey(pubkey, $profile))
|
||||||
|
|
||||||
// Relay selections
|
// Relay selections
|
||||||
|
|
||||||
export const getReadRelayUrls = (event?: TrustedEvent): string[] =>
|
export const getReadRelayUrls = (event?: TrustedEvent): string[] =>
|
||||||
|
|||||||
@@ -30,6 +30,10 @@ export const LinkExtension = Node.create({
|
|||||||
|
|
||||||
inline: true,
|
inline: true,
|
||||||
|
|
||||||
|
selectable: true,
|
||||||
|
|
||||||
|
draggable: true,
|
||||||
|
|
||||||
priority: 1000,
|
priority: 1000,
|
||||||
|
|
||||||
addAttributes() {
|
addAttributes() {
|
||||||
|
|||||||
@@ -0,0 +1,3 @@
|
|||||||
|
import {createPopoverNode} from '@lib/tiptap/common'
|
||||||
|
|
||||||
|
export const Mention = createPopoverNode('mention', '@')
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
import {createPopoverNode} from '@lib/tiptap/common'
|
||||||
|
|
||||||
|
export const Topic = createPopoverNode('topic', '#')
|
||||||
@@ -0,0 +1,149 @@
|
|||||||
|
import tippy, {type Instance} from 'tippy.js'
|
||||||
|
import {mergeAttributes, Node} from '@tiptap/core'
|
||||||
|
import {PluginKey} from '@tiptap/pm/state'
|
||||||
|
import Suggestion from '@tiptap/suggestion'
|
||||||
|
|
||||||
|
export type PopoverOptions = {
|
||||||
|
tippyOptions: Record<string, any>
|
||||||
|
getLabel?: (id: string) => string
|
||||||
|
onStart?: (opts: any) => void
|
||||||
|
onUpdate?: (opts: any) => void
|
||||||
|
onKeyDown?: (opts: any) => boolean | undefined
|
||||||
|
onExit?: () => void
|
||||||
|
}
|
||||||
|
|
||||||
|
export const createPopoverNode = (name: string, char: string) => {
|
||||||
|
const pluginKey = new PluginKey(name)
|
||||||
|
|
||||||
|
return Node.create<PopoverOptions>({
|
||||||
|
name,
|
||||||
|
group: 'inline',
|
||||||
|
inline: true,
|
||||||
|
selectable: false,
|
||||||
|
atom: true,
|
||||||
|
addAttributes: () => ({
|
||||||
|
id: {
|
||||||
|
default: null,
|
||||||
|
parseHTML: el => el.getAttribute('data-id'),
|
||||||
|
renderHTML: ({id}) => id ? {'data-id': id} : {},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
parseHTML() {
|
||||||
|
return [{tag: `span[data-type="${this.name}"]`}]
|
||||||
|
},
|
||||||
|
renderHTML({node, HTMLAttributes}) {
|
||||||
|
const label = this.options.getLabel?.(node.attrs.id) || node.attrs.id
|
||||||
|
|
||||||
|
return ['span', mergeAttributes({'data-type': this.name}, HTMLAttributes), `${char}${label}`]
|
||||||
|
},
|
||||||
|
renderText({node}) {
|
||||||
|
return `${char}${this.options.getLabel?.(node.attrs.id) || node.attrs.id}`
|
||||||
|
},
|
||||||
|
addKeyboardShortcuts() {
|
||||||
|
return {
|
||||||
|
Backspace: () => this.editor.commands.command(({ tr, state }) => {
|
||||||
|
let isMention = false
|
||||||
|
const { selection } = state
|
||||||
|
const { empty, anchor } = selection
|
||||||
|
|
||||||
|
if (!empty) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
state.doc.nodesBetween(anchor - 1, anchor, (node, pos) => {
|
||||||
|
if (node.type.name === this.name) {
|
||||||
|
isMention = true
|
||||||
|
tr.insertText('', pos, pos + node.nodeSize)
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return isMention
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
},
|
||||||
|
addProseMirrorPlugins() {
|
||||||
|
return [
|
||||||
|
Suggestion({
|
||||||
|
char,
|
||||||
|
pluginKey,
|
||||||
|
editor: this.editor,
|
||||||
|
command: ({editor, range, props}) => {
|
||||||
|
// increase range.to by one when the next node is of type "text"
|
||||||
|
// and starts with a space character
|
||||||
|
const nodeAfter = editor.view.state.selection.$to.nodeAfter
|
||||||
|
const overrideSpace = nodeAfter?.text?.startsWith(' ')
|
||||||
|
|
||||||
|
if (overrideSpace) {
|
||||||
|
range.to += 1
|
||||||
|
}
|
||||||
|
|
||||||
|
editor
|
||||||
|
.chain()
|
||||||
|
.focus()
|
||||||
|
.insertContentAt(range, [
|
||||||
|
{type: this.name, attrs: props},
|
||||||
|
{type: 'text', text: ' '},
|
||||||
|
])
|
||||||
|
.run()
|
||||||
|
|
||||||
|
window.getSelection()?.collapseToEnd()
|
||||||
|
},
|
||||||
|
allow: ({ state, range }) => {
|
||||||
|
const $from = state.doc.resolve(range.from)
|
||||||
|
const type = state.schema.nodes[this.name]
|
||||||
|
const allow = !!$from.parent.type.contentMatch.matchType(type)
|
||||||
|
|
||||||
|
return allow
|
||||||
|
},
|
||||||
|
render: () => {
|
||||||
|
let popover: Instance[]
|
||||||
|
let target: HTMLElement
|
||||||
|
|
||||||
|
return {
|
||||||
|
onStart: props => {
|
||||||
|
target = document.createElement("div")
|
||||||
|
|
||||||
|
popover = tippy('body', {
|
||||||
|
getReferenceClientRect: props.clientRect as any,
|
||||||
|
appendTo: document.body,
|
||||||
|
content: target,
|
||||||
|
showOnCreate: true,
|
||||||
|
interactive: true,
|
||||||
|
trigger: "manual",
|
||||||
|
placement: "bottom-start",
|
||||||
|
...this.options.tippyOptions,
|
||||||
|
})
|
||||||
|
|
||||||
|
this.options.onStart?.({target, props})
|
||||||
|
},
|
||||||
|
onUpdate: props => {
|
||||||
|
this.options.onUpdate?.({props})
|
||||||
|
|
||||||
|
if (props.clientRect) {
|
||||||
|
popover[0].setProps({
|
||||||
|
getReferenceClientRect: props.clientRect as any,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onKeyDown: props => {
|
||||||
|
if (props.event.key === "Escape") {
|
||||||
|
popover[0].hide()
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return Boolean(this.options.onKeyDown?.(props))
|
||||||
|
},
|
||||||
|
onExit: () => {
|
||||||
|
popover[0].destroy()
|
||||||
|
this.options.onExit?.()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
]
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -1 +1,3 @@
|
|||||||
export * from '@lib/tiptap/LinkExtension'
|
export {LinkExtension} from '@lib/tiptap/LinkExtension'
|
||||||
|
export {Mention} from '@lib//tiptap/Mention'
|
||||||
|
export {Topic} from '@lib//tiptap/Topic'
|
||||||
|
|||||||
Reference in New Issue
Block a user