Memoize some stores, improve room support
This commit is contained in:
@@ -1,5 +1,5 @@
|
|||||||
import {derived, Readable} from "svelte/store"
|
import {derived, Readable} from "svelte/store"
|
||||||
import {withGetter} from "@welshman/store"
|
import {withGetter, memoized} from "@welshman/store"
|
||||||
import {pubkey} from "./session.js"
|
import {pubkey} from "./session.js"
|
||||||
import {profilesByPubkey, loadProfile} from "./profiles.js"
|
import {profilesByPubkey, loadProfile} from "./profiles.js"
|
||||||
import {followsByPubkey, loadFollows} from "./follows.js"
|
import {followsByPubkey, loadFollows} from "./follows.js"
|
||||||
@@ -19,13 +19,15 @@ export type MakeUserDataOptions<T> = {
|
|||||||
|
|
||||||
export const makeUserData = <T>({mapStore, loadItem}: MakeUserDataOptions<T>) =>
|
export const makeUserData = <T>({mapStore, loadItem}: MakeUserDataOptions<T>) =>
|
||||||
withGetter(
|
withGetter(
|
||||||
derived([mapStore, pubkey], ([$mapStore, $pubkey]) => {
|
memoized(
|
||||||
if (!$pubkey) return undefined
|
derived([mapStore, pubkey], ([$mapStore, $pubkey]) => {
|
||||||
|
if (!$pubkey) return undefined
|
||||||
|
|
||||||
loadItem($pubkey)
|
loadItem($pubkey)
|
||||||
|
|
||||||
return $mapStore.get($pubkey)
|
return $mapStore.get($pubkey)
|
||||||
}),
|
}),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
export const makeUserLoader =
|
export const makeUserLoader =
|
||||||
|
|||||||
@@ -1030,11 +1030,11 @@ export const call = <T>(f: () => T, ...args: unknown[]) => f()
|
|||||||
* @param f - Function to memoize
|
* @param f - Function to memoize
|
||||||
* @returns Memoized function
|
* @returns Memoized function
|
||||||
*/
|
*/
|
||||||
export const memoize = <T>(f: (...args: any[]) => T) => {
|
export const memoize = <T, Args extends any[]>(f: (...args: Args) => T) => {
|
||||||
let prevArgs: any[]
|
let prevArgs: Args
|
||||||
let result: T
|
let result: T
|
||||||
|
|
||||||
return (...args: any[]) => {
|
return (...args: Args): T => {
|
||||||
if (!equals(prevArgs, args)) {
|
if (!equals(prevArgs, args)) {
|
||||||
prevArgs = args
|
prevArgs = args
|
||||||
result = f(...args)
|
result = f(...args)
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import {readable, derived, writable, Readable, Subscriber} from "svelte/store"
|
import {readable, derived, writable, Readable, Subscriber} from "svelte/store"
|
||||||
import {batch, indexBy, remove, assoc, now} from "@welshman/lib"
|
import {batch, indexBy, remove, assoc, now} from "@welshman/lib"
|
||||||
import {withGetter, ReadableWithGetter} from "./getter.js"
|
import {withGetter, ReadableWithGetter} from "./getter.js"
|
||||||
|
import {memoized} from "./memoize.js"
|
||||||
|
|
||||||
// Collection utility
|
// Collection utility
|
||||||
|
|
||||||
@@ -126,7 +127,7 @@ export const collection = <T>({name, store, getKey, load}: CollectionOptions<T>)
|
|||||||
// store will update when it arrives
|
// store will update when it arrives
|
||||||
loadItem(key, relays)
|
loadItem(key, relays)
|
||||||
|
|
||||||
return derived(indexStore, $index => $index.get(key))
|
return memoized<T | undefined>(derived(indexStore, $index => $index.get(key)))
|
||||||
}
|
}
|
||||||
|
|
||||||
const onItem = (cb: Subscriber<T>) => {
|
const onItem = (cb: Subscriber<T>) => {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
export * from "./synced.js"
|
export * from "./synced.js"
|
||||||
export * from "./getter.js"
|
export * from "./getter.js"
|
||||||
export * from "./throttle.js"
|
export * from "./throttle.js"
|
||||||
|
export * from "./memoize.js"
|
||||||
export * from "./custom.js"
|
export * from "./custom.js"
|
||||||
export * from "./repository.js"
|
export * from "./repository.js"
|
||||||
export * from "./collection.js"
|
export * from "./collection.js"
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
import {Readable, Subscriber} from "svelte/store"
|
||||||
|
import {memoize} from "@welshman/lib"
|
||||||
|
|
||||||
|
export const memoized = <T>(store: Readable<T>) => {
|
||||||
|
const {subscribe} = store
|
||||||
|
|
||||||
|
return {...store, subscribe: (f: Subscriber<T>) => subscribe(memoize(f))}
|
||||||
|
}
|
||||||
+69
-20
@@ -1,4 +1,4 @@
|
|||||||
import {randomId, nthNe} from "@welshman/lib"
|
import {randomId, spec} from "@welshman/lib"
|
||||||
import {
|
import {
|
||||||
ROOM_META,
|
ROOM_META,
|
||||||
ROOM_DELETE,
|
ROOM_DELETE,
|
||||||
@@ -8,10 +8,18 @@ import {
|
|||||||
ROOM_LEAVE,
|
ROOM_LEAVE,
|
||||||
} from "./Kinds.js"
|
} from "./Kinds.js"
|
||||||
import {makeEvent, TrustedEvent, getIdentifier} from "./Events.js"
|
import {makeEvent, TrustedEvent, getIdentifier} from "./Events.js"
|
||||||
|
import {getTag, getTagValue} from "./Tags.js"
|
||||||
|
|
||||||
export type RoomMeta = {
|
export type RoomMeta = {
|
||||||
id: string
|
h: string
|
||||||
tags: string[][]
|
name?: string
|
||||||
|
about?: string
|
||||||
|
picture?: string
|
||||||
|
pictureMeta?: string[]
|
||||||
|
isClosed?: boolean
|
||||||
|
isHidden?: boolean
|
||||||
|
isPrivate?: boolean
|
||||||
|
isRestricted?: boolean
|
||||||
event?: TrustedEvent
|
event?: TrustedEvent
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -19,38 +27,79 @@ export type PublishedRoomMeta = Omit<RoomMeta, "event"> & {
|
|||||||
event: TrustedEvent
|
event: TrustedEvent
|
||||||
}
|
}
|
||||||
|
|
||||||
export const makeRoomMeta = (room: Partial<RoomMeta> = {}): RoomMeta => ({
|
export const makeRoomMeta = (room: Partial<RoomMeta> = {}): RoomMeta => {
|
||||||
id: randomId(),
|
return {
|
||||||
tags: [],
|
h: randomId(),
|
||||||
...room,
|
...room,
|
||||||
})
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export const readRoomMeta = (event: TrustedEvent): PublishedRoomMeta => {
|
export const readRoomMeta = (event: TrustedEvent): PublishedRoomMeta => {
|
||||||
if (event.kind !== ROOM_META) {
|
if (event.kind !== ROOM_META) {
|
||||||
throw new Error("Invalid group meta event")
|
throw new Error("Invalid group meta event")
|
||||||
}
|
}
|
||||||
|
|
||||||
const id = getIdentifier(event)
|
const h = getIdentifier(event)
|
||||||
|
|
||||||
if (!id) {
|
if (!h) {
|
||||||
throw new Error("Group meta event had no d tag")
|
throw new Error("Group meta event had no d tag")
|
||||||
}
|
}
|
||||||
|
|
||||||
const tags = event.tags.filter(nthNe(0, "d"))
|
return {
|
||||||
|
h,
|
||||||
return {id, tags, event}
|
event,
|
||||||
|
name: getTagValue("name", event.tags),
|
||||||
|
about: getTagValue("about", event.tags),
|
||||||
|
picture: getTagValue("picture", event.tags),
|
||||||
|
pictureMeta: getTag("picture", event.tags)?.slice(2),
|
||||||
|
isClosed: event.tags.some(spec(["closed"])),
|
||||||
|
isHidden: event.tags.some(spec(["hidden"])),
|
||||||
|
isPrivate: event.tags.some(spec(["private"])),
|
||||||
|
isRestricted: event.tags.some(spec(["restricted"])),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const makeRoomCreateEvent = (room: RoomMeta) =>
|
export const makeRoomCreateEvent = (room: RoomMeta) =>
|
||||||
makeEvent(ROOM_CREATE, {tags: [["h", room.id]]})
|
makeEvent(ROOM_CREATE, {tags: [["h", room.h]]})
|
||||||
|
|
||||||
export const makeRoomDeleteEvent = (room: RoomMeta) =>
|
export const makeRoomDeleteEvent = (room: RoomMeta) =>
|
||||||
makeEvent(ROOM_DELETE, {tags: [["h", room.id]]})
|
makeEvent(ROOM_DELETE, {tags: [["h", room.h]]})
|
||||||
|
|
||||||
export const makeRoomEditEvent = (room: RoomMeta) =>
|
export const makeRoomEditEvent = (room: RoomMeta) => {
|
||||||
makeEvent(ROOM_EDIT_META, {tags: [["h", room.id], ...room.tags]})
|
const tags = [["h", room.h]]
|
||||||
|
|
||||||
export const makeRoomJoinEvent = (room: RoomMeta) => makeEvent(ROOM_JOIN, {tags: [["h", room.id]]})
|
if (room.name) tags.push(["name", room.name])
|
||||||
|
if (room.about) tags.push(["about", room.about])
|
||||||
|
|
||||||
export const makeRoomLeaveEvent = (room: RoomMeta) =>
|
if (room.picture) {
|
||||||
makeEvent(ROOM_LEAVE, {tags: [["h", room.id]]})
|
const tag = ["picture", room.picture]
|
||||||
|
|
||||||
|
if (room.pictureMeta) {
|
||||||
|
tag.push(...room.pictureMeta)
|
||||||
|
}
|
||||||
|
|
||||||
|
tags.push(tag)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (room.isClosed) tags.push(["closed"])
|
||||||
|
if (room.isHidden) tags.push(["hidden"])
|
||||||
|
if (room.isPrivate) tags.push(["private"])
|
||||||
|
if (room.isRestricted) tags.push(["restricted"])
|
||||||
|
|
||||||
|
if (room.event) {
|
||||||
|
for (const t of room.event.tags) {
|
||||||
|
if (tags.some(spec(t.slice(0, 1)))) continue
|
||||||
|
if (["closed", "hidden", "private", "restricted"].includes(t[0])) continue
|
||||||
|
|
||||||
|
tags.push(t)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(room, tags)
|
||||||
|
|
||||||
|
return makeEvent(ROOM_EDIT_META, {tags})
|
||||||
|
}
|
||||||
|
|
||||||
|
export const makeRoomJoinEvent = (room: RoomMeta) => makeEvent(ROOM_JOIN, {tags: [["h", room.h]]})
|
||||||
|
|
||||||
|
export const makeRoomLeaveEvent = (room: RoomMeta) => makeEvent(ROOM_LEAVE, {tags: [["h", room.h]]})
|
||||||
|
|||||||
Reference in New Issue
Block a user