aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/client/pages/StudyPage.test.tsx
diff options
context:
space:
mode:
Diffstat (limited to 'src/client/pages/StudyPage.test.tsx')
-rw-r--r--src/client/pages/StudyPage.test.tsx114
1 files changed, 114 insertions, 0 deletions
diff --git a/src/client/pages/StudyPage.test.tsx b/src/client/pages/StudyPage.test.tsx
index 6e605c9..9c48083 100644
--- a/src/client/pages/StudyPage.test.tsx
+++ b/src/client/pages/StudyPage.test.tsx
@@ -601,6 +601,120 @@ describe("StudyPage", () => {
});
});
+ describe("Navigation flushes pending review", () => {
+ function getSessionCompleteButton(name: string) {
+ const container = screen.getByTestId("session-complete");
+ const buttons = container.querySelectorAll("button");
+ for (const button of buttons) {
+ if (button.textContent?.includes(name)) return button;
+ }
+ throw new Error(`Button "${name}" not found in session-complete`);
+ }
+
+ it("flushes pending review when clicking 'Back to Deck' on session complete", async () => {
+ const user = userEvent.setup();
+
+ mockStudyPost.mockResolvedValue({
+ card: { ...mockFirstCard, reps: 1 },
+ });
+
+ renderWithProviders({
+ initialStudyData: { deck: mockDeck, cards: [mockFirstCard] },
+ });
+
+ // Review the only card
+ await user.click(screen.getByTestId("card-container"));
+ await user.click(screen.getByTestId("rating-3"));
+
+ await waitFor(() => {
+ expect(screen.getByTestId("session-complete")).toBeDefined();
+ });
+
+ // API should NOT have been called yet (still pending)
+ expect(mockStudyPost).not.toHaveBeenCalled();
+
+ // Click "Back to Deck" in session complete area
+ await user.click(getSessionCompleteButton("Back to Deck"));
+
+ // Now the pending review should have been flushed
+ await waitFor(() => {
+ expect(mockStudyPost).toHaveBeenCalledTimes(1);
+ });
+ expect(mockStudyPost).toHaveBeenCalledWith(
+ expect.objectContaining({
+ param: { deckId: "deck-1", cardId: "card-1" },
+ json: expect.objectContaining({ rating: 3 }),
+ }),
+ );
+ });
+
+ it("flushes pending review when clicking 'All Decks' on session complete", async () => {
+ const user = userEvent.setup();
+
+ mockStudyPost.mockResolvedValue({
+ card: { ...mockFirstCard, reps: 1 },
+ });
+
+ renderWithProviders({
+ initialStudyData: { deck: mockDeck, cards: [mockFirstCard] },
+ });
+
+ // Review the only card
+ await user.click(screen.getByTestId("card-container"));
+ await user.click(screen.getByTestId("rating-3"));
+
+ await waitFor(() => {
+ expect(screen.getByTestId("session-complete")).toBeDefined();
+ });
+
+ expect(mockStudyPost).not.toHaveBeenCalled();
+
+ // Click "All Decks"
+ await user.click(getSessionCompleteButton("All Decks"));
+
+ // Pending review should have been flushed
+ await waitFor(() => {
+ expect(mockStudyPost).toHaveBeenCalledTimes(1);
+ });
+ expect(mockStudyPost).toHaveBeenCalledWith(
+ expect.objectContaining({
+ param: { deckId: "deck-1", cardId: "card-1" },
+ json: expect.objectContaining({ rating: 3 }),
+ }),
+ );
+ });
+
+ it("navigates even if flush fails", async () => {
+ const user = userEvent.setup();
+
+ mockStudyPost.mockRejectedValue(new Error("Network error"));
+
+ renderWithProviders({
+ initialStudyData: { deck: mockDeck, cards: [mockFirstCard] },
+ });
+
+ // Review the only card
+ await user.click(screen.getByTestId("card-container"));
+ await user.click(screen.getByTestId("rating-3"));
+
+ await waitFor(() => {
+ expect(screen.getByTestId("session-complete")).toBeDefined();
+ });
+
+ // Click "Back to Deck" — flush will fail but navigation should still happen
+ await user.click(getSessionCompleteButton("Back to Deck"));
+
+ await waitFor(() => {
+ expect(mockStudyPost).toHaveBeenCalledTimes(1);
+ });
+
+ // Buttons should be disabled during navigation (isNavigating = true)
+ await waitFor(() => {
+ expect(getSessionCompleteButton("Back to Deck").disabled).toBe(true);
+ });
+ });
+ });
+
describe("Undo", () => {
it("does not show undo button before any rating", () => {
renderWithProviders({