* feat: adjust default window size * feat: save window state * feat: add window state plugin * feat: add search * feat: use negentropy for newsfeed * feat: live feeds * feat: add search user
297 lines
6.2 KiB
TypeScript
297 lines
6.2 KiB
TypeScript
import type { EventTag, Kind, Meta, NostrEvent } from "@lume/types";
|
|
import { type Result, commands } from "./commands";
|
|
|
|
export class LumeEvent {
|
|
public id: string;
|
|
public pubkey: string;
|
|
public created_at: number;
|
|
public kind: Kind;
|
|
public tags: string[][];
|
|
public content: string;
|
|
public sig: string;
|
|
public meta: Meta;
|
|
public relay?: string;
|
|
public replies?: LumeEvent[];
|
|
#raw: NostrEvent;
|
|
|
|
constructor(event: NostrEvent) {
|
|
this.#raw = event;
|
|
Object.assign(this, event);
|
|
}
|
|
|
|
get isQuote() {
|
|
return this.tags.filter((tag) => tag[0] === "q").length > 0;
|
|
}
|
|
|
|
get isConversation() {
|
|
const tags = this.tags.filter(
|
|
(tag) => tag[0] === "e" && tag[3] !== "mention",
|
|
);
|
|
return tags.length > 0;
|
|
}
|
|
|
|
get mentions() {
|
|
return this.tags.filter((tag) => tag[0] === "p").map((tag) => tag[1]);
|
|
}
|
|
|
|
get repostId() {
|
|
return this.tags.find((tag) => tag[0] === "e")[1];
|
|
}
|
|
|
|
get thread() {
|
|
let root: EventTag = null;
|
|
let reply: EventTag = null;
|
|
|
|
// Get all event references from tags, ignore mention.
|
|
const events = this.tags.filter(
|
|
(el) => el[0] === "e" && el[3] !== "mention",
|
|
);
|
|
|
|
if (events.length === 1) {
|
|
root = { id: events[0][1], relayHint: events[0][2] };
|
|
}
|
|
|
|
if (events.length === 2) {
|
|
root = { id: events[0][1], relayHint: events[0][2] };
|
|
reply = { id: events[1][1], relayHint: events[1][2] };
|
|
}
|
|
|
|
if (events.length > 2) {
|
|
for (const tag of events) {
|
|
if (tag[3] === "root") root = { id: tag[1], relayHint: tag[2] };
|
|
if (tag[3] === "reply") reply = { id: tag[1], relayHint: tag[2] };
|
|
}
|
|
}
|
|
|
|
// Fix some rare case when root same as reply
|
|
if (root && reply && root.id === reply.id) {
|
|
reply = null;
|
|
}
|
|
|
|
return {
|
|
root,
|
|
reply,
|
|
};
|
|
}
|
|
|
|
get quote() {
|
|
const tag = this.tags.filter(
|
|
(tag) => tag[0] === "q" || tag[3] === "mention",
|
|
);
|
|
const id = tag[0][1];
|
|
const relayHint = tag[0][2];
|
|
|
|
return { id, relayHint };
|
|
}
|
|
|
|
get warning() {
|
|
const warningTag = this.tags.filter(
|
|
(tag) => tag[0] === "content-warning",
|
|
)?.[0];
|
|
|
|
if (warningTag) {
|
|
return warningTag[1];
|
|
} else {
|
|
const nsfwTag = this.tags.filter(
|
|
(tag) => tag[0] === "t" && tag[1] === "NSFW",
|
|
)?.[0];
|
|
|
|
if (nsfwTag) {
|
|
return "NSFW";
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
}
|
|
|
|
public async getEventReplies() {
|
|
const query = await commands.getReplies(this.id);
|
|
|
|
if (query.status === "ok") {
|
|
const events = query.data
|
|
// Create Lume Events
|
|
.map((item) => LumeEvent.from(item.raw, item.parsed))
|
|
// Filter quote event
|
|
.filter(
|
|
(ev) =>
|
|
!ev.tags.filter((t) => t[0] === "q" || t[3] === "mention").length,
|
|
);
|
|
|
|
if (events.length > 1) {
|
|
const removeQueues = new Set();
|
|
|
|
for (const event of events) {
|
|
const tags = event.tags.filter(
|
|
(t) => t[0] === "e" && t[1] !== this.id,
|
|
);
|
|
|
|
if (tags.length === 1) {
|
|
const index = events.findIndex((ev) => ev.id === tags[0][1]);
|
|
|
|
if (index !== -1) {
|
|
const rootEvent = events[index];
|
|
|
|
if (rootEvent.replies?.length) {
|
|
rootEvent.replies.push(event);
|
|
} else {
|
|
rootEvent.replies = [event];
|
|
}
|
|
|
|
// Add current event to queue
|
|
removeQueues.add(event.id);
|
|
|
|
continue;
|
|
}
|
|
}
|
|
|
|
for (const tag of tags) {
|
|
const id = tag[1];
|
|
const rootIndex = events.findIndex((ev) => ev.id === id);
|
|
|
|
if (rootIndex !== -1) {
|
|
const rootEvent = events[rootIndex];
|
|
|
|
if (rootEvent.replies?.length) {
|
|
const childIndex = rootEvent.replies.findIndex(
|
|
(ev) => ev.id === id,
|
|
);
|
|
|
|
if (childIndex !== -1) {
|
|
const childEvent = rootEvent.replies[rootIndex];
|
|
|
|
if (childEvent.replies?.length) {
|
|
childEvent.replies.push(event);
|
|
} else {
|
|
childEvent.replies = [event];
|
|
}
|
|
|
|
// Add current event to queue
|
|
removeQueues.add(event.id);
|
|
}
|
|
} else {
|
|
rootEvent.replies = [event];
|
|
// Add current event to queue
|
|
removeQueues.add(event.id);
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
return events.filter((ev) => !removeQueues.has(ev.id));
|
|
}
|
|
|
|
return events;
|
|
} else {
|
|
console.error(query.error);
|
|
return [];
|
|
}
|
|
}
|
|
|
|
public async listenEventReply() {
|
|
const query = await commands.listenEventReply(this.id);
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
public async unlistenEventReply() {
|
|
const query = await commands.unlisten(this.id);
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
public async zap(amount: number, message: string) {
|
|
const query = await commands.zapEvent(this.id, amount.toString(), message);
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
public async idAsBech32() {
|
|
const query = await commands.eventToBech32(this.id);
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
public async pubkeyAsBech32() {
|
|
const query = await commands.userToBech32(this.pubkey);
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
public async repost() {
|
|
const query = await commands.repost(JSON.stringify(this.#raw));
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
static async publish(
|
|
content: string,
|
|
warning?: string,
|
|
difficulty?: number,
|
|
reply_to?: string,
|
|
root_to?: string,
|
|
) {
|
|
let query: Result<string, string>;
|
|
|
|
if (reply_to) {
|
|
query = await commands.reply(content, reply_to, root_to);
|
|
} else {
|
|
query = await commands.publish(content, warning, difficulty);
|
|
}
|
|
|
|
if (query.status === "ok") {
|
|
return query.data;
|
|
} else {
|
|
throw new Error(query.error);
|
|
}
|
|
}
|
|
|
|
static async build(event: NostrEvent) {
|
|
const query = await commands.getEventMeta(event.content);
|
|
|
|
if (query.status === "ok") {
|
|
event.meta = query.data;
|
|
return new LumeEvent(event);
|
|
} else {
|
|
return new LumeEvent(event);
|
|
}
|
|
}
|
|
|
|
static from(raw: string, parsed?: Meta) {
|
|
const nostrEvent: NostrEvent = JSON.parse(raw);
|
|
|
|
if (parsed) {
|
|
nostrEvent.meta = parsed;
|
|
} else {
|
|
nostrEvent.meta = null;
|
|
}
|
|
|
|
return new LumeEvent(nostrEvent);
|
|
}
|
|
}
|