trying to add an assignment

This commit is contained in:
2024-09-18 14:15:38 -06:00
parent 3862743e4c
commit de0b1ed719
7 changed files with 248 additions and 104 deletions

View File

@@ -7,6 +7,17 @@ import {
import { localAssignmentMarkdown } from "@/models/local/assignment/localAssignment"; import { localAssignmentMarkdown } from "@/models/local/assignment/localAssignment";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import AssignmentPreview from "./AssignmentPreview"; import AssignmentPreview from "./AssignmentPreview";
import { getCourseUrl } from "@/services/urlUtils";
import Link from "next/link";
import { useCourseContext } from "@/app/course/[courseName]/context/courseContext";
import { useLocalCourseSettingsQuery } from "@/hooks/localCourse/localCoursesHooks";
import {
useAddAssignmentToCanvasMutation,
useCanvasAssignmentsQuery,
useDeleteAssignmentFromCanvasMutation,
} from "@/hooks/canvas/canvasAssignmentHooks";
import { Spinner } from "@/components/Spinner";
import { baseCanvasUrl } from "@/services/canvas/canvasServiceUtils";
export default function EditAssignment({ export default function EditAssignment({
moduleName, moduleName,
@@ -68,9 +79,71 @@ export default function EditAssignment({
<AssignmentPreview assignment={assignment} /> <AssignmentPreview assignment={assignment} />
</div> </div>
</div> </div>
<div className="p-5"> <AssignmentButtons
<button>Add to canvas....</button> moduleName={moduleName}
</div> assignmentName={assignmentName}
/>
</div>
);
}
function AssignmentButtons({
moduleName,
assignmentName,
}: {
assignmentName: string;
moduleName: string;
}) {
const { courseName } = useCourseContext();
const { data: settings } = useLocalCourseSettingsQuery();
const { data: canvasAssignments } = useCanvasAssignmentsQuery();
const { data: assignment } = useAssignmentQuery(moduleName, assignmentName);
const addToCanvas = useAddAssignmentToCanvasMutation();
const deleteFromCanvas = useDeleteAssignmentFromCanvasMutation();
const assignmentInCanvas = canvasAssignments.find(
(a) => a.name === assignmentName
);
return (
<div className="p-5 flex flex-row justify-end gap-3">
{(addToCanvas.isPending || deleteFromCanvas.isPending) && <Spinner />}
{assignmentInCanvas && !assignmentInCanvas.published && (
<div className="text-rose-300 my-auto">Not Published</div>
)}
{!assignmentInCanvas && (
<button
disabled={addToCanvas.isPending}
onClick={() => addToCanvas.mutate(assignment)}
>
Add to canvas....
</button>
)}
{assignmentInCanvas && (
<a
className="btn"
target="_blank"
href={`${baseCanvasUrl}/courses/${settings.canvasId}/assignments/${assignmentInCanvas.id}`}
>
View in Canvas
</a>
)}
{assignmentInCanvas && (
<button
className="btn-danger"
disabled={deleteFromCanvas.isPending}
onClick={() =>
deleteFromCanvas.mutate({
canvasAssignmentId: assignmentInCanvas.id,
assignmentName: assignment.name,
})
}
>
Delete from Canvas
</button>
)}
<Link className="btn" href={getCourseUrl(courseName)} shallow={true}>
Go Back
</Link>
</div> </div>
); );
} }

View File

@@ -135,10 +135,10 @@ function QuizButtons({
moduleName: string; moduleName: string;
toggleHelp: () => void; toggleHelp: () => void;
}) { }) {
const { courseName } = useCourseContext();
const { data: settings } = useLocalCourseSettingsQuery();
const { data: canvasQuizzes } = useCanvasQuizzesQuery(); const { data: canvasQuizzes } = useCanvasQuizzesQuery();
const { data: quiz } = useQuizQuery(moduleName, quizName); const { data: quiz } = useQuizQuery(moduleName, quizName);
const { data: settings } = useLocalCourseSettingsQuery();
const { courseName } = useCourseContext();
const addToCanvas = useAddQuizToCanvasMutation(); const addToCanvas = useAddQuizToCanvasMutation();
const deleteFromCanvas = useDeleteQuizFromCanvasMutation(); const deleteFromCanvas = useDeleteQuizFromCanvasMutation();

View File

@@ -46,14 +46,6 @@ export default function InnerMonacoEditor({
} }
}, [onChange, value]); }, [onChange, value]);
useEffect(() => {
window.addEventListener("resize", () => {
if (editorRef.current) {
editorRef.current.layout();
}
});
}, []);
return ( return (
<div <div
className="Editor" className="Editor"

View File

@@ -0,0 +1,90 @@
import { canvasAssignmentService } from "@/services/canvas/canvasAssignmentService";
import { canvasService } from "@/services/canvas/canvasService";
import {
useMutation,
useQueryClient,
useSuspenseQueries,
useSuspenseQuery,
} from "@tanstack/react-query";
import { useLocalCourseSettingsQuery } from "../localCourse/localCoursesHooks";
import { LocalAssignment } from "@/models/local/assignment/localAssignment";
export const canvasAssignmentKeys = {
assignments: (canvasCourseId: number) =>
["canvas", canvasCourseId, "assignments"] as const,
assignment: (canvasCourseId: number, assignmentName: string) =>
["canvas", canvasCourseId, "assignment", assignmentName] as const,
};
export const useCanvasAssignmentsQuery = () => {
const { data: settings } = useLocalCourseSettingsQuery();
return useSuspenseQuery({
queryKey: canvasAssignmentKeys.assignments(settings.canvasId),
queryFn: async () => canvasAssignmentService.getAll(settings.canvasId),
});
};
// export const useCanvasAssignmentsQuery = () => {
// const { data: settings } = useLocalCourseSettingsQuery();
// const { data: allAssignments } = useInnerCanvasAssignmentsQuery();
// return useSuspenseQueries({
// queries: allAssignments.map((a) => ({
// queryKey: canvasAssignmentKeys.assignment(settings.canvasId, a.name),
// queryFn: () => a,
// })),
// combine: (results) => ({
// data: results.map((r) => r.data),
// pending: results.some((r) => r.isPending),
// }),
// });
// };
export const useAddAssignmentToCanvasMutation = () => {
const { data: settings } = useLocalCourseSettingsQuery();
const queryClient = useQueryClient();
return useMutation({
mutationFn: async (assignmnet: LocalAssignment) => {
const assignmentGroup = settings.assignmentGroups.find(
(g) => g.name === assignmnet.localAssignmentGroupName
);
await canvasAssignmentService.create(
settings.canvasId,
assignmnet,
assignmentGroup?.canvasId
);
},
onSuccess: () => {
queryClient.invalidateQueries({
queryKey: canvasAssignmentKeys.assignments(settings.canvasId),
});
},
});
};
export const useDeleteAssignmentFromCanvasMutation = () => {
const { data: settings } = useLocalCourseSettingsQuery();
const queryClient = useQueryClient();
return useMutation({
mutationFn: async ({
canvasAssignmentId,
assignmentName,
}: {
canvasAssignmentId: number;
assignmentName: string;
}) => {
await canvasAssignmentService.delete(
settings.canvasId,
canvasAssignmentId,
assignmentName
);
},
onSuccess: () => {
queryClient.invalidateQueries({
queryKey: canvasAssignmentKeys.assignments(settings.canvasId),
});
},
});
};

View File

@@ -8,7 +8,8 @@ import { canvasQuizService } from "@/services/canvas/canvasQuizService";
import { LocalQuiz } from "@/models/local/quiz/localQuiz"; import { LocalQuiz } from "@/models/local/quiz/localQuiz";
export const canvasQuizKeys = { export const canvasQuizKeys = {
quizzes: (canvasCourseId: number) => ["canvas", canvasCourseId, "quizzes"], quizzes: (canvasCourseId: number) =>
["canvas", canvasCourseId, "quizzes"] as const,
}; };
export const useCanvasQuizzesQuery = () => { export const useCanvasQuizzesQuery = () => {
@@ -29,16 +30,13 @@ export const useAddQuizToCanvasMutation = () => {
const assignmentGroup = settings.assignmentGroups.find( const assignmentGroup = settings.assignmentGroups.find(
(g) => g.name === quiz.localAssignmentGroupName (g) => g.name === quiz.localAssignmentGroupName
); );
console.log("starting");
await canvasQuizService.create( await canvasQuizService.create(
settings.canvasId, settings.canvasId,
quiz, quiz,
assignmentGroup?.canvasId assignmentGroup?.canvasId
); );
console.log("ending");
}, },
onSuccess: () => { onSuccess: () => {
console.log("invalidating");
queryClient.invalidateQueries({ queryClient.invalidateQueries({
queryKey: canvasQuizKeys.quizzes(settings.canvasId), queryKey: canvasQuizKeys.quizzes(settings.canvasId),
}); });

View File

@@ -6,7 +6,6 @@ import {
useSuspenseQueries, useSuspenseQueries,
useSuspenseQuery, useSuspenseQuery,
} from "@tanstack/react-query"; } from "@tanstack/react-query";
import axios from "axios";
import { localCourseKeys } from "./localCourseKeys"; import { localCourseKeys } from "./localCourseKeys";
import { useCourseContext } from "@/app/course/[courseName]/context/courseContext"; import { useCourseContext } from "@/app/course/[courseName]/context/courseContext";
import { axiosClient } from "@/services/axiosUtils"; import { axiosClient } from "@/services/axiosUtils";

View File

@@ -6,6 +6,56 @@ import { markdownToHTMLSafe } from "../htmlMarkdownUtils";
import { CanvasRubricCreationResponse } from "@/models/canvas/assignments/canvasRubricCreationResponse"; import { CanvasRubricCreationResponse } from "@/models/canvas/assignments/canvasRubricCreationResponse";
import { assignmentPoints } from "@/models/local/assignment/utils/assignmentPointsUtils"; import { assignmentPoints } from "@/models/local/assignment/utils/assignmentPointsUtils";
const createRubric = async (
courseId: number,
assignmentCanvasId: number,
localAssignment: LocalAssignment
) => {
const criterion = localAssignment.rubric.map((rubricItem, i) => ({
description: rubricItem.label,
points: rubricItem.points,
ratings: [
{ description: "Full Marks", points: rubricItem.points },
{ description: "No Marks", points: 0 },
],
}));
const rubricBody = {
rubric_association_id: assignmentCanvasId,
rubric: {
title: `Rubric for Assignment: ${localAssignment.name}`,
association_id: assignmentCanvasId,
association_type: "Assignment",
use_for_grading: true,
criteria: criterion,
},
rubric_association: {
association_id: assignmentCanvasId,
association_type: "Assignment",
purpose: "grading",
use_for_grading: true,
},
};
const rubricUrl = `${canvasApi}/courses/${courseId}/rubrics`;
const rubricResponse = await axiosClient.post<CanvasRubricCreationResponse>(
rubricUrl,
rubricBody
);
if (!rubricResponse.data) throw new Error("Failed to create rubric");
const assignmentPointAdjustmentUrl = `${canvasApi}/courses/${courseId}/assignments/${assignmentCanvasId}`;
const assignmentPointAdjustmentBody = {
assignment: { points_possible: assignmentPoints(localAssignment) },
};
await axiosClient.put(
assignmentPointAdjustmentUrl,
assignmentPointAdjustmentBody
);
};
export const canvasAssignmentService = { export const canvasAssignmentService = {
async getAll(courseId: number): Promise<CanvasAssignment[]> { async getAll(courseId: number): Promise<CanvasAssignment[]> {
const url = `${canvasApi}/courses/${courseId}/assignments`; const url = `${canvasApi}/courses/${courseId}/assignments`;
@@ -23,24 +73,22 @@ export const canvasAssignmentService = {
canvasCourseId: number, canvasCourseId: number,
localAssignment: LocalAssignment, localAssignment: LocalAssignment,
canvasAssignmentGroupId?: number canvasAssignmentGroupId?: number
): Promise<number> { ) {
console.log(`Creating assignment: ${localAssignment.name}`); console.log(`Creating assignment: ${localAssignment.name}`);
const url = `${canvasApi}/courses/${canvasCourseId}/assignments`; const url = `${canvasApi}/courses/${canvasCourseId}/assignments`;
const body = { const body = {
assignment: { name: localAssignment.name,
name: localAssignment.name, submission_types: localAssignment.submissionTypes.map((t) =>
submission_types: localAssignment.submissionTypes.map((t) => t.toString()
t.toString() ),
), allowed_extensions: localAssignment.allowedFileUploadExtensions.map((e) =>
allowed_extensions: localAssignment.allowedFileUploadExtensions.map( e.toString()
(e) => e.toString() ),
), description: markdownToHTMLSafe(localAssignment.description),
description: markdownToHTMLSafe(localAssignment.description), due_at: localAssignment.dueAt,
due_at: localAssignment.dueAt, lock_at: localAssignment.lockAt,
lock_at: localAssignment.lockAt, points_possible: assignmentPoints(localAssignment),
points_possible: assignmentPoints(localAssignment), assignment_group_id: canvasAssignmentGroupId,
assignment_group_id: canvasAssignmentGroupId,
},
}; };
const response = await axiosClient.post<CanvasAssignment>(url, body); const response = await axiosClient.post<CanvasAssignment>(url, body);
@@ -48,11 +96,7 @@ export const canvasAssignmentService = {
if (!canvasAssignment) throw new Error("Created Canvas assignment is null"); if (!canvasAssignment) throw new Error("Created Canvas assignment is null");
await this.createRubric( await createRubric(canvasCourseId, canvasAssignment.id, localAssignment);
canvasCourseId,
canvasAssignment.id,
localAssignment
);
return canvasAssignment.id; return canvasAssignment.id;
}, },
@@ -62,28 +106,26 @@ export const canvasAssignmentService = {
canvasAssignmentId: number, canvasAssignmentId: number,
localAssignment: LocalAssignment, localAssignment: LocalAssignment,
canvasAssignmentGroupId?: number canvasAssignmentGroupId?: number
): Promise<void> { ) {
console.log(`Updating assignment: ${localAssignment.name}`); console.log(`Updating assignment: ${localAssignment.name}`);
const url = `${canvasApi}/courses/${courseId}/assignments/${canvasAssignmentId}`; const url = `${canvasApi}/courses/${courseId}/assignments/${canvasAssignmentId}`;
const body = { const body = {
assignment: { name: localAssignment.name,
name: localAssignment.name, submission_types: localAssignment.submissionTypes.map((t) =>
submission_types: localAssignment.submissionTypes.map((t) => t.toString()
t.toString() ),
), allowed_extensions: localAssignment.allowedFileUploadExtensions.map((e) =>
allowed_extensions: localAssignment.allowedFileUploadExtensions.map( e.toString()
(e) => e.toString() ),
), description: markdownToHTMLSafe(localAssignment.description),
description: markdownToHTMLSafe(localAssignment.description), due_at: localAssignment.dueAt,
due_at: localAssignment.dueAt, lock_at: localAssignment.lockAt,
lock_at: localAssignment.lockAt, points_possible: assignmentPoints(localAssignment),
points_possible: assignmentPoints(localAssignment), assignment_group_id: canvasAssignmentGroupId,
assignment_group_id: canvasAssignmentGroupId,
},
}; };
await axiosClient.put(url, body); await axiosClient.put(url, body);
await this.createRubric(courseId, canvasAssignmentId, localAssignment); await createRubric(courseId, canvasAssignmentId, localAssignment);
}, },
async delete( async delete(
@@ -100,54 +142,4 @@ export const canvasAssignmentService = {
throw new Error("Failed to delete assignment"); throw new Error("Failed to delete assignment");
} }
}, },
async createRubric(
courseId: number,
assignmentCanvasId: number,
localAssignment: LocalAssignment
): Promise<void> {
const criterion = localAssignment.rubric.map((rubricItem, i) => ({
description: rubricItem.label,
points: rubricItem.points,
ratings: [
{ description: "Full Marks", points: rubricItem.points },
{ description: "No Marks", points: 0 },
],
}));
const rubricBody = {
rubric_association_id: assignmentCanvasId,
rubric: {
title: `Rubric for Assignment: ${localAssignment.name}`,
association_id: assignmentCanvasId,
association_type: "Assignment",
use_for_grading: true,
criteria: criterion,
},
rubric_association: {
association_id: assignmentCanvasId,
association_type: "Assignment",
purpose: "grading",
use_for_grading: true,
},
};
const rubricUrl = `${canvasApi}/courses/${courseId}/rubrics`;
const rubricResponse = await axiosClient.post<CanvasRubricCreationResponse>(
rubricUrl,
rubricBody
);
if (!rubricResponse.data) throw new Error("Failed to create rubric");
const assignmentPointAdjustmentUrl = `${canvasApi}/courses/${courseId}/assignments/${assignmentCanvasId}`;
const assignmentPointAdjustmentBody = {
assignment: { points_possible: assignmentPoints(localAssignment) },
};
await axiosClient.put(
assignmentPointAdjustmentUrl,
assignmentPointAdjustmentBody
);
},
}; };