lots of hooks

This commit is contained in:
2024-08-30 11:25:23 -06:00
parent d975add636
commit 09f911c18a
12 changed files with 219 additions and 107 deletions

View File

@@ -1,5 +1,5 @@
"use client"; "use client";
import { useLocalCourseNamesQuery } from "@/hooks/localCoursesHooks"; import { useLocalCourseNamesQuery } from "@/hooks/localCourse/localCoursesHooks";
import Link from "next/link"; import Link from "next/link";
export default function CourseList() { export default function CourseList() {

View File

@@ -1,6 +1,7 @@
"use client"; "use client";
import { useLocalCourseSettingsQuery } from "@/hooks/localCoursesHooks"; import { useLocalCourseSettingsQuery } from "@/hooks/localCourse/localCoursesHooks";
export default function CourseSettings({ courseName }: { courseName: string }) { export default function CourseSettings({ courseName }: { courseName: string }) {
const { data: settings } = useLocalCourseSettingsQuery(courseName); const { data: settings } = useLocalCourseSettingsQuery(courseName);

View File

@@ -50,6 +50,7 @@ export default function CourseContextProvider({
return ( return (
<CourseContext.Provider <CourseContext.Provider
value={{ value={{
courseName: localCourseName,
startItemDrag: (d) => { startItemDrag: (d) => {
setItemBeingDragged(d); setItemBeingDragged(d);
}, },

View File

@@ -8,15 +8,17 @@ export interface DraggableItem {
} }
export interface CourseContextInterface { export interface CourseContextInterface {
courseName: string;
startItemDrag: (dragging: DraggableItem) => void; startItemDrag: (dragging: DraggableItem) => void;
endItemDrag: () => void; endItemDrag: () => void;
itemDrop: (droppedOnDay?: Date) => void; itemDrop: (droppedOnDay?: Date) => void;
} }
const defaultValue: CourseContextInterface = { const defaultValue: CourseContextInterface = {
startItemDrag: () => {}, startItemDrag: () => { },
endItemDrag: () => {}, endItemDrag: () => { },
itemDrop: () => {}, itemDrop: () => { },
courseName: ""
}; };
export const CourseContext = export const CourseContext =

View File

@@ -2,14 +2,50 @@ import { IModuleItem } from "@/models/local/IModuleItem";
import { LocalModule } from "@/models/local/localModules"; import { LocalModule } from "@/models/local/localModules";
import { getDateFromStringOrThrow } from "@/models/local/timeUtils"; import { getDateFromStringOrThrow } from "@/models/local/timeUtils";
import React, { useState } from "react"; import React, { useState } from "react";
import { useCourseContext } from "../context/courseContext";
import {
useAssignmentNamesQuery,
useAssignmentsQueries,
} from "@/hooks/localCourse/assignmentHooks";
import {
useQuizNamesQuery,
useQuizzesQueries,
} from "@/hooks/localCourse/quizHooks";
import {
usePageNamesQuery,
usePagesQueries,
} from "@/hooks/localCourse/pageHooks";
export default function ExpandableModule({
moduleName,
}: {
moduleName: string;
}) {
const { courseName } = useCourseContext();
const { data: assignmentNames } = useAssignmentNamesQuery(
courseName,
moduleName
);
const { data: assignments } = useAssignmentsQueries(
courseName,
moduleName,
assignmentNames
);
const { data: quizNames } = useQuizNamesQuery(courseName, moduleName);
const { data: quizzes } = useQuizzesQueries(
courseName,
moduleName,
quizNames
);
const { data: pageNames } = usePageNamesQuery(courseName, moduleName);
const { data: pages } = usePagesQueries(courseName, moduleName, pageNames);
export default function ExpandableModule({ module }: { module: LocalModule }) {
const [expanded, setExpanded] = useState(false); const [expanded, setExpanded] = useState(false);
const moduleItems: { const moduleItems: {
type: "assignment" | "quiz" | "page"; type: "assignment" | "quiz" | "page";
item: IModuleItem; item: IModuleItem;
}[] = module.assignments }[] = assignments
.map( .map(
( (
a a
@@ -21,8 +57,8 @@ export default function ExpandableModule({ module }: { module: LocalModule }) {
item: a, item: a,
}) })
) )
.concat(module.quizzes.map((q) => ({ type: "quiz", item: q }))) .concat(quizzes.map((q) => ({ type: "quiz", item: q })))
.concat(module.pages.map((p) => ({ type: "page", item: p }))) .concat(pages.map((p) => ({ type: "page", item: p })))
.sort( .sort(
(a, b) => (a, b) =>
getDateFromStringOrThrow( getDateFromStringOrThrow(
@@ -42,7 +78,7 @@ export default function ExpandableModule({ module }: { module: LocalModule }) {
role="button" role="button"
onClick={() => setExpanded((e) => !e)} onClick={() => setExpanded((e) => !e)}
> >
{module.name} {moduleName}
</div> </div>
<div <div
className={ className={

View File

@@ -1,14 +1,17 @@
"use client"; "use client";
import { useModuleNamesQuery } from "@/hooks/localCourse/localCoursesHooks";
import { useCourseContext } from "../context/courseContext"; import { useCourseContext } from "../context/courseContext";
import ExpandableModule from "./ExpandableModule"; import ExpandableModule from "./ExpandableModule";
export default function ModuleList() { export default function ModuleList() {
const { courseName } = useCourseContext();
const { data: moduleNames } = useModuleNamesQuery(courseName);
return ( return (
<div> <div>
modules here modules here
{/* {modules.map((m) => ( {moduleNames.map((m) => (
<ExpandableModule key={m.name} module={m} /> <ExpandableModule key={m} moduleName={m}/>
))} */} ))}
</div> </div>
); );
} }

View File

@@ -1,5 +1,5 @@
import { QueryClient } from "@tanstack/react-query"; import { QueryClient } from "@tanstack/react-query";
import { localCourseKeys } from "./localCoursesHooks"; import { localCourseKeys } from "./localCourse/localCoursesHooks";
import { fileStorageService } from "@/services/fileStorage/fileStorageService"; import { fileStorageService } from "@/services/fileStorage/fileStorageService";
export const hydrateCourses = async (queryClient: QueryClient) => { export const hydrateCourses = async (queryClient: QueryClient) => {

View File

@@ -0,0 +1,59 @@
import axios from "axios";
import { localCourseKeys } from "./localCoursesHooks";
import { LocalAssignment } from "@/models/local/assignmnet/localAssignment";
import { useSuspenseQuery, useSuspenseQueries } from "@tanstack/react-query";
export const useAssignmentNamesQuery = (
courseName: string,
moduleName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.moduleAssignmentNames(courseName, moduleName),
queryFn: async (): Promise<string[]> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/assignments`;
const response = await axios.get(url);
return response.data;
},
});
const getAssignmentQueryConfig = (
courseName: string,
moduleName: string,
assignmentName: string
) => {
return {
queryKey: localCourseKeys.assignment(
courseName,
moduleName,
assignmentName
),
queryFn: async (): Promise<LocalAssignment> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/assignments/${assignmentName}`;
const response = await axios.get(url);
return response.data;
},
};
};
export const useAssignmentQuery = (
courseName: string,
moduleName: string,
assignmentName: string
) =>
useSuspenseQuery(
getAssignmentQueryConfig(courseName, moduleName, assignmentName)
);
export const useAssignmentsQueries = (
courseName: string,
moduleName: string,
assignmentNames: string[]
) =>
useSuspenseQueries({
queries: assignmentNames.map((name) =>
getAssignmentQueryConfig(courseName, moduleName, name)
),
combine: (results) => ({
data: results.map((r) => r.data),
pending: results.some((r) => r.isPending),
}),
});

View File

@@ -1,12 +1,5 @@
import { LocalAssignment } from "@/models/local/assignmnet/localAssignment"; import { LocalCourseSettings } from "@/models/local/localCourse";
import { LocalCourse, LocalCourseSettings } from "@/models/local/localCourse";
import { LocalModule } from "@/models/local/localModules";
import { LocalCoursePage } from "@/models/local/page/localCoursePage";
import { LocalQuiz } from "@/models/local/quiz/localQuiz";
import { import {
useMutation,
useQueryClient,
useSuspenseQueries,
useSuspenseQuery, useSuspenseQuery,
} from "@tanstack/react-query"; } from "@tanstack/react-query";
import axios from "axios"; import axios from "axios";
@@ -97,89 +90,7 @@ export const useModuleNamesQuery = (courseName: string) =>
}, },
}); });
export const useModuleAssignmentNamesQuery = (
courseName: string,
moduleName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.moduleAssignmentNames(courseName, moduleName),
queryFn: async (): Promise<string[]> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/assignments`;
const response = await axios.get(url);
return response.data;
},
});
export const useModuleQuizNamesQuery = (
courseName: string,
moduleName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.moduleQuizzeNames(courseName, moduleName),
queryFn: async (): Promise<string[]> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/quizzes`;
const response = await axios.get(url);
return response.data;
},
});
export const useModulePageNamesQuery = (
courseName: string,
moduleName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.modulePageNames(courseName, moduleName),
queryFn: async (): Promise<string[]> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/pages`;
const response = await axios.get(url);
return response.data;
},
});
export const useAssignmentQuery = (
courseName: string,
moduleName: string,
assignmentName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.assignment(
courseName,
moduleName,
assignmentName
),
queryFn: async (): Promise<LocalAssignment> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/assignments/${assignmentName}`;
const response = await axios.get(url);
return response.data;
},
});
export const useQuizQuery = (
courseName: string,
moduleName: string,
quizName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.quiz(courseName, moduleName, quizName),
queryFn: async (): Promise<LocalQuiz> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/quizzes/${quizName}`;
const response = await axios.get(url);
return response.data;
},
});
export const usePageQuery = (
courseName: string,
moduleName: string,
pageName: string
) =>
useSuspenseQuery({
queryKey: localCourseKeys.quiz(courseName, moduleName, pageName),
queryFn: async (): Promise<LocalCoursePage> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/pages/${pageName}`;
const response = await axios.get(url);
return response.data;
},
});
// export const useUpdateCourseMutation = (courseName: string) => { // export const useUpdateCourseMutation = (courseName: string) => {
// const queryClient = useQueryClient(); // const queryClient = useQueryClient();

View File

@@ -0,0 +1,49 @@
import { LocalCoursePage } from "@/models/local/page/localCoursePage";
import { useSuspenseQueries, useSuspenseQuery } from "@tanstack/react-query";
import axios from "axios";
import { localCourseKeys } from "./localCoursesHooks";
export const usePageNamesQuery = (courseName: string, moduleName: string) =>
useSuspenseQuery({
queryKey: localCourseKeys.modulePageNames(courseName, moduleName),
queryFn: async (): Promise<string[]> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/pages`;
const response = await axios.get(url);
return response.data;
},
});
export const usePageQuery = (
courseName: string,
moduleName: string,
pageName: string
) => useSuspenseQuery(getPageQueryConfig(courseName, moduleName, pageName));
export const usePagesQueries = (
courseName: string,
moduleName: string,
pageNames: string[]
) =>
useSuspenseQueries({
queries: pageNames.map((name) =>
getPageQueryConfig(courseName, moduleName, name)
),
combine: (results) => ({
data: results.map((r) => r.data),
pending: results.some((r) => r.isPending),
}),
});
function getPageQueryConfig(
courseName: string,
moduleName: string,
pageName: string
) {
return {
queryKey: localCourseKeys.quiz(courseName, moduleName, pageName),
queryFn: async (): Promise<LocalCoursePage> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/pages/${pageName}`;
const response = await axios.get(url);
return response.data;
},
};
}

View File

@@ -0,0 +1,50 @@
import { LocalQuiz } from "@/models/local/quiz/localQuiz";
import { useSuspenseQueries, useSuspenseQuery } from "@tanstack/react-query";
import axios from "axios";
import { localCourseKeys } from "./localCoursesHooks";
export const useQuizNamesQuery = (courseName: string, moduleName: string) =>
useSuspenseQuery({
queryKey: localCourseKeys.moduleQuizzeNames(courseName, moduleName),
queryFn: async (): Promise<string[]> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/quizzes`;
const response = await axios.get(url);
return response.data;
},
});
export const useQuizQuery = (
courseName: string,
moduleName: string,
quizName: string
) => useSuspenseQuery(getQuizQueryConfig(courseName, moduleName, quizName));
export const useQuizzesQueries = (
courseName: string,
moduleName: string,
quizNames: string[]
) =>
useSuspenseQueries({
queries: quizNames.map((name) =>
getQuizQueryConfig(courseName, moduleName, name)
),
combine: (results) => ({
data: results.map((r) => r.data),
pending: results.some((r) => r.isPending),
}),
});
function getQuizQueryConfig(
courseName: string,
moduleName: string,
quizName: string
) {
return {
queryKey: localCourseKeys.quiz(courseName, moduleName, quizName),
queryFn: async (): Promise<LocalQuiz> => {
const url = `/api/courses/${courseName}/modules/${moduleName}/quizzes/${quizName}`;
const response = await axios.get(url);
return response.data;
},
};
}

View File

@@ -127,7 +127,7 @@ export const fileStorageService = {
courseName, courseName,
moduleName, moduleName,
"assignments", "assignments",
assignmentName + ".md" assignmentName
); );
const rawFile = (await fs.readFile(filePath, "utf-8")).replace( const rawFile = (await fs.readFile(filePath, "utf-8")).replace(
/\r\n/g, /\r\n/g,
@@ -142,7 +142,7 @@ export const fileStorageService = {
courseName, courseName,
moduleName, moduleName,
"quizzes", "quizzes",
quizName + ".md" quizName
); );
const rawFile = (await fs.readFile(filePath, "utf-8")).replace( const rawFile = (await fs.readFile(filePath, "utf-8")).replace(
/\r\n/g, /\r\n/g,
@@ -157,7 +157,7 @@ export const fileStorageService = {
courseName, courseName,
moduleName, moduleName,
"pages", "pages",
pageName + ".md" pageName
); );
const rawFile = (await fs.readFile(filePath, "utf-8")).replace( const rawFile = (await fs.readFile(filePath, "utf-8")).replace(
/\r\n/g, /\r\n/g,