mirror of
https://github.com/hoornet/vega.git
synced 2026-05-07 12:49:13 -07:00
Article discovery feed with Latest/Following tabs, article search (NIP-50 + hashtag for kind 30023), Notes/Articles tab on profiles, reading time + bookmark + like buttons on article reader. Event passed directly from card to reader to avoid relay re-fetch failures.
69 lines
2.6 KiB
TypeScript
69 lines
2.6 KiB
TypeScript
import { create } from "zustand";
|
|
|
|
import { NDKEvent } from "@nostr-dev-kit/ndk";
|
|
|
|
type View = "feed" | "search" | "relays" | "settings" | "profile" | "thread" | "article-editor" | "article" | "articles" | "about" | "zaps" | "dm" | "notifications" | "bookmarks";
|
|
type FeedTab = "global" | "following" | "trending";
|
|
|
|
interface UIState {
|
|
currentView: View;
|
|
sidebarCollapsed: boolean;
|
|
selectedPubkey: string | null;
|
|
selectedNote: NDKEvent | null;
|
|
previousView: View;
|
|
feedTab: FeedTab;
|
|
pendingSearch: string | null;
|
|
pendingDMPubkey: string | null;
|
|
pendingArticleNaddr: string | null;
|
|
pendingArticleEvent: NDKEvent | null;
|
|
showHelp: boolean;
|
|
feedLanguageFilter: string | null;
|
|
setView: (view: View) => void;
|
|
setFeedTab: (tab: FeedTab) => void;
|
|
openProfile: (pubkey: string) => void;
|
|
openThread: (note: NDKEvent, from: View) => void;
|
|
openSearch: (query: string) => void;
|
|
openDM: (pubkey: string) => void;
|
|
openArticle: (naddr: string, event?: NDKEvent) => void;
|
|
goBack: () => void;
|
|
setFeedLanguageFilter: (filter: string | null) => void;
|
|
toggleSidebar: () => void;
|
|
toggleHelp: () => void;
|
|
}
|
|
|
|
const SIDEBAR_KEY = "wrystr_sidebar_collapsed";
|
|
|
|
export const useUIStore = create<UIState>((set, _get) => ({
|
|
currentView: "feed",
|
|
sidebarCollapsed: localStorage.getItem(SIDEBAR_KEY) === "true",
|
|
selectedPubkey: null,
|
|
selectedNote: null,
|
|
previousView: "feed",
|
|
feedTab: "global",
|
|
pendingSearch: null,
|
|
pendingDMPubkey: null,
|
|
pendingArticleNaddr: null,
|
|
pendingArticleEvent: null,
|
|
showHelp: false,
|
|
feedLanguageFilter: null,
|
|
setView: (currentView) => set({ currentView }),
|
|
setFeedTab: (feedTab) => set({ feedTab }),
|
|
openProfile: (pubkey) => set((s) => ({ currentView: "profile", selectedPubkey: pubkey, previousView: s.currentView as View })),
|
|
openThread: (note, from) => set({ currentView: "thread", selectedNote: note, previousView: from }),
|
|
openSearch: (query) => set({ currentView: "search", pendingSearch: query }),
|
|
openDM: (pubkey) => set({ currentView: "dm", pendingDMPubkey: pubkey }),
|
|
openArticle: (naddr, event) => set((s) => ({ currentView: "article", pendingArticleNaddr: naddr, pendingArticleEvent: event ?? null, previousView: s.currentView as View })),
|
|
goBack: () => set((s) => ({
|
|
showHelp: false,
|
|
currentView: s.previousView !== s.currentView ? s.previousView : "feed",
|
|
selectedNote: null,
|
|
})),
|
|
setFeedLanguageFilter: (feedLanguageFilter) => set({ feedLanguageFilter }),
|
|
toggleSidebar: () => set((s) => {
|
|
const next = !s.sidebarCollapsed;
|
|
localStorage.setItem(SIDEBAR_KEY, String(next));
|
|
return { sidebarCollapsed: next };
|
|
}),
|
|
toggleHelp: () => set((s) => ({ showHelp: !s.showHelp })),
|
|
}));
|