Files
vega/src/stores/ui.ts

130 lines
5.1 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" | "media" | "podcasts" | "about" | "zaps" | "dm" | "notifications" | "bookmarks" | "hashtag" | "follows";
type FeedTab = "global" | "following" | "trending";
interface ViewStackEntry {
view: View;
selectedNote: NDKEvent | null;
selectedPubkey: string | null;
}
const MAX_STACK = 20;
interface UIState {
currentView: View;
sidebarCollapsed: boolean;
selectedPubkey: string | null;
selectedNote: NDKEvent | null;
previousView: View;
viewStack: ViewStackEntry[];
feedTab: FeedTab;
pendingSearch: string | null;
pendingDMPubkey: string | null;
pendingArticleNaddr: string | null;
pendingArticleEvent: NDKEvent | null;
pendingHashtag: string | null;
showHelp: boolean;
showDebugPanel: boolean;
feedLanguageFilter: string | null;
followsTab: "followers" | "following";
fontSize: number;
themeId: string;
setView: (view: View) => void;
setFollowsTab: (tab: "followers" | "following") => void;
setFeedTab: (tab: FeedTab) => void;
openProfile: (pubkey: string) => void;
openThread: (note: NDKEvent, from?: View) => void;
openSearch: (query: string) => void;
openHashtag: (tag: string) => void;
openDM: (pubkey: string) => void;
openArticle: (naddr: string, event?: NDKEvent) => void;
goBack: () => void;
setFeedLanguageFilter: (filter: string | null) => void;
setFontSize: (size: number) => void;
setTheme: (id: string) => void;
toggleSidebar: () => void;
toggleHelp: () => void;
toggleDebugPanel: () => void;
}
const SIDEBAR_KEY = "wrystr_sidebar_collapsed";
const FONT_SIZE_KEY = "wrystr_font_size";
const THEME_KEY = "wrystr_theme";
const SCRIPT_FILTER_KEY = "wrystr_script_filter";
export const useUIStore = create<UIState>((set, _get) => ({
currentView: "feed",
sidebarCollapsed: localStorage.getItem(SIDEBAR_KEY) === "true",
selectedPubkey: null,
selectedNote: null,
previousView: "feed",
viewStack: [],
feedTab: "global",
pendingSearch: null,
pendingDMPubkey: null,
pendingArticleNaddr: null,
pendingArticleEvent: null,
pendingHashtag: null,
showHelp: false,
showDebugPanel: false,
feedLanguageFilter: localStorage.getItem(SCRIPT_FILTER_KEY) || null,
followsTab: "followers",
fontSize: parseInt(localStorage.getItem(FONT_SIZE_KEY) || "14", 10),
themeId: localStorage.getItem(THEME_KEY) || "midnight",
setView: (currentView) => set({ currentView }),
setFeedTab: (feedTab) => set({ feedTab }),
setFollowsTab: (followsTab) => set({ followsTab }),
openProfile: (pubkey) => set((s) => {
const stack = [...s.viewStack, { view: s.currentView, selectedNote: s.selectedNote, selectedPubkey: s.selectedPubkey }].slice(-MAX_STACK);
return { currentView: "profile", selectedPubkey: pubkey, previousView: s.currentView as View, viewStack: stack };
}),
openThread: (note, _from) => set((s) => {
const stack = [...s.viewStack, { view: s.currentView, selectedNote: s.selectedNote, selectedPubkey: s.selectedPubkey }].slice(-MAX_STACK);
return { currentView: "thread", selectedNote: note, previousView: s.currentView as View, viewStack: stack };
}),
openSearch: (query) => set({ currentView: "search", pendingSearch: query }),
openHashtag: (tag) => set((s) => {
const stack = [...s.viewStack, { view: s.currentView, selectedNote: s.selectedNote, selectedPubkey: s.selectedPubkey }].slice(-MAX_STACK);
return { currentView: "hashtag", pendingHashtag: tag, previousView: s.currentView as View, viewStack: stack };
}),
openDM: (pubkey) => set({ currentView: "dm", pendingDMPubkey: pubkey }),
openArticle: (naddr, event) => set((s) => {
const stack = [...s.viewStack, { view: s.currentView, selectedNote: s.selectedNote, selectedPubkey: s.selectedPubkey }].slice(-MAX_STACK);
return { currentView: "article", pendingArticleNaddr: naddr, pendingArticleEvent: event ?? null, previousView: s.currentView as View, viewStack: stack };
}),
goBack: () => set((s) => {
const stack = [...s.viewStack];
const prev = stack.pop();
if (prev) {
return { showHelp: false, currentView: prev.view, selectedNote: prev.selectedNote, selectedPubkey: prev.selectedPubkey, viewStack: stack };
}
return { showHelp: false, currentView: "feed", selectedNote: null, viewStack: [] };
}),
setFeedLanguageFilter: (feedLanguageFilter) => {
if (feedLanguageFilter) {
localStorage.setItem(SCRIPT_FILTER_KEY, feedLanguageFilter);
} else {
localStorage.removeItem(SCRIPT_FILTER_KEY);
}
set({ feedLanguageFilter });
},
setFontSize: (fontSize) => {
localStorage.setItem(FONT_SIZE_KEY, String(fontSize));
set({ fontSize });
},
setTheme: (themeId) => {
localStorage.setItem(THEME_KEY, themeId);
set({ themeId });
},
toggleSidebar: () => set((s) => {
const next = !s.sidebarCollapsed;
localStorage.setItem(SIDEBAR_KEY, String(next));
return { sidebarCollapsed: next };
}),
toggleHelp: () => set((s) => ({ showHelp: !s.showHelp })),
toggleDebugPanel: () => set((s) => ({ showDebugPanel: !s.showDebugPanel })),
}));