aboutsummaryrefslogtreecommitdiffhomepage
path: root/frontend/src/atoms/articles.ts
diff options
context:
space:
mode:
authornsfisis <nsfisis@gmail.com>2026-02-14 12:17:23 +0900
committernsfisis <nsfisis@gmail.com>2026-02-14 12:17:23 +0900
commitfffd36268a216044523c3f5227c3d375608c36dc (patch)
treeb289735cb9d478af763775af9b15214b9595e747 /frontend/src/atoms/articles.ts
parent2889b562e64993482bd13fd806af8ed0865bab8b (diff)
downloadfeedaka-fffd36268a216044523c3f5227c3d375608c36dc.tar.gz
feedaka-fffd36268a216044523c3f5227c3d375608c36dc.tar.zst
feedaka-fffd36268a216044523c3f5227c3d375608c36dc.zip
refactor(frontend): migrate state management to jotai and jotai-tanstack-query
Replace React Context + manual useEffect data fetching with jotai atoms for state management and jotai-tanstack-query for server state caching. - Add jotai, jotai-tanstack-query, @tanstack/query-core dependencies - Create atoms for auth (primitive + action), feeds (suspense query), and articles (infinite query with cursor-based pagination) - Wire up Provider, HydrateQueryClient, and StoreInitializer in main.tsx - Migrate all components from useAuth() context to jotai atoms - Replace manual fetch logic in FeedSidebar/FeedList with feedsAtom - Replace usePaginatedArticles hook with articlesInfiniteAtom - Add queryClient.invalidateQueries() after mutations for automatic cache refresh - Add ErrorBoundary and LoadingSpinner components for Suspense support - Remove callback prop chains (onFeedAdded, onFeedChanged, etc.) in favor of query invalidation Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
Diffstat (limited to 'frontend/src/atoms/articles.ts')
-rw-r--r--frontend/src/atoms/articles.ts46
1 files changed, 46 insertions, 0 deletions
diff --git a/frontend/src/atoms/articles.ts b/frontend/src/atoms/articles.ts
new file mode 100644
index 0000000..46c57a4
--- /dev/null
+++ b/frontend/src/atoms/articles.ts
@@ -0,0 +1,46 @@
+import type { InfiniteData } from "@tanstack/query-core";
+import { atom } from "jotai";
+import { atomWithInfiniteQuery } from "jotai-tanstack-query";
+import type { components } from "../api/generated";
+import { api } from "../services/api-client";
+
+type ArticleConnection = components["schemas"]["ArticleConnection"];
+
+export const articleViewAtom = atom<"read" | "unread">("unread");
+export const articleFeedFilterAtom = atom<string | null>(null);
+
+export const articlesInfiniteAtom = atomWithInfiniteQuery<
+ ArticleConnection,
+ Error,
+ InfiniteData<ArticleConnection, string | null>,
+ string[],
+ string | null
+>((get) => {
+ const view = get(articleViewAtom);
+ const feedId = get(articleFeedFilterAtom);
+
+ return {
+ queryKey: ["articles", view, feedId ?? "all"],
+ queryFn: async ({ pageParam }) => {
+ const query: { feedId?: string; after?: string } = {};
+ if (feedId) query.feedId = feedId;
+ if (pageParam) query.after = pageParam;
+
+ if (view === "read") {
+ const { data } = await api.GET("/api/articles/read", {
+ params: { query },
+ });
+ return data ?? { articles: [], pageInfo: { hasNextPage: false } };
+ }
+ const { data } = await api.GET("/api/articles/unread", {
+ params: { query },
+ });
+ return data ?? { articles: [], pageInfo: { hasNextPage: false } };
+ },
+ initialPageParam: null as string | null,
+ getNextPageParam: (lastPage: ArticleConnection) =>
+ lastPage.pageInfo.hasNextPage
+ ? (lastPage.pageInfo.endCursor ?? null)
+ : null,
+ };
+});