Merge remote-tracking branch 'origin/dev' into feat/indieningen-kunnen-posten-en-bekijken-#194
# Conflicts: # backend/tests/setup-tests.ts
This commit is contained in:
		
						commit
						dd2cdf3fe9
					
				
					 46 changed files with 1670 additions and 123 deletions
				
			
		|  | @ -6,7 +6,7 @@ import { Language } from '@dwengo-1/common/util/language'; | ||||||
| import { BadRequestException } from '../exceptions/bad-request-exception.js'; | import { BadRequestException } from '../exceptions/bad-request-exception.js'; | ||||||
| import { NotFoundException } from '../exceptions/not-found-exception.js'; | import { NotFoundException } from '../exceptions/not-found-exception.js'; | ||||||
| import {Group} from "../entities/assignments/group.entity"; | import {Group} from "../entities/assignments/group.entity"; | ||||||
| import {getGroupRepository} from "../data/repositories"; | import {getAssignmentRepository, getGroupRepository} from "../data/repositories"; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Fetch learning paths based on query parameters. |  * Fetch learning paths based on query parameters. | ||||||
|  | @ -27,15 +27,14 @@ export async function getLearningPaths(req: Request, res: Response): Promise<voi | ||||||
|         if (!assignmentNo || !classId) { |         if (!assignmentNo || !classId) { | ||||||
|             throw new BadRequestException('If forGroupNo is specified, assignmentNo and classId must also be specified.'); |             throw new BadRequestException('If forGroupNo is specified, assignmentNo and classId must also be specified.'); | ||||||
|         } |         } | ||||||
|         forGroup = await getGroupRepository().findOne({ |         const assignment = await getAssignmentRepository().findByClassIdAndAssignmentId( | ||||||
|             assignment: { |             classId, parseInt(assignmentNo) | ||||||
|                 id: parseInt(assignmentNo), |         ); | ||||||
|                 within: { |         if (assignment) { | ||||||
|                     classId |             forGroup = await getGroupRepository().findByAssignmentAndGroupNumber( | ||||||
|  |                 assignment, parseInt(forGroupNo) | ||||||
|  |             ) ?? undefined; | ||||||
|         } |         } | ||||||
|             }, |  | ||||||
|             groupNumber: parseInt(forGroupNo) |  | ||||||
|         }) ?? undefined; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     let hruidList; |     let hruidList; | ||||||
|  |  | ||||||
|  | @ -4,7 +4,7 @@ import { | ||||||
|     deleteSubmission, |     deleteSubmission, | ||||||
|     getAllSubmissions, |     getAllSubmissions, | ||||||
|     getSubmission, |     getSubmission, | ||||||
|     getSubmissionsForLearningObjectAndAssignment, |     getSubmissionsForLearningObjectAndAssignment, getSubmissionsForLearningObjectAndGroup, | ||||||
| } from '../services/submissions.js'; | } from '../services/submissions.js'; | ||||||
| import { SubmissionDTO } from '@dwengo-1/common/interfaces/submission'; | import { SubmissionDTO } from '@dwengo-1/common/interfaces/submission'; | ||||||
| import { Language, languageMap } from '@dwengo-1/common/util/language'; | import { Language, languageMap } from '@dwengo-1/common/util/language'; | ||||||
|  | @ -17,13 +17,25 @@ export async function getSubmissionsHandler(req: Request, res: Response): Promis | ||||||
|     const lang = languageMap[req.query.language as string] || Language.Dutch; |     const lang = languageMap[req.query.language as string] || Language.Dutch; | ||||||
|     const version = parseInt(req.query.version as string) ?? 1; |     const version = parseInt(req.query.version as string) ?? 1; | ||||||
| 
 | 
 | ||||||
|     const submissions = await getSubmissionsForLearningObjectAndAssignment( |     let submissions: SubmissionDTO[] | ||||||
|  |     if (req.query.groupId) { | ||||||
|  |         submissions = await getSubmissionsForLearningObjectAndGroup( | ||||||
|  |             loHruid, | ||||||
|  |             lang, | ||||||
|  |             version, | ||||||
|  |             req.query.classId as string, | ||||||
|  |             parseInt(req.query.assignmentId as string), | ||||||
|  |             parseInt(req.query.groupId as string) | ||||||
|  |         ); | ||||||
|  |     } else { | ||||||
|  |         submissions = await getSubmissionsForLearningObjectAndAssignment( | ||||||
|             loHruid, |             loHruid, | ||||||
|             lang, |             lang, | ||||||
|             version, |             version, | ||||||
|             req.query.classId as string, |             req.query.classId as string, | ||||||
|             parseInt(req.query.assignmentId as string) |             parseInt(req.query.assignmentId as string) | ||||||
|     ); |         ) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     res.json(submissions); |     res.json(submissions); | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										66
									
								
								backend/src/controllers/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								backend/src/controllers/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,66 @@ | ||||||
|  | import { Request, Response } from 'express'; | ||||||
|  | import { requireFields } from './error-helper'; | ||||||
|  | import { createInvitation, deleteInvitation, getAllInvitations, getInvitation, updateInvitation } from '../services/teacher-invitations'; | ||||||
|  | import { TeacherInvitationData } from '@dwengo-1/common/interfaces/teacher-invitation'; | ||||||
|  | 
 | ||||||
|  | export async function getAllInvitationsHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const username = req.params.username; | ||||||
|  |     const by = req.query.sent === 'true'; | ||||||
|  |     requireFields({ username }); | ||||||
|  | 
 | ||||||
|  |     const invitations = await getAllInvitations(username, by); | ||||||
|  | 
 | ||||||
|  |     res.json({ invitations }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function getInvitationHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const sender = req.params.sender; | ||||||
|  |     const receiver = req.params.receiver; | ||||||
|  |     const classId = req.params.classId; | ||||||
|  |     requireFields({ sender, receiver, classId }); | ||||||
|  | 
 | ||||||
|  |     const invitation = await getInvitation(sender, receiver, classId); | ||||||
|  | 
 | ||||||
|  |     res.json({ invitation }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function createInvitationHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const sender = req.body.sender; | ||||||
|  |     const receiver = req.body.receiver; | ||||||
|  |     const classId = req.body.class; | ||||||
|  |     requireFields({ sender, receiver, classId }); | ||||||
|  | 
 | ||||||
|  |     const data = req.body as TeacherInvitationData; | ||||||
|  |     const invitation = await createInvitation(data); | ||||||
|  | 
 | ||||||
|  |     res.json({ invitation }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function updateInvitationHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const sender = req.body.sender; | ||||||
|  |     const receiver = req.body.receiver; | ||||||
|  |     const classId = req.body.class; | ||||||
|  |     req.body.accepted = req.body.accepted !== 'false'; | ||||||
|  |     requireFields({ sender, receiver, classId }); | ||||||
|  | 
 | ||||||
|  |     const data = req.body as TeacherInvitationData; | ||||||
|  |     const invitation = await updateInvitation(data); | ||||||
|  | 
 | ||||||
|  |     res.json({ invitation }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function deleteInvitationHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const sender = req.params.sender; | ||||||
|  |     const receiver = req.params.receiver; | ||||||
|  |     const classId = req.params.classId; | ||||||
|  |     requireFields({ sender, receiver, classId }); | ||||||
|  | 
 | ||||||
|  |     const data: TeacherInvitationData = { | ||||||
|  |         sender, | ||||||
|  |         receiver, | ||||||
|  |         class: classId, | ||||||
|  |     }; | ||||||
|  |     const invitation = await deleteInvitation(data); | ||||||
|  | 
 | ||||||
|  |     res.json({ invitation }); | ||||||
|  | } | ||||||
|  | @ -2,14 +2,14 @@ import { DwengoEntityRepository } from '../dwengo-entity-repository.js'; | ||||||
| import { Class } from '../../entities/classes/class.entity.js'; | import { Class } from '../../entities/classes/class.entity.js'; | ||||||
| import { ClassJoinRequest } from '../../entities/classes/class-join-request.entity.js'; | import { ClassJoinRequest } from '../../entities/classes/class-join-request.entity.js'; | ||||||
| import { Student } from '../../entities/users/student.entity.js'; | import { Student } from '../../entities/users/student.entity.js'; | ||||||
| import { ClassJoinRequestStatus } from '@dwengo-1/common/util/class-join-request'; | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export class ClassJoinRequestRepository extends DwengoEntityRepository<ClassJoinRequest> { | export class ClassJoinRequestRepository extends DwengoEntityRepository<ClassJoinRequest> { | ||||||
|     public async findAllRequestsBy(requester: Student): Promise<ClassJoinRequest[]> { |     public async findAllRequestsBy(requester: Student): Promise<ClassJoinRequest[]> { | ||||||
|         return this.findAll({ where: { requester: requester } }); |         return this.findAll({ where: { requester: requester } }); | ||||||
|     } |     } | ||||||
|     public async findAllOpenRequestsTo(clazz: Class): Promise<ClassJoinRequest[]> { |     public async findAllOpenRequestsTo(clazz: Class): Promise<ClassJoinRequest[]> { | ||||||
|         return this.findAll({ where: { class: clazz, status: ClassJoinRequestStatus.Open } }); // TODO check if works like this
 |         return this.findAll({ where: { class: clazz, status: ClassStatus.Open } }); // TODO check if works like this
 | ||||||
|     } |     } | ||||||
|     public async findByStudentAndClass(requester: Student, clazz: Class): Promise<ClassJoinRequest | null> { |     public async findByStudentAndClass(requester: Student, clazz: Class): Promise<ClassJoinRequest | null> { | ||||||
|         return this.findOne({ requester, class: clazz }); |         return this.findOne({ requester, class: clazz }); | ||||||
|  |  | ||||||
|  | @ -2,6 +2,7 @@ import { DwengoEntityRepository } from '../dwengo-entity-repository.js'; | ||||||
| import { Class } from '../../entities/classes/class.entity.js'; | import { Class } from '../../entities/classes/class.entity.js'; | ||||||
| import { TeacherInvitation } from '../../entities/classes/teacher-invitation.entity.js'; | import { TeacherInvitation } from '../../entities/classes/teacher-invitation.entity.js'; | ||||||
| import { Teacher } from '../../entities/users/teacher.entity.js'; | import { Teacher } from '../../entities/users/teacher.entity.js'; | ||||||
|  | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export class TeacherInvitationRepository extends DwengoEntityRepository<TeacherInvitation> { | export class TeacherInvitationRepository extends DwengoEntityRepository<TeacherInvitation> { | ||||||
|     public async findAllInvitationsForClass(clazz: Class): Promise<TeacherInvitation[]> { |     public async findAllInvitationsForClass(clazz: Class): Promise<TeacherInvitation[]> { | ||||||
|  | @ -11,7 +12,7 @@ export class TeacherInvitationRepository extends DwengoEntityRepository<TeacherI | ||||||
|         return this.findAll({ where: { sender: sender } }); |         return this.findAll({ where: { sender: sender } }); | ||||||
|     } |     } | ||||||
|     public async findAllInvitationsFor(receiver: Teacher): Promise<TeacherInvitation[]> { |     public async findAllInvitationsFor(receiver: Teacher): Promise<TeacherInvitation[]> { | ||||||
|         return this.findAll({ where: { receiver: receiver } }); |         return this.findAll({ where: { receiver: receiver, status: ClassStatus.Open } }); | ||||||
|     } |     } | ||||||
|     public async deleteBy(clazz: Class, sender: Teacher, receiver: Teacher): Promise<void> { |     public async deleteBy(clazz: Class, sender: Teacher, receiver: Teacher): Promise<void> { | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|  | @ -20,4 +21,11 @@ export class TeacherInvitationRepository extends DwengoEntityRepository<TeacherI | ||||||
|             class: clazz, |             class: clazz, | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |     public async findBy(clazz: Class, sender: Teacher, receiver: Teacher): Promise<TeacherInvitation | null> { | ||||||
|  |         return this.findOne({ | ||||||
|  |             sender: sender, | ||||||
|  |             receiver: receiver, | ||||||
|  |             class: clazz, | ||||||
|  |         }); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -6,6 +6,9 @@ import { Language } from '@dwengo-1/common/util/language'; | ||||||
| 
 | 
 | ||||||
| @Entity({ repository: () => SubmissionRepository }) | @Entity({ repository: () => SubmissionRepository }) | ||||||
| export class Submission { | export class Submission { | ||||||
|  |     @PrimaryKey({ type: 'integer', autoincrement: true }) | ||||||
|  |     submissionNumber?: number; | ||||||
|  | 
 | ||||||
|     @PrimaryKey({ type: 'string' }) |     @PrimaryKey({ type: 'string' }) | ||||||
|     learningObjectHruid!: string; |     learningObjectHruid!: string; | ||||||
| 
 | 
 | ||||||
|  | @ -18,9 +21,6 @@ export class Submission { | ||||||
|     @PrimaryKey({ type: 'numeric' }) |     @PrimaryKey({ type: 'numeric' }) | ||||||
|     learningObjectVersion = 1; |     learningObjectVersion = 1; | ||||||
| 
 | 
 | ||||||
|     @PrimaryKey({ type: 'integer', autoincrement: true }) |  | ||||||
|     submissionNumber?: number; |  | ||||||
| 
 |  | ||||||
|     @ManyToOne({ |     @ManyToOne({ | ||||||
|         entity: () => Group, |         entity: () => Group, | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  | @ -2,7 +2,7 @@ import { Entity, Enum, ManyToOne } from '@mikro-orm/core'; | ||||||
| import { Student } from '../users/student.entity.js'; | import { Student } from '../users/student.entity.js'; | ||||||
| import { Class } from './class.entity.js'; | import { Class } from './class.entity.js'; | ||||||
| import { ClassJoinRequestRepository } from '../../data/classes/class-join-request-repository.js'; | import { ClassJoinRequestRepository } from '../../data/classes/class-join-request-repository.js'; | ||||||
| import { ClassJoinRequestStatus } from '@dwengo-1/common/util/class-join-request'; | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| @Entity({ | @Entity({ | ||||||
|     repository: () => ClassJoinRequestRepository, |     repository: () => ClassJoinRequestRepository, | ||||||
|  | @ -20,6 +20,6 @@ export class ClassJoinRequest { | ||||||
|     }) |     }) | ||||||
|     class!: Class; |     class!: Class; | ||||||
| 
 | 
 | ||||||
|     @Enum(() => ClassJoinRequestStatus) |     @Enum(() => ClassStatus) | ||||||
|     status!: ClassJoinRequestStatus; |     status!: ClassStatus; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,7 +1,8 @@ | ||||||
| import { Entity, ManyToOne } from '@mikro-orm/core'; | import { Entity, Enum, ManyToOne } from '@mikro-orm/core'; | ||||||
| import { Teacher } from '../users/teacher.entity.js'; | import { Teacher } from '../users/teacher.entity.js'; | ||||||
| import { Class } from './class.entity.js'; | import { Class } from './class.entity.js'; | ||||||
| import { TeacherInvitationRepository } from '../../data/classes/teacher-invitation-repository.js'; | import { TeacherInvitationRepository } from '../../data/classes/teacher-invitation-repository.js'; | ||||||
|  | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Invitation of a teacher into a class (in order to teach it). |  * Invitation of a teacher into a class (in order to teach it). | ||||||
|  | @ -25,4 +26,7 @@ export class TeacherInvitation { | ||||||
|         primary: true, |         primary: true, | ||||||
|     }) |     }) | ||||||
|     class!: Class; |     class!: Class; | ||||||
|  | 
 | ||||||
|  |     @Enum(() => ClassStatus) | ||||||
|  |     status!: ClassStatus; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -4,7 +4,7 @@ import { getClassJoinRequestRepository } from '../data/repositories.js'; | ||||||
| import { Student } from '../entities/users/student.entity.js'; | import { Student } from '../entities/users/student.entity.js'; | ||||||
| import { Class } from '../entities/classes/class.entity.js'; | import { Class } from '../entities/classes/class.entity.js'; | ||||||
| import { ClassJoinRequestDTO } from '@dwengo-1/common/interfaces/class-join-request'; | import { ClassJoinRequestDTO } from '@dwengo-1/common/interfaces/class-join-request'; | ||||||
| import { ClassJoinRequestStatus } from '@dwengo-1/common/util/class-join-request'; | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export function mapToStudentRequestDTO(request: ClassJoinRequest): ClassJoinRequestDTO { | export function mapToStudentRequestDTO(request: ClassJoinRequest): ClassJoinRequestDTO { | ||||||
|     return { |     return { | ||||||
|  | @ -18,6 +18,6 @@ export function mapToStudentRequest(student: Student, cls: Class): ClassJoinRequ | ||||||
|     return getClassJoinRequestRepository().create({ |     return getClassJoinRequestRepository().create({ | ||||||
|         requester: student, |         requester: student, | ||||||
|         class: cls, |         class: cls, | ||||||
|         status: ClassJoinRequestStatus.Open, |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,13 +1,17 @@ | ||||||
| import { TeacherInvitation } from '../entities/classes/teacher-invitation.entity.js'; | import { TeacherInvitation } from '../entities/classes/teacher-invitation.entity.js'; | ||||||
| import { mapToClassDTO } from './class.js'; |  | ||||||
| import { mapToUserDTO } from './user.js'; | import { mapToUserDTO } from './user.js'; | ||||||
| import { TeacherInvitationDTO } from '@dwengo-1/common/interfaces/teacher-invitation'; | import { TeacherInvitationDTO } from '@dwengo-1/common/interfaces/teacher-invitation'; | ||||||
|  | import { getTeacherInvitationRepository } from '../data/repositories'; | ||||||
|  | import { Teacher } from '../entities/users/teacher.entity'; | ||||||
|  | import { Class } from '../entities/classes/class.entity'; | ||||||
|  | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export function mapToTeacherInvitationDTO(invitation: TeacherInvitation): TeacherInvitationDTO { | export function mapToTeacherInvitationDTO(invitation: TeacherInvitation): TeacherInvitationDTO { | ||||||
|     return { |     return { | ||||||
|         sender: mapToUserDTO(invitation.sender), |         sender: mapToUserDTO(invitation.sender), | ||||||
|         receiver: mapToUserDTO(invitation.receiver), |         receiver: mapToUserDTO(invitation.receiver), | ||||||
|         class: mapToClassDTO(invitation.class), |         classId: invitation.class.classId!, | ||||||
|  |         status: invitation.status, | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -15,6 +19,16 @@ export function mapToTeacherInvitationDTOIds(invitation: TeacherInvitation): Tea | ||||||
|     return { |     return { | ||||||
|         sender: invitation.sender.username, |         sender: invitation.sender.username, | ||||||
|         receiver: invitation.receiver.username, |         receiver: invitation.receiver.username, | ||||||
|         class: invitation.class.classId!, |         classId: invitation.class.classId!, | ||||||
|  |         status: invitation.status, | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | export function mapToInvitation(sender: Teacher, receiver: Teacher, cls: Class): TeacherInvitation { | ||||||
|  |     return getTeacherInvitationRepository().create({ | ||||||
|  |         sender, | ||||||
|  |         receiver, | ||||||
|  |         class: cls, | ||||||
|  |         status: ClassStatus.Open, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -5,7 +5,7 @@ const router = express.Router({ mergeParams: true }); | ||||||
| // Root endpoint used to search objects
 | // Root endpoint used to search objects
 | ||||||
| router.get('/', getSubmissionsHandler); | router.get('/', getSubmissionsHandler); | ||||||
| 
 | 
 | ||||||
| router.post('/:id', createSubmissionHandler); | router.post('/', createSubmissionHandler); | ||||||
| 
 | 
 | ||||||
| // Information about an submission with id 'id'
 | // Information about an submission with id 'id'
 | ||||||
| router.get('/:id', getSubmissionHandler); | router.get('/:id', getSubmissionHandler); | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								backend/src/routes/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								backend/src/routes/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | import express from 'express'; | ||||||
|  | import { | ||||||
|  |     createInvitationHandler, | ||||||
|  |     deleteInvitationHandler, | ||||||
|  |     getAllInvitationsHandler, | ||||||
|  |     getInvitationHandler, | ||||||
|  |     updateInvitationHandler, | ||||||
|  | } from '../controllers/teacher-invitations'; | ||||||
|  | 
 | ||||||
|  | const router = express.Router({ mergeParams: true }); | ||||||
|  | 
 | ||||||
|  | router.get('/:username', getAllInvitationsHandler); | ||||||
|  | 
 | ||||||
|  | router.get('/:sender/:receiver/:classId', getInvitationHandler); | ||||||
|  | 
 | ||||||
|  | router.post('/', createInvitationHandler); | ||||||
|  | 
 | ||||||
|  | router.put('/', updateInvitationHandler); | ||||||
|  | 
 | ||||||
|  | router.delete('/:sender/:receiver/:classId', deleteInvitationHandler); | ||||||
|  | 
 | ||||||
|  | export default router; | ||||||
|  | @ -10,6 +10,8 @@ import { | ||||||
|     getTeacherStudentHandler, |     getTeacherStudentHandler, | ||||||
|     updateStudentJoinRequestHandler, |     updateStudentJoinRequestHandler, | ||||||
| } from '../controllers/teachers.js'; | } from '../controllers/teachers.js'; | ||||||
|  | import invitationRouter from './teacher-invitations.js'; | ||||||
|  | 
 | ||||||
| const router = express.Router(); | const router = express.Router(); | ||||||
| 
 | 
 | ||||||
| // Root endpoint used to search objects
 | // Root endpoint used to search objects
 | ||||||
|  | @ -32,10 +34,6 @@ router.get('/:username/joinRequests/:classId', getStudentJoinRequestHandler); | ||||||
| router.put('/:username/joinRequests/:classId/:studentUsername', updateStudentJoinRequestHandler); | router.put('/:username/joinRequests/:classId/:studentUsername', updateStudentJoinRequestHandler); | ||||||
| 
 | 
 | ||||||
| // Invitations to other classes a teacher received
 | // Invitations to other classes a teacher received
 | ||||||
| router.get('/:id/invitations', (_req, res) => { | router.get('/invitations', invitationRouter); | ||||||
|     res.json({ |  | ||||||
|         invitations: ['0'], |  | ||||||
|     }); |  | ||||||
| }); |  | ||||||
| 
 | 
 | ||||||
| export default router; | export default router; | ||||||
|  |  | ||||||
|  | @ -111,6 +111,7 @@ async function convertNode( | ||||||
|         updatedAt: node.updatedAt.toISOString(), |         updatedAt: node.updatedAt.toISOString(), | ||||||
|         learningobject_hruid: node.learningObjectHruid, |         learningobject_hruid: node.learningObjectHruid, | ||||||
|         version: learningObject.version, |         version: learningObject.version, | ||||||
|  |         done: personalizedFor ? lastSubmission !== null : undefined, | ||||||
|         transitions, |         transitions, | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,4 +1,4 @@ | ||||||
| import { getAssignmentRepository, getSubmissionRepository } from '../data/repositories.js'; | import {getAssignmentRepository, getGroupRepository, getSubmissionRepository} from '../data/repositories.js'; | ||||||
| import { LearningObjectIdentifier } from '../entities/content/learning-object-identifier.js'; | import { LearningObjectIdentifier } from '../entities/content/learning-object-identifier.js'; | ||||||
| import { NotFoundException } from '../exceptions/not-found-exception.js'; | import { NotFoundException } from '../exceptions/not-found-exception.js'; | ||||||
| import { mapToSubmission, mapToSubmissionDTO } from '../interfaces/submission.js'; | import { mapToSubmission, mapToSubmissionDTO } from '../interfaces/submission.js'; | ||||||
|  | @ -69,3 +69,24 @@ export async function getSubmissionsForLearningObjectAndAssignment( | ||||||
| 
 | 
 | ||||||
|     return submissions.map((s) => mapToSubmissionDTO(s)); |     return submissions.map((s) => mapToSubmissionDTO(s)); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | export async function getSubmissionsForLearningObjectAndGroup( | ||||||
|  |     learningObjectHruid: string, | ||||||
|  |     language: Language, | ||||||
|  |     version: number, | ||||||
|  |     classId: string, | ||||||
|  |     assignmentNo: number, | ||||||
|  |     groupNumber: number | ||||||
|  | ): Promise<SubmissionDTO[]> { | ||||||
|  |     const loId = new LearningObjectIdentifier(learningObjectHruid, language, version); | ||||||
|  |     const assignment = await getAssignmentRepository().findByClassIdAndAssignmentId(classId, assignmentNo); | ||||||
|  |     if (!assignment) { | ||||||
|  |         throw new NotFoundException("Assignment not found!"); | ||||||
|  |     } | ||||||
|  |     const group = await getGroupRepository().findByAssignmentAndGroupNumber(assignment, groupNumber); | ||||||
|  |     if (!group) { | ||||||
|  |         throw new NotFoundException("Group not found!"); | ||||||
|  |     } | ||||||
|  |     const submissions = await getSubmissionRepository().findAllSubmissionsForGroup(group); | ||||||
|  |     return submissions.map((s) => mapToSubmissionDTO(s)); | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										87
									
								
								backend/src/services/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								backend/src/services/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,87 @@ | ||||||
|  | import { fetchTeacher } from './teachers'; | ||||||
|  | import { getTeacherInvitationRepository } from '../data/repositories'; | ||||||
|  | import { mapToInvitation, mapToTeacherInvitationDTO } from '../interfaces/teacher-invitation'; | ||||||
|  | import { addClassTeacher, fetchClass } from './classes'; | ||||||
|  | import { TeacherInvitationData, TeacherInvitationDTO } from '@dwengo-1/common/interfaces/teacher-invitation'; | ||||||
|  | import { ConflictException } from '../exceptions/conflict-exception'; | ||||||
|  | import { NotFoundException } from '../exceptions/not-found-exception'; | ||||||
|  | import { TeacherInvitation } from '../entities/classes/teacher-invitation.entity'; | ||||||
|  | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
|  | 
 | ||||||
|  | export async function getAllInvitations(username: string, sent: boolean): Promise<TeacherInvitationDTO[]> { | ||||||
|  |     const teacher = await fetchTeacher(username); | ||||||
|  |     const teacherInvitationRepository = getTeacherInvitationRepository(); | ||||||
|  | 
 | ||||||
|  |     let invitations; | ||||||
|  |     if (sent) { | ||||||
|  |         invitations = await teacherInvitationRepository.findAllInvitationsBy(teacher); | ||||||
|  |     } else { | ||||||
|  |         invitations = await teacherInvitationRepository.findAllInvitationsFor(teacher); | ||||||
|  |     } | ||||||
|  |     return invitations.map(mapToTeacherInvitationDTO); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function createInvitation(data: TeacherInvitationData): Promise<TeacherInvitationDTO> { | ||||||
|  |     const teacherInvitationRepository = getTeacherInvitationRepository(); | ||||||
|  |     const sender = await fetchTeacher(data.sender); | ||||||
|  |     const receiver = await fetchTeacher(data.receiver); | ||||||
|  | 
 | ||||||
|  |     const cls = await fetchClass(data.class); | ||||||
|  | 
 | ||||||
|  |     if (!cls.teachers.contains(sender)) { | ||||||
|  |         throw new ConflictException('The teacher sending the invite is not part of the class'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const newInvitation = mapToInvitation(sender, receiver, cls); | ||||||
|  |     await teacherInvitationRepository.save(newInvitation, { preventOverwrite: true }); | ||||||
|  | 
 | ||||||
|  |     return mapToTeacherInvitationDTO(newInvitation); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function fetchInvitation(usernameSender: string, usernameReceiver: string, classId: string): Promise<TeacherInvitation> { | ||||||
|  |     const sender = await fetchTeacher(usernameSender); | ||||||
|  |     const receiver = await fetchTeacher(usernameReceiver); | ||||||
|  |     const cls = await fetchClass(classId); | ||||||
|  | 
 | ||||||
|  |     const teacherInvitationRepository = getTeacherInvitationRepository(); | ||||||
|  |     const invite = await teacherInvitationRepository.findBy(cls, sender, receiver); | ||||||
|  | 
 | ||||||
|  |     if (!invite) { | ||||||
|  |         throw new NotFoundException('Teacher invite not found'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return invite; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function getInvitation(sender: string, receiver: string, classId: string): Promise<TeacherInvitationDTO> { | ||||||
|  |     const invitation = await fetchInvitation(sender, receiver, classId); | ||||||
|  |     return mapToTeacherInvitationDTO(invitation); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function updateInvitation(data: TeacherInvitationData): Promise<TeacherInvitationDTO> { | ||||||
|  |     const invitation = await fetchInvitation(data.sender, data.receiver, data.class); | ||||||
|  |     invitation.status = ClassStatus.Declined; | ||||||
|  | 
 | ||||||
|  |     if (data.accepted) { | ||||||
|  |         invitation.status = ClassStatus.Accepted; | ||||||
|  |         await addClassTeacher(data.class, data.receiver); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const teacherInvitationRepository = getTeacherInvitationRepository(); | ||||||
|  |     await teacherInvitationRepository.save(invitation); | ||||||
|  | 
 | ||||||
|  |     return mapToTeacherInvitationDTO(invitation); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function deleteInvitation(data: TeacherInvitationData): Promise<TeacherInvitationDTO> { | ||||||
|  |     const invitation = await fetchInvitation(data.sender, data.receiver, data.class); | ||||||
|  | 
 | ||||||
|  |     const sender = await fetchTeacher(data.sender); | ||||||
|  |     const receiver = await fetchTeacher(data.receiver); | ||||||
|  |     const cls = await fetchClass(data.class); | ||||||
|  | 
 | ||||||
|  |     const teacherInvitationRepository = getTeacherInvitationRepository(); | ||||||
|  |     await teacherInvitationRepository.deleteBy(cls, sender, receiver); | ||||||
|  | 
 | ||||||
|  |     return mapToTeacherInvitationDTO(invitation); | ||||||
|  | } | ||||||
|  | @ -28,7 +28,7 @@ import { ClassDTO } from '@dwengo-1/common/interfaces/class'; | ||||||
| import { StudentDTO } from '@dwengo-1/common/interfaces/student'; | import { StudentDTO } from '@dwengo-1/common/interfaces/student'; | ||||||
| import { QuestionDTO, QuestionId } from '@dwengo-1/common/interfaces/question'; | import { QuestionDTO, QuestionId } from '@dwengo-1/common/interfaces/question'; | ||||||
| import { ClassJoinRequestDTO } from '@dwengo-1/common/interfaces/class-join-request'; | import { ClassJoinRequestDTO } from '@dwengo-1/common/interfaces/class-join-request'; | ||||||
| import { ClassJoinRequestStatus } from '@dwengo-1/common/util/class-join-request'; | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| import { ConflictException } from '../exceptions/conflict-exception.js'; | import { ConflictException } from '../exceptions/conflict-exception.js'; | ||||||
| 
 | 
 | ||||||
| export async function getAllTeachers(full: boolean): Promise<TeacherDTO[] | string[]> { | export async function getAllTeachers(full: boolean): Promise<TeacherDTO[] | string[]> { | ||||||
|  | @ -160,10 +160,10 @@ export async function updateClassJoinRequestStatus(studentUsername: string, clas | ||||||
|         throw new NotFoundException('Join request not found'); |         throw new NotFoundException('Join request not found'); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     request.status = ClassJoinRequestStatus.Declined; |     request.status = ClassStatus.Declined; | ||||||
| 
 | 
 | ||||||
|     if (accepted) { |     if (accepted) { | ||||||
|         request.status = ClassJoinRequestStatus.Accepted; |         request.status = ClassStatus.Accepted; | ||||||
|         await addClassStudent(classId, studentUsername); |         await addClassStudent(classId, studentUsername); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										123
									
								
								backend/tests/controllers/teacher-invitations.test.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								backend/tests/controllers/teacher-invitations.test.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,123 @@ | ||||||
|  | import { beforeAll, beforeEach, describe, expect, it, Mock, vi } from 'vitest'; | ||||||
|  | import { Request, Response } from 'express'; | ||||||
|  | import { setupTestApp } from '../setup-tests.js'; | ||||||
|  | import { | ||||||
|  |     createInvitationHandler, | ||||||
|  |     deleteInvitationHandler, | ||||||
|  |     getAllInvitationsHandler, | ||||||
|  |     getInvitationHandler, | ||||||
|  |     updateInvitationHandler, | ||||||
|  | } from '../../src/controllers/teacher-invitations'; | ||||||
|  | import { TeacherInvitationData } from '@dwengo-1/common/interfaces/teacher-invitation'; | ||||||
|  | import { getClassHandler } from '../../src/controllers/classes'; | ||||||
|  | import { BadRequestException } from '../../src/exceptions/bad-request-exception'; | ||||||
|  | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
|  | 
 | ||||||
|  | describe('Teacher controllers', () => { | ||||||
|  |     let req: Partial<Request>; | ||||||
|  |     let res: Partial<Response>; | ||||||
|  | 
 | ||||||
|  |     let jsonMock: Mock; | ||||||
|  | 
 | ||||||
|  |     beforeAll(async () => { | ||||||
|  |         await setupTestApp(); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     beforeEach(() => { | ||||||
|  |         jsonMock = vi.fn(); | ||||||
|  |         res = { | ||||||
|  |             json: jsonMock, | ||||||
|  |         }; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('Get teacher invitations by', async () => { | ||||||
|  |         req = { params: { username: 'LimpBizkit' }, query: { sent: 'true' } }; | ||||||
|  | 
 | ||||||
|  |         await getAllInvitationsHandler(req as Request, res as Response); | ||||||
|  | 
 | ||||||
|  |         expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ invitations: expect.anything() })); | ||||||
|  | 
 | ||||||
|  |         const result = jsonMock.mock.lastCall?.[0]; | ||||||
|  |         // Console.log(result.invitations);
 | ||||||
|  |         expect(result.invitations).to.have.length.greaterThan(0); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('Get teacher invitations for', async () => { | ||||||
|  |         req = { params: { username: 'FooFighters' }, query: { by: 'false' } }; | ||||||
|  | 
 | ||||||
|  |         await getAllInvitationsHandler(req as Request, res as Response); | ||||||
|  | 
 | ||||||
|  |         expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ invitations: expect.anything() })); | ||||||
|  | 
 | ||||||
|  |         const result = jsonMock.mock.lastCall?.[0]; | ||||||
|  |         expect(result.invitations).to.have.length.greaterThan(0); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('Create and delete invitation', async () => { | ||||||
|  |         const body = { | ||||||
|  |             sender: 'LimpBizkit', | ||||||
|  |             receiver: 'testleerkracht1', | ||||||
|  |             class: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', | ||||||
|  |         } as TeacherInvitationData; | ||||||
|  |         req = { body }; | ||||||
|  | 
 | ||||||
|  |         await createInvitationHandler(req as Request, res as Response); | ||||||
|  | 
 | ||||||
|  |         req = { | ||||||
|  |             params: { | ||||||
|  |                 sender: 'LimpBizkit', | ||||||
|  |                 receiver: 'testleerkracht1', | ||||||
|  |                 classId: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', | ||||||
|  |             }, | ||||||
|  |             body: { accepted: 'false' }, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         await deleteInvitationHandler(req as Request, res as Response); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('Get invitation', async () => { | ||||||
|  |         req = { | ||||||
|  |             params: { | ||||||
|  |                 sender: 'LimpBizkit', | ||||||
|  |                 receiver: 'FooFighters', | ||||||
|  |                 classId: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |         await getInvitationHandler(req as Request, res as Response); | ||||||
|  | 
 | ||||||
|  |         expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ invitation: expect.anything() })); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('Get invitation error', async () => { | ||||||
|  |         req = { | ||||||
|  |             params: { no: 'no params' }, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         await expect(async () => getInvitationHandler(req as Request, res as Response)).rejects.toThrowError(BadRequestException); | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     it('Accept invitation', async () => { | ||||||
|  |         const body = { | ||||||
|  |             sender: 'LimpBizkit', | ||||||
|  |             receiver: 'FooFighters', | ||||||
|  |             class: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', | ||||||
|  |         } as TeacherInvitationData; | ||||||
|  |         req = { body }; | ||||||
|  | 
 | ||||||
|  |         await updateInvitationHandler(req as Request, res as Response); | ||||||
|  | 
 | ||||||
|  |         const result1 = jsonMock.mock.lastCall?.[0]; | ||||||
|  |         expect(result1.invitation.status).toEqual(ClassStatus.Accepted); | ||||||
|  | 
 | ||||||
|  |         req = { | ||||||
|  |             params: { | ||||||
|  |                 id: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         await getClassHandler(req as Request, res as Response); | ||||||
|  | 
 | ||||||
|  |         const result = jsonMock.mock.lastCall?.[0]; | ||||||
|  |         expect(result.class.teachers).toContain('FooFighters'); | ||||||
|  |     }); | ||||||
|  | }); | ||||||
|  | @ -2,9 +2,11 @@ import { EntityManager } from '@mikro-orm/core'; | ||||||
| import { Assignment } from '../../../src/entities/assignments/assignment.entity'; | import { Assignment } from '../../../src/entities/assignments/assignment.entity'; | ||||||
| import { Class } from '../../../src/entities/classes/class.entity'; | import { Class } from '../../../src/entities/classes/class.entity'; | ||||||
| import { Language } from '@dwengo-1/common/util/language'; | import { Language } from '@dwengo-1/common/util/language'; | ||||||
|  | import {testLearningPathWithConditions} from "../content/learning-paths.testdata"; | ||||||
|  | import {getClassWithTestleerlingAndTestleerkracht} from "../classes/classes.testdata"; | ||||||
| 
 | 
 | ||||||
| export function makeTestAssignemnts(em: EntityManager, classes: Class[]): Assignment[] { | export function makeTestAssignemnts(em: EntityManager, classes: Class[]): Assignment[] { | ||||||
|     const assignment01 = em.create(Assignment, { |     assignment01 = em.create(Assignment, { | ||||||
|         within: classes[0], |         within: classes[0], | ||||||
|         id: 1, |         id: 1, | ||||||
|         title: 'dire straits', |         title: 'dire straits', | ||||||
|  | @ -14,7 +16,7 @@ export function makeTestAssignemnts(em: EntityManager, classes: Class[]): Assign | ||||||
|         groups: [], |         groups: [], | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const assignment02 = em.create(Assignment, { |     assignment02 = em.create(Assignment, { | ||||||
|         within: classes[1], |         within: classes[1], | ||||||
|         id: 2, |         id: 2, | ||||||
|         title: 'tool', |         title: 'tool', | ||||||
|  | @ -24,7 +26,7 @@ export function makeTestAssignemnts(em: EntityManager, classes: Class[]): Assign | ||||||
|         groups: [], |         groups: [], | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const assignment03 = em.create(Assignment, { |     assignment03 = em.create(Assignment, { | ||||||
|         within: classes[0], |         within: classes[0], | ||||||
|         id: 3, |         id: 3, | ||||||
|         title: 'delete', |         title: 'delete', | ||||||
|  | @ -34,7 +36,7 @@ export function makeTestAssignemnts(em: EntityManager, classes: Class[]): Assign | ||||||
|         groups: [], |         groups: [], | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const assignment04 = em.create(Assignment, { |     assignment04 = em.create(Assignment, { | ||||||
|         within: classes[0], |         within: classes[0], | ||||||
|         id: 4, |         id: 4, | ||||||
|         title: 'another assignment', |         title: 'another assignment', | ||||||
|  | @ -44,5 +46,41 @@ export function makeTestAssignemnts(em: EntityManager, classes: Class[]): Assign | ||||||
|         groups: [], |         groups: [], | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return [assignment01, assignment02, assignment03, assignment04]; |     conditionalPathAssignment = em.create(Assignment, { | ||||||
|  |         within: getClassWithTestleerlingAndTestleerkracht(), | ||||||
|  |         id: 1, | ||||||
|  |         title: 'Assignment: Conditional Learning Path', | ||||||
|  |         description: 'You have to do the testing learning path with a condition.', | ||||||
|  |         learningPathHruid: testLearningPathWithConditions.hruid, | ||||||
|  |         learningPathLanguage: testLearningPathWithConditions.language as Language, | ||||||
|  |         groups: [], | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     return [assignment01, assignment02, assignment03, assignment04, conditionalPathAssignment]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | let assignment01: Assignment; | ||||||
|  | let assignment02: Assignment; | ||||||
|  | let assignment03: Assignment; | ||||||
|  | let assignment04: Assignment; | ||||||
|  | let conditionalPathAssignment: Assignment; | ||||||
|  | 
 | ||||||
|  | export function getAssignment01(): Assignment { | ||||||
|  |     return assignment01; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getAssignment02(): Assignment { | ||||||
|  |     return assignment02; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getAssignment03(): Assignment { | ||||||
|  |     return assignment03; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getAssignment04(): Assignment { | ||||||
|  |     return assignment04; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getConditionalPathAssignment(): Assignment { | ||||||
|  |     return conditionalPathAssignment; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,6 +2,8 @@ import {EntityManager} from '@mikro-orm/core'; | ||||||
| import { Group } from '../../../src/entities/assignments/group.entity'; | import { Group } from '../../../src/entities/assignments/group.entity'; | ||||||
| import { Assignment } from '../../../src/entities/assignments/assignment.entity'; | import { Assignment } from '../../../src/entities/assignments/assignment.entity'; | ||||||
| import { Student } from '../../../src/entities/users/student.entity'; | import { Student } from '../../../src/entities/users/student.entity'; | ||||||
|  | import {getConditionalPathAssignment} from "./assignments.testdata"; | ||||||
|  | import {getTestleerling1} from "../users/students.testdata"; | ||||||
| 
 | 
 | ||||||
| export function makeTestGroups(em: EntityManager, students: Student[], assignments: Assignment[]): Group[] { | export function makeTestGroups(em: EntityManager, students: Student[], assignments: Assignment[]): Group[] { | ||||||
|     /* |     /* | ||||||
|  | @ -54,7 +56,16 @@ export function makeTestGroups(em: EntityManager, students: Student[], assignmen | ||||||
|         members: students.slice(0, 2), |         members: students.slice(0, 2), | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return [group01, group02, group03, group04, group05]; |     /** | ||||||
|  |      * Group 1 for the assignment of the testing learning path with conditions. | ||||||
|  |      */ | ||||||
|  |     group1ConditionalLearningPath = em.create(Group, { | ||||||
|  |         assignment: getConditionalPathAssignment(), | ||||||
|  |         groupNumber: 1, | ||||||
|  |         members: [getTestleerling1()] | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     return [group01, group02, group03, group04, group05, group1ConditionalLearningPath]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| let group01: Group; | let group01: Group; | ||||||
|  | @ -62,6 +73,7 @@ let group02: Group; | ||||||
| let group03: Group; | let group03: Group; | ||||||
| let group04: Group; | let group04: Group; | ||||||
| let group05: Group; | let group05: Group; | ||||||
|  | let group1ConditionalLearningPath: Group; | ||||||
| 
 | 
 | ||||||
| export function getTestGroup01() { | export function getTestGroup01() { | ||||||
|     return group01; |     return group01; | ||||||
|  | @ -83,3 +95,6 @@ export function getTestGroup05() { | ||||||
|     return group05; |     return group05; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export function getGroup1ConditionalLearningPath() { | ||||||
|  |     return group1ConditionalLearningPath; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -2,31 +2,31 @@ import { EntityManager } from '@mikro-orm/core'; | ||||||
| import { ClassJoinRequest } from '../../../src/entities/classes/class-join-request.entity'; | import { ClassJoinRequest } from '../../../src/entities/classes/class-join-request.entity'; | ||||||
| import { Student } from '../../../src/entities/users/student.entity'; | import { Student } from '../../../src/entities/users/student.entity'; | ||||||
| import { Class } from '../../../src/entities/classes/class.entity'; | import { Class } from '../../../src/entities/classes/class.entity'; | ||||||
| import { ClassJoinRequestStatus } from '@dwengo-1/common/util/class-join-request'; | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export function makeTestClassJoinRequests(em: EntityManager, students: Student[], classes: Class[]): ClassJoinRequest[] { | export function makeTestClassJoinRequests(em: EntityManager, students: Student[], classes: Class[]): ClassJoinRequest[] { | ||||||
|     const classJoinRequest01 = em.create(ClassJoinRequest, { |     const classJoinRequest01 = em.create(ClassJoinRequest, { | ||||||
|         requester: students[4], |         requester: students[4], | ||||||
|         class: classes[1], |         class: classes[1], | ||||||
|         status: ClassJoinRequestStatus.Open, |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const classJoinRequest02 = em.create(ClassJoinRequest, { |     const classJoinRequest02 = em.create(ClassJoinRequest, { | ||||||
|         requester: students[2], |         requester: students[2], | ||||||
|         class: classes[1], |         class: classes[1], | ||||||
|         status: ClassJoinRequestStatus.Open, |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const classJoinRequest03 = em.create(ClassJoinRequest, { |     const classJoinRequest03 = em.create(ClassJoinRequest, { | ||||||
|         requester: students[4], |         requester: students[4], | ||||||
|         class: classes[2], |         class: classes[2], | ||||||
|         status: ClassJoinRequestStatus.Open, |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const classJoinRequest04 = em.create(ClassJoinRequest, { |     const classJoinRequest04 = em.create(ClassJoinRequest, { | ||||||
|         requester: students[3], |         requester: students[3], | ||||||
|         class: classes[2], |         class: classes[2], | ||||||
|         status: ClassJoinRequestStatus.Open, |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return [classJoinRequest01, classJoinRequest02, classJoinRequest03, classJoinRequest04]; |     return [classJoinRequest01, classJoinRequest02, classJoinRequest03, classJoinRequest04]; | ||||||
|  |  | ||||||
|  | @ -2,12 +2,14 @@ import { EntityManager } from '@mikro-orm/core'; | ||||||
| import { Class } from '../../../src/entities/classes/class.entity'; | import { Class } from '../../../src/entities/classes/class.entity'; | ||||||
| import { Student } from '../../../src/entities/users/student.entity'; | import { Student } from '../../../src/entities/users/student.entity'; | ||||||
| import { Teacher } from '../../../src/entities/users/teacher.entity'; | import { Teacher } from '../../../src/entities/users/teacher.entity'; | ||||||
|  | import {getTestleerkracht1} from "../users/teachers.testdata"; | ||||||
|  | import {getTestleerling1} from "../users/students.testdata"; | ||||||
| 
 | 
 | ||||||
| export function makeTestClasses(em: EntityManager, students: Student[], teachers: Teacher[]): Class[] { | export function makeTestClasses(em: EntityManager, students: Student[], teachers: Teacher[]): Class[] { | ||||||
|     const studentsClass01 = students.slice(0, 8); |     const studentsClass01 = students.slice(0, 8); | ||||||
|     const teacherClass01: Teacher[] = teachers.slice(4, 5); |     const teacherClass01: Teacher[] = teachers.slice(4, 5); | ||||||
| 
 | 
 | ||||||
|     const class01 = em.create(Class, { |     class01 = em.create(Class, { | ||||||
|         classId: '8764b861-90a6-42e5-9732-c0d9eb2f55f9', |         classId: '8764b861-90a6-42e5-9732-c0d9eb2f55f9', | ||||||
|         displayName: 'class01', |         displayName: 'class01', | ||||||
|         teachers: teacherClass01, |         teachers: teacherClass01, | ||||||
|  | @ -17,7 +19,7 @@ export function makeTestClasses(em: EntityManager, students: Student[], teachers | ||||||
|     const studentsClass02: Student[] = students.slice(0, 2).concat(students.slice(3, 4)); |     const studentsClass02: Student[] = students.slice(0, 2).concat(students.slice(3, 4)); | ||||||
|     const teacherClass02: Teacher[] = teachers.slice(1, 2); |     const teacherClass02: Teacher[] = teachers.slice(1, 2); | ||||||
| 
 | 
 | ||||||
|     const class02 = em.create(Class, { |     class02 = em.create(Class, { | ||||||
|         classId: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', |         classId: '34d484a1-295f-4e9f-bfdc-3e7a23d86a89', | ||||||
|         displayName: 'class02', |         displayName: 'class02', | ||||||
|         teachers: teacherClass02, |         teachers: teacherClass02, | ||||||
|  | @ -27,7 +29,7 @@ export function makeTestClasses(em: EntityManager, students: Student[], teachers | ||||||
|     const studentsClass03: Student[] = students.slice(1, 4); |     const studentsClass03: Student[] = students.slice(1, 4); | ||||||
|     const teacherClass03: Teacher[] = teachers.slice(2, 3); |     const teacherClass03: Teacher[] = teachers.slice(2, 3); | ||||||
| 
 | 
 | ||||||
|     const class03 = em.create(Class, { |     class03 = em.create(Class, { | ||||||
|         classId: '80dcc3e0-1811-4091-9361-42c0eee91cfa', |         classId: '80dcc3e0-1811-4091-9361-42c0eee91cfa', | ||||||
|         displayName: 'class03', |         displayName: 'class03', | ||||||
|         teachers: teacherClass03, |         teachers: teacherClass03, | ||||||
|  | @ -37,12 +39,45 @@ export function makeTestClasses(em: EntityManager, students: Student[], teachers | ||||||
|     const studentsClass04: Student[] = students.slice(0, 2); |     const studentsClass04: Student[] = students.slice(0, 2); | ||||||
|     const teacherClass04: Teacher[] = teachers.slice(2, 3); |     const teacherClass04: Teacher[] = teachers.slice(2, 3); | ||||||
| 
 | 
 | ||||||
|     const class04 = em.create(Class, { |     class04 = em.create(Class, { | ||||||
|         classId: '33d03536-83b8-4880-9982-9bbf2f908ddf', |         classId: '33d03536-83b8-4880-9982-9bbf2f908ddf', | ||||||
|         displayName: 'class04', |         displayName: 'class04', | ||||||
|         teachers: teacherClass04, |         teachers: teacherClass04, | ||||||
|         students: studentsClass04, |         students: studentsClass04, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return [class01, class02, class03, class04]; |     classWithTestleerlingAndTestleerkracht = em.create(Class, { | ||||||
|  |         classId: "a75298b5-18aa-471d-8eeb-5d77eb989393", | ||||||
|  |         displayName: 'Testklasse', | ||||||
|  |         teachers: [getTestleerkracht1()], | ||||||
|  |         students: [getTestleerling1()] | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     return [class01, class02, class03, class04, classWithTestleerlingAndTestleerkracht]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | let class01: Class; | ||||||
|  | let class02: Class; | ||||||
|  | let class03: Class; | ||||||
|  | let class04: Class; | ||||||
|  | let classWithTestleerlingAndTestleerkracht: Class; | ||||||
|  | 
 | ||||||
|  | export function getClass01(): Class { | ||||||
|  |     return class01; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getClass02(): Class { | ||||||
|  |     return class02; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getClass03(): Class { | ||||||
|  |     return class03; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getClass04(): Class { | ||||||
|  |     return class04; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getClassWithTestleerlingAndTestleerkracht(): Class { | ||||||
|  |     return classWithTestleerlingAndTestleerkracht; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,30 +2,35 @@ import { EntityManager } from '@mikro-orm/core'; | ||||||
| import { TeacherInvitation } from '../../../src/entities/classes/teacher-invitation.entity'; | import { TeacherInvitation } from '../../../src/entities/classes/teacher-invitation.entity'; | ||||||
| import { Teacher } from '../../../src/entities/users/teacher.entity'; | import { Teacher } from '../../../src/entities/users/teacher.entity'; | ||||||
| import { Class } from '../../../src/entities/classes/class.entity'; | import { Class } from '../../../src/entities/classes/class.entity'; | ||||||
|  | import { ClassStatus } from '@dwengo-1/common/util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export function makeTestTeacherInvitations(em: EntityManager, teachers: Teacher[], classes: Class[]): TeacherInvitation[] { | export function makeTestTeacherInvitations(em: EntityManager, teachers: Teacher[], classes: Class[]): TeacherInvitation[] { | ||||||
|     const teacherInvitation01 = em.create(TeacherInvitation, { |     const teacherInvitation01 = em.create(TeacherInvitation, { | ||||||
|         sender: teachers[1], |         sender: teachers[1], | ||||||
|         receiver: teachers[0], |         receiver: teachers[0], | ||||||
|         class: classes[1], |         class: classes[1], | ||||||
|  |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const teacherInvitation02 = em.create(TeacherInvitation, { |     const teacherInvitation02 = em.create(TeacherInvitation, { | ||||||
|         sender: teachers[1], |         sender: teachers[1], | ||||||
|         receiver: teachers[2], |         receiver: teachers[2], | ||||||
|         class: classes[1], |         class: classes[1], | ||||||
|  |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const teacherInvitation03 = em.create(TeacherInvitation, { |     const teacherInvitation03 = em.create(TeacherInvitation, { | ||||||
|         sender: teachers[2], |         sender: teachers[2], | ||||||
|         receiver: teachers[0], |         receiver: teachers[0], | ||||||
|         class: classes[2], |         class: classes[2], | ||||||
|  |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const teacherInvitation04 = em.create(TeacherInvitation, { |     const teacherInvitation04 = em.create(TeacherInvitation, { | ||||||
|         sender: teachers[0], |         sender: teachers[0], | ||||||
|         receiver: teachers[1], |         receiver: teachers[1], | ||||||
|         class: classes[0], |         class: classes[0], | ||||||
|  |         status: ClassStatus.Open, | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return [teacherInvitation01, teacherInvitation02, teacherInvitation03, teacherInvitation04]; |     return [teacherInvitation01, teacherInvitation02, teacherInvitation03, teacherInvitation04]; | ||||||
|  |  | ||||||
|  | @ -15,7 +15,14 @@ export const TEST_STUDENTS = [ | ||||||
|     { username: 'testleerling1', firstName: 'Gerald', lastName: 'Schmittinger' }, |     { username: 'testleerling1', firstName: 'Gerald', lastName: 'Schmittinger' }, | ||||||
| ]; | ]; | ||||||
| 
 | 
 | ||||||
|  | let testStudents: Student[]; | ||||||
|  | 
 | ||||||
| // 🏗️ Functie die ORM entities maakt uit de data array
 | // 🏗️ Functie die ORM entities maakt uit de data array
 | ||||||
| export function makeTestStudents(em: EntityManager): Student[] { | export function makeTestStudents(em: EntityManager): Student[] { | ||||||
|     return TEST_STUDENTS.map((data) => em.create(Student, data)); |     testStudents = TEST_STUDENTS.map((data) => em.create(Student, data)); | ||||||
|  |     return testStudents; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getTestleerling1(): Student { | ||||||
|  |     return testStudents.find(it => it.username == "testleerling1"); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,37 +2,64 @@ import { Teacher } from '../../../src/entities/users/teacher.entity'; | ||||||
| import { EntityManager } from '@mikro-orm/core'; | import { EntityManager } from '@mikro-orm/core'; | ||||||
| 
 | 
 | ||||||
| export function makeTestTeachers(em: EntityManager): Teacher[] { | export function makeTestTeachers(em: EntityManager): Teacher[] { | ||||||
|     const teacher01 = em.create(Teacher, { |     teacher01 = em.create(Teacher, { | ||||||
|         username: 'FooFighters', |         username: 'FooFighters', | ||||||
|         firstName: 'Dave', |         firstName: 'Dave', | ||||||
|         lastName: 'Grohl', |         lastName: 'Grohl', | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const teacher02 = em.create(Teacher, { |     teacher02 = em.create(Teacher, { | ||||||
|         username: 'LimpBizkit', |         username: 'LimpBizkit', | ||||||
|         firstName: 'Fred', |         firstName: 'Fred', | ||||||
|         lastName: 'Durst', |         lastName: 'Durst', | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const teacher03 = em.create(Teacher, { |     teacher03 = em.create(Teacher, { | ||||||
|         username: 'Staind', |         username: 'Staind', | ||||||
|         firstName: 'Aaron', |         firstName: 'Aaron', | ||||||
|         lastName: 'Lewis', |         lastName: 'Lewis', | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     // Should not be used, gets deleted in a unit test
 |     // Should not be used, gets deleted in a unit test
 | ||||||
|     const teacher04 = em.create(Teacher, { |     teacher04 = em.create(Teacher, { | ||||||
|         username: 'ZesdeMetaal', |         username: 'ZesdeMetaal', | ||||||
|         firstName: 'Wannes', |         firstName: 'Wannes', | ||||||
|         lastName: 'Cappelle', |         lastName: 'Cappelle', | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     // Makes sure when logged in as testleerkracht1, there exists a corresponding user
 |     // Makes sure when logged in as testleerkracht1, there exists a corresponding user
 | ||||||
|     const teacher05 = em.create(Teacher, { |     testleerkracht1 = em.create(Teacher, { | ||||||
|         username: 'testleerkracht1', |         username: 'testleerkracht1', | ||||||
|         firstName: 'Bob', |         firstName: 'Kris', | ||||||
|         lastName: 'Dylan', |         lastName: 'Coolsaet', | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return [teacher01, teacher02, teacher03, teacher04, teacher05]; |     return [teacher01, teacher02, teacher03, teacher04, testleerkracht1]; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | let teacher01: Teacher; | ||||||
|  | let teacher02: Teacher; | ||||||
|  | let teacher03: Teacher; | ||||||
|  | let teacher04: Teacher; | ||||||
|  | let testleerkracht1: Teacher; | ||||||
|  | 
 | ||||||
|  | export function getTeacher01(): Teacher { | ||||||
|  |     return teacher01; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getTeacher02(): Teacher { | ||||||
|  |     return teacher02; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getTeacher03(): Teacher { | ||||||
|  |     return teacher03; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getTeacher04(): Teacher { | ||||||
|  |     return teacher04; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function getTestleerkracht1(): Teacher { | ||||||
|  |     return testleerkracht1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -49,7 +49,6 @@ export async function seedDatabase(): Promise<void> { | ||||||
|     const answers = makeTestAnswers(em, teachers, questions); |     const answers = makeTestAnswers(em, teachers, questions); | ||||||
|     const submissions = makeTestSubmissions(em, students, groups); |     const submissions = makeTestSubmissions(em, students, groups); | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     // Persist all entities
 |     // Persist all entities
 | ||||||
|     await em.persistAndFlush([ |     await em.persistAndFlush([ | ||||||
|         ...students, |         ...students, | ||||||
|  |  | ||||||
|  | @ -1,8 +1,8 @@ | ||||||
| import { StudentDTO } from './student'; | import { StudentDTO } from './student'; | ||||||
| import { ClassJoinRequestStatus } from '../util/class-join-request'; | import { ClassStatus } from '../util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export interface ClassJoinRequestDTO { | export interface ClassJoinRequestDTO { | ||||||
|     requester: StudentDTO; |     requester: StudentDTO; | ||||||
|     class: string; |     class: string; | ||||||
|     status: ClassJoinRequestStatus; |     status: ClassStatus; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,8 +1,16 @@ | ||||||
| import { UserDTO } from './user'; | import { UserDTO } from './user'; | ||||||
| import { ClassDTO } from './class'; | import { ClassStatus } from '../util/class-join-request'; | ||||||
| 
 | 
 | ||||||
| export interface TeacherInvitationDTO { | export interface TeacherInvitationDTO { | ||||||
|     sender: string | UserDTO; |     sender: string | UserDTO; | ||||||
|     receiver: string | UserDTO; |     receiver: string | UserDTO; | ||||||
|     class: string | ClassDTO; |     classId: string; | ||||||
|  |     status: ClassStatus; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface TeacherInvitationData { | ||||||
|  |     sender: string; | ||||||
|  |     receiver: string; | ||||||
|  |     class: string; | ||||||
|  |     accepted?: boolean; // Use for put requests, else skip
 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,4 +1,4 @@ | ||||||
| export enum ClassJoinRequestStatus { | export enum ClassStatus { | ||||||
|     Open = 'open', |     Open = 'open', | ||||||
|     Accepted = 'accepted', |     Accepted = 'accepted', | ||||||
|     Declined = 'declined', |     Declined = 'declined', | ||||||
|  |  | ||||||
|  | @ -17,6 +17,7 @@ | ||||||
|         "test:unit": "vitest --run" |         "test:unit": "vitest --run" | ||||||
|     }, |     }, | ||||||
|     "dependencies": { |     "dependencies": { | ||||||
|  |         "@dwengo-1/common": "^0.1.1", | ||||||
|         "@tanstack/react-query": "^5.69.0", |         "@tanstack/react-query": "^5.69.0", | ||||||
|         "@tanstack/vue-query": "^5.69.0", |         "@tanstack/vue-query": "^5.69.0", | ||||||
|         "axios": "^1.8.2", |         "axios": "^1.8.2", | ||||||
|  |  | ||||||
|  | @ -2,8 +2,8 @@ import { BaseController } from "./base-controller"; | ||||||
| import type { ClassDTO } from "@dwengo-1/common/interfaces/class"; | import type { ClassDTO } from "@dwengo-1/common/interfaces/class"; | ||||||
| import type { StudentsResponse } from "./students"; | import type { StudentsResponse } from "./students"; | ||||||
| import type { AssignmentsResponse } from "./assignments"; | import type { AssignmentsResponse } from "./assignments"; | ||||||
| import type { TeacherInvitationDTO } from "@dwengo-1/common/interfaces/teacher-invitation"; |  | ||||||
| import type { TeachersResponse } from "@/controllers/teachers.ts"; | import type { TeachersResponse } from "@/controllers/teachers.ts"; | ||||||
|  | import type { TeacherInvitationsResponse } from "@/controllers/teacher-invitations.ts"; | ||||||
| 
 | 
 | ||||||
| export interface ClassesResponse { | export interface ClassesResponse { | ||||||
|     classes: ClassDTO[] | string[]; |     classes: ClassDTO[] | string[]; | ||||||
|  | @ -13,14 +13,6 @@ export interface ClassResponse { | ||||||
|     class: ClassDTO; |     class: ClassDTO; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export interface TeacherInvitationsResponse { |  | ||||||
|     invites: TeacherInvitationDTO[]; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export interface TeacherInvitationResponse { |  | ||||||
|     invite: TeacherInvitationDTO; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export class ClassController extends BaseController { | export class ClassController extends BaseController { | ||||||
|     constructor() { |     constructor() { | ||||||
|         super("class"); |         super("class"); | ||||||
|  |  | ||||||
|  | @ -36,11 +36,11 @@ export class GroupController extends BaseController { | ||||||
|         return this.put<GroupResponse>(`/${num}`, data); |         return this.put<GroupResponse>(`/${num}`, data); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async getSubmissions(groupNumber: number, full = true): Promise<SubmissionsResponse> { |     async getSubmissions(num: number, full = true): Promise<SubmissionsResponse> { | ||||||
|         return this.get<SubmissionsResponse>(`/${groupNumber}/submissions`, { full }); |         return this.get<SubmissionsResponse>(`/${num}/submissions`, { full }); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async getQuestions(groupNumber: number, full = true): Promise<QuestionsResponse> { |     async getQuestions(num: number, full = true): Promise<QuestionsResponse> { | ||||||
|         return this.get<QuestionsResponse>(`/${groupNumber}/questions`, { full }); |         return this.get<QuestionsResponse>(`/${num}/questions`, { full }); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -15,13 +15,14 @@ export class LearningPathController extends BaseController { | ||||||
|     async getBy( |     async getBy( | ||||||
|         hruid: string, |         hruid: string, | ||||||
|         language: Language, |         language: Language, | ||||||
|         options?: { forGroup?: string; forStudent?: string }, |         forGroup?: { forGroup: number, assignmentNo: number, classId: string }, | ||||||
|     ): Promise<LearningPath> { |     ): Promise<LearningPath> { | ||||||
|         const dtos = await this.get<LearningPathDTO[]>("/", { |         const dtos = await this.get<LearningPathDTO[]>("/", { | ||||||
|             hruid, |             hruid, | ||||||
|             language, |             language, | ||||||
|             forGroup: options?.forGroup, |             forGroup: forGroup?.forGroup, | ||||||
|             forStudent: options?.forStudent, |             assignmentNo: forGroup?.assignmentNo, | ||||||
|  |             classId: forGroup?.classId | ||||||
|         }); |         }); | ||||||
|         return LearningPath.fromDTO(single(dtos)); |         return LearningPath.fromDTO(single(dtos)); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| import { BaseController } from "./base-controller"; | import { BaseController } from "./base-controller"; | ||||||
| import type { SubmissionDTO, SubmissionDTOId } from "@dwengo-1/common/interfaces/submission"; | import type { SubmissionDTO, SubmissionDTOId } from "@dwengo-1/common/interfaces/submission"; | ||||||
|  | import type {Language} from "@dwengo-1/common/util/language"; | ||||||
| 
 | 
 | ||||||
| export interface SubmissionsResponse { | export interface SubmissionsResponse { | ||||||
|     submissions: SubmissionDTO[] | SubmissionDTOId[]; |     submissions: SubmissionDTO[] | SubmissionDTOId[]; | ||||||
|  | @ -10,19 +11,35 @@ export interface SubmissionResponse { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export class SubmissionController extends BaseController { | export class SubmissionController extends BaseController { | ||||||
|     constructor(classid: string, assignmentNumber: number, groupNumber: number) { | 
 | ||||||
|         super(`class/${classid}/assignments/${assignmentNumber}/groups/${groupNumber}`); |     constructor(hruid: string) { | ||||||
|  |         super(`learningObject/${hruid}/submissions`); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async getAll(full = true): Promise<SubmissionsResponse> { |     async getAll( | ||||||
|         return this.get<SubmissionsResponse>(`/`, { full }); |         language: Language, version: number, classId: string, assignmentId: number, groupId?: number, full = true | ||||||
|  |     ): Promise<SubmissionsResponse> { | ||||||
|  |         return this.get<SubmissionsResponse>( | ||||||
|  |             `/`, | ||||||
|  |             { language, version, classId, assignmentId, groupId, full } | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async getByNumber(submissionNumber: number): Promise<SubmissionResponse> { |     async getByNumber( | ||||||
|         return this.get<SubmissionResponse>(`/${submissionNumber}`); |         language: Language, | ||||||
|  |         version: number, | ||||||
|  |         classId: string, | ||||||
|  |         assignmentId: number, | ||||||
|  |         groupId: number, | ||||||
|  |         submissionNumber: number | ||||||
|  |     ): Promise<SubmissionResponse> { | ||||||
|  |         return this.get<SubmissionResponse>( | ||||||
|  |             `/${submissionNumber}`, | ||||||
|  |             { language, version, classId, assignmentId, groupId }, | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     async createSubmission(data: unknown): Promise<SubmissionResponse> { |     async createSubmission(data: SubmissionDTO): Promise<SubmissionResponse> { | ||||||
|         return this.post<SubmissionResponse>(`/`, data); |         return this.post<SubmissionResponse>(`/`, data); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										36
									
								
								frontend/src/controllers/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								frontend/src/controllers/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,36 @@ | ||||||
|  | import { BaseController } from "@/controllers/base-controller.ts"; | ||||||
|  | import type { TeacherInvitationData, TeacherInvitationDTO } from "@dwengo-1/common/interfaces/teacher-invitation"; | ||||||
|  | 
 | ||||||
|  | export interface TeacherInvitationsResponse { | ||||||
|  |     invitations: TeacherInvitationDTO[]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface TeacherInvitationResponse { | ||||||
|  |     invitation: TeacherInvitationDTO; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export class TeacherInvitationController extends BaseController { | ||||||
|  |     constructor() { | ||||||
|  |         super("teachers/invitations"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     async getAll(username: string, sent: boolean): Promise<TeacherInvitationsResponse> { | ||||||
|  |         return this.get<TeacherInvitationsResponse>(`/${username}`, { sent }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     async getBy(data: TeacherInvitationData): Promise<TeacherInvitationResponse> { | ||||||
|  |         return this.get<TeacherInvitationResponse>(`/${data.sender}/${data.receiver}/${data.class}`); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     async create(data: TeacherInvitationData): Promise<TeacherInvitationResponse> { | ||||||
|  |         return this.post<TeacherInvitationResponse>("/", data); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     async remove(data: TeacherInvitationData): Promise<TeacherInvitationResponse> { | ||||||
|  |         return this.delete<TeacherInvitationResponse>(`/${data.sender}/${data.receiver}/${data.class}`); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     async respond(data: TeacherInvitationData): Promise<TeacherInvitationResponse> { | ||||||
|  |         return this.put<TeacherInvitationResponse>("/", data); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										188
									
								
								frontend/src/queries/assignments.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										188
									
								
								frontend/src/queries/assignments.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,188 @@ | ||||||
|  | import { AssignmentController, type AssignmentResponse, type AssignmentsResponse } from "@/controllers/assignments"; | ||||||
|  | import type { QuestionsResponse } from "@/controllers/questions"; | ||||||
|  | import type { SubmissionsResponse } from "@/controllers/submissions"; | ||||||
|  | import { | ||||||
|  |     useMutation, | ||||||
|  |     useQuery, | ||||||
|  |     useQueryClient, | ||||||
|  |     type UseMutationReturnType, | ||||||
|  |     type UseQueryReturnType, | ||||||
|  | } from "@tanstack/vue-query"; | ||||||
|  | import { computed, toValue, type MaybeRefOrGetter } from "vue"; | ||||||
|  | import { groupsQueryKey, invalidateAllGroupKeys } from "./groups"; | ||||||
|  | import type { GroupsResponse } from "@/controllers/groups"; | ||||||
|  | import type { AssignmentDTO } from "@dwengo-1/common/interfaces/assignment"; | ||||||
|  | import type { QueryClient } from "@tanstack/react-query"; | ||||||
|  | import { invalidateAllSubmissionKeys } from "./submissions"; | ||||||
|  | 
 | ||||||
|  | function assignmentsQueryKey(classid: string, full: boolean) { | ||||||
|  |     return ["assignments", classid, full]; | ||||||
|  | } | ||||||
|  | function assignmentQueryKey(classid: string, assignmentNumber: number) { | ||||||
|  |     return ["assignment", classid, assignmentNumber]; | ||||||
|  | } | ||||||
|  | function assignmentSubmissionsQueryKey(classid: string, assignmentNumber: number, full: boolean) { | ||||||
|  |     return ["assignment-submissions", classid, assignmentNumber, full]; | ||||||
|  | } | ||||||
|  | function assignmentQuestionsQueryKey(classid: string, assignmentNumber: number, full: boolean) { | ||||||
|  |     return ["assignment-questions", classid, assignmentNumber, full]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function invalidateAllAssignmentKeys( | ||||||
|  |     queryClient: QueryClient, | ||||||
|  |     classid?: string, | ||||||
|  |     assignmentNumber?: number, | ||||||
|  | ) { | ||||||
|  |     const keys = ["assignment", "assignment-submissions", "assignment-questions"]; | ||||||
|  | 
 | ||||||
|  |     for (const key of keys) { | ||||||
|  |         const queryKey = [key, classid, assignmentNumber].filter((arg) => arg !== undefined); | ||||||
|  |         await queryClient.invalidateQueries({ queryKey: queryKey }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     await queryClient.invalidateQueries({ queryKey: ["assignments", classid].filter((arg) => arg !== undefined) }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function checkEnabled( | ||||||
|  |     classid: string | undefined, | ||||||
|  |     assignmentNumber: number | undefined, | ||||||
|  |     groupNumber: number | undefined, | ||||||
|  | ): boolean { | ||||||
|  |     return Boolean(classid) && !isNaN(Number(groupNumber)) && !isNaN(Number(assignmentNumber)); | ||||||
|  | } | ||||||
|  | function toValues( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean>, | ||||||
|  | ) { | ||||||
|  |     return { cid: toValue(classid), an: toValue(assignmentNumber), gn: toValue(groupNumber), f: toValue(full) }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useAssignmentsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<AssignmentsResponse, Error> { | ||||||
|  |     const { cid, f } = toValues(classid, 1, 1, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => assignmentsQueryKey(cid!, f)), | ||||||
|  |         queryFn: async () => new AssignmentController(cid!).getAll(f), | ||||||
|  |         enabled: () => checkEnabled(cid, 1, 1), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useAssignmentQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  | ): UseQueryReturnType<AssignmentsResponse, Error> { | ||||||
|  |     const { cid, an } = toValues(classid, assignmentNumber, 1, true); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => assignmentQueryKey(cid!, an!)), | ||||||
|  |         queryFn: async () => new AssignmentController(cid!).getByNumber(an!), | ||||||
|  |         enabled: () => checkEnabled(cid, an, 1), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useCreateAssignmentMutation(): UseMutationReturnType< | ||||||
|  |     AssignmentResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; data: AssignmentDTO }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, data }) => new AssignmentController(cid).createAssignment(data), | ||||||
|  |         onSuccess: async (_) => { | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: ["assignments"] }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useDeleteAssignmentMutation(): UseMutationReturnType< | ||||||
|  |     AssignmentResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; an: number }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, an }) => new AssignmentController(cid).deleteAssignment(an), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             const cid = response.assignment.within; | ||||||
|  |             const an = response.assignment.id; | ||||||
|  | 
 | ||||||
|  |             await invalidateAllAssignmentKeys(queryClient, cid, an); | ||||||
|  |             await invalidateAllGroupKeys(queryClient, cid, an); | ||||||
|  |             await invalidateAllSubmissionKeys(queryClient, cid, an); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useUpdateAssignmentMutation(): UseMutationReturnType< | ||||||
|  |     AssignmentResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; an: number; data: Partial<AssignmentDTO> }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, an, data }) => new AssignmentController(cid).updateAssignment(an, data), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             const cid = response.assignment.within; | ||||||
|  |             const an = response.assignment.id; | ||||||
|  | 
 | ||||||
|  |             await invalidateAllGroupKeys(queryClient, cid, an); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: ["assignments"] }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useAssignmentSubmissionsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<SubmissionsResponse, Error> { | ||||||
|  |     const { cid, an, gn, f } = toValues(classid, assignmentNumber, groupNumber, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => assignmentSubmissionsQueryKey(cid!, an!, f)), | ||||||
|  |         queryFn: async () => new AssignmentController(cid!).getSubmissions(gn!, f), | ||||||
|  |         enabled: () => checkEnabled(cid, an, gn), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useAssignmentQuestionsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<QuestionsResponse, Error> { | ||||||
|  |     const { cid, an, gn, f } = toValues(classid, assignmentNumber, groupNumber, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => assignmentQuestionsQueryKey(cid!, an!, f)), | ||||||
|  |         queryFn: async () => new AssignmentController(cid!).getQuestions(gn!, f), | ||||||
|  |         enabled: () => checkEnabled(cid, an, gn), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useAssignmentGroupsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<GroupsResponse, Error> { | ||||||
|  |     const { cid, an, gn, f } = toValues(classid, assignmentNumber, groupNumber, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => groupsQueryKey(cid!, an!, f)), | ||||||
|  |         queryFn: async () => new AssignmentController(cid!).getQuestions(gn!, f), | ||||||
|  |         enabled: () => checkEnabled(cid, an, gn), | ||||||
|  |     }); | ||||||
|  | } | ||||||
							
								
								
									
										224
									
								
								frontend/src/queries/classes.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										224
									
								
								frontend/src/queries/classes.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,224 @@ | ||||||
|  | import { ClassController, type ClassesResponse, type ClassResponse } from "@/controllers/classes"; | ||||||
|  | import type { StudentsResponse } from "@/controllers/students"; | ||||||
|  | import type { ClassDTO } from "@dwengo-1/common/interfaces/class"; | ||||||
|  | import { | ||||||
|  |     QueryClient, | ||||||
|  |     useMutation, | ||||||
|  |     useQuery, | ||||||
|  |     useQueryClient, | ||||||
|  |     type UseMutationReturnType, | ||||||
|  |     type UseQueryReturnType, | ||||||
|  | } from "@tanstack/vue-query"; | ||||||
|  | import { computed, toValue, type MaybeRefOrGetter } from "vue"; | ||||||
|  | import { invalidateAllAssignmentKeys } from "./assignments"; | ||||||
|  | import { invalidateAllGroupKeys } from "./groups"; | ||||||
|  | import { invalidateAllSubmissionKeys } from "./submissions"; | ||||||
|  | 
 | ||||||
|  | const classController = new ClassController(); | ||||||
|  | 
 | ||||||
|  | /* Query cache keys */ | ||||||
|  | function classesQueryKey(full: boolean) { | ||||||
|  |     return ["classes", full]; | ||||||
|  | } | ||||||
|  | function classQueryKey(classid: string) { | ||||||
|  |     return ["class", classid]; | ||||||
|  | } | ||||||
|  | function classStudentsKey(classid: string, full: boolean) { | ||||||
|  |     return ["class-students", classid, full]; | ||||||
|  | } | ||||||
|  | function classTeachersKey(classid: string, full: boolean) { | ||||||
|  |     return ["class-teachers", classid, full]; | ||||||
|  | } | ||||||
|  | function classTeacherInvitationsKey(classid: string, full: boolean) { | ||||||
|  |     return ["class-teacher-invitations", classid, full]; | ||||||
|  | } | ||||||
|  | function classAssignmentsKey(classid: string, full: boolean) { | ||||||
|  |     return ["class-assignments", classid, full]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function invalidateAllClassKeys(queryClient: QueryClient, classid?: string) { | ||||||
|  |     const keys = ["class", "class-students", "class-teachers", "class-teacher-invitations", "class-assignments"]; | ||||||
|  | 
 | ||||||
|  |     for (const key of keys) { | ||||||
|  |         const queryKey = [key, classid].filter((arg) => arg !== undefined); | ||||||
|  |         await queryClient.invalidateQueries({ queryKey: queryKey }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     await queryClient.invalidateQueries({ queryKey: ["classes"] }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Queries */ | ||||||
|  | export function useClassesQuery(full: MaybeRefOrGetter<boolean> = true): UseQueryReturnType<ClassesResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => classesQueryKey(toValue(full))), | ||||||
|  |         queryFn: async () => classController.getAll(toValue(full)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassQuery(id: MaybeRefOrGetter<string | undefined>): UseQueryReturnType<ClassResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => classQueryKey(toValue(id)!)), | ||||||
|  |         queryFn: async () => classController.getById(toValue(id)!), | ||||||
|  |         enabled: () => Boolean(toValue(id)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useCreateClassMutation(): UseMutationReturnType<ClassResponse, Error, ClassDTO, unknown> { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async (data) => classController.createClass(data), | ||||||
|  |         onSuccess: async () => { | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: ["classes"] }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useDeleteClassMutation(): UseMutationReturnType<ClassResponse, Error, string, unknown> { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async (id) => classController.deleteClass(id), | ||||||
|  |         onSuccess: async (data) => { | ||||||
|  |             await invalidateAllClassKeys(queryClient, data.class.id); | ||||||
|  |             await invalidateAllAssignmentKeys(queryClient, data.class.id); | ||||||
|  |             await invalidateAllGroupKeys(queryClient, data.class.id); | ||||||
|  |             await invalidateAllSubmissionKeys(queryClient, data.class.id); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useUpdateClassMutation(): UseMutationReturnType< | ||||||
|  |     ClassResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; data: Partial<ClassDTO> }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, data }) => classController.updateClass(cid, data), | ||||||
|  |         onSuccess: async (data) => { | ||||||
|  |             await invalidateAllClassKeys(queryClient, data.class.id); | ||||||
|  |             await invalidateAllAssignmentKeys(queryClient, data.class.id); | ||||||
|  |             await invalidateAllGroupKeys(queryClient, data.class.id); | ||||||
|  |             await invalidateAllSubmissionKeys(queryClient, data.class.id); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassStudentsQuery( | ||||||
|  |     id: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<StudentsResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => classStudentsKey(toValue(id)!, toValue(full))), | ||||||
|  |         queryFn: async () => classController.getStudents(toValue(id)!, toValue(full)), | ||||||
|  |         enabled: () => Boolean(toValue(id)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassAddStudentMutation(): UseMutationReturnType< | ||||||
|  |     ClassResponse, | ||||||
|  |     Error, | ||||||
|  |     { id: string; username: string }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ id, username }) => classController.addStudent(id, username), | ||||||
|  |         onSuccess: async (data) => { | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classQueryKey(data.class.id) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classStudentsKey(data.class.id, true) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classStudentsKey(data.class.id, false) }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassDeleteStudentMutation(): UseMutationReturnType< | ||||||
|  |     ClassResponse, | ||||||
|  |     Error, | ||||||
|  |     { id: string; username: string }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ id, username }) => classController.deleteStudent(id, username), | ||||||
|  |         onSuccess: async (data) => { | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classQueryKey(data.class.id) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classStudentsKey(data.class.id, true) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classStudentsKey(data.class.id, false) }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassTeachersQuery( | ||||||
|  |     id: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<StudentsResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => classTeachersKey(toValue(id)!, toValue(full))), | ||||||
|  |         queryFn: async () => classController.getTeachers(toValue(id)!, toValue(full)), | ||||||
|  |         enabled: () => Boolean(toValue(id)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassAddTeacherMutation(): UseMutationReturnType< | ||||||
|  |     ClassResponse, | ||||||
|  |     Error, | ||||||
|  |     { id: string; username: string }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ id, username }) => classController.addTeacher(id, username), | ||||||
|  |         onSuccess: async (data) => { | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classQueryKey(data.class.id) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classTeachersKey(data.class.id, true) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classTeachersKey(data.class.id, false) }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassDeleteTeacherMutation(): UseMutationReturnType< | ||||||
|  |     ClassResponse, | ||||||
|  |     Error, | ||||||
|  |     { id: string; username: string }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ id, username }) => classController.deleteTeacher(id, username), | ||||||
|  |         onSuccess: async (data) => { | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classQueryKey(data.class.id) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classTeachersKey(data.class.id, true) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: classTeachersKey(data.class.id, false) }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassTeacherInvitationsQuery( | ||||||
|  |     id: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<StudentsResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => classTeacherInvitationsKey(toValue(id)!, toValue(full))), | ||||||
|  |         queryFn: async () => classController.getTeacherInvitations(toValue(id)!, toValue(full)), | ||||||
|  |         enabled: () => Boolean(toValue(id)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useClassAssignmentsQuery( | ||||||
|  |     id: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<StudentsResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => classAssignmentsKey(toValue(id)!, toValue(full))), | ||||||
|  |         queryFn: async () => classController.getAssignments(toValue(id)!, toValue(full)), | ||||||
|  |         enabled: () => Boolean(toValue(id)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
							
								
								
									
										191
									
								
								frontend/src/queries/groups.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										191
									
								
								frontend/src/queries/groups.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,191 @@ | ||||||
|  | import type { ClassesResponse } from "@/controllers/classes"; | ||||||
|  | import { GroupController, type GroupResponse, type GroupsResponse } from "@/controllers/groups"; | ||||||
|  | import type { QuestionsResponse } from "@/controllers/questions"; | ||||||
|  | import type { SubmissionsResponse } from "@/controllers/submissions"; | ||||||
|  | import type { GroupDTO } from "@dwengo-1/common/interfaces/group"; | ||||||
|  | import { | ||||||
|  |     QueryClient, | ||||||
|  |     useMutation, | ||||||
|  |     useQuery, | ||||||
|  |     useQueryClient, | ||||||
|  |     type UseMutationReturnType, | ||||||
|  |     type UseQueryReturnType, | ||||||
|  | } from "@tanstack/vue-query"; | ||||||
|  | import { computed, toValue, type MaybeRefOrGetter } from "vue"; | ||||||
|  | import { invalidateAllAssignmentKeys } from "./assignments"; | ||||||
|  | import { invalidateAllSubmissionKeys } from "./submissions"; | ||||||
|  | 
 | ||||||
|  | export function groupsQueryKey(classid: string, assignmentNumber: number, full: boolean) { | ||||||
|  |     return ["groups", classid, assignmentNumber, full]; | ||||||
|  | } | ||||||
|  | function groupQueryKey(classid: string, assignmentNumber: number, groupNumber: number) { | ||||||
|  |     return ["group", classid, assignmentNumber, groupNumber]; | ||||||
|  | } | ||||||
|  | function groupSubmissionsQueryKey(classid: string, assignmentNumber: number, groupNumber: number, full: boolean) { | ||||||
|  |     return ["group-submissions", classid, assignmentNumber, groupNumber, full]; | ||||||
|  | } | ||||||
|  | function groupQuestionsQueryKey(classid: string, assignmentNumber: number, groupNumber: number, full: boolean) { | ||||||
|  |     return ["group-questions", classid, assignmentNumber, groupNumber, full]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function invalidateAllGroupKeys( | ||||||
|  |     queryClient: QueryClient, | ||||||
|  |     classid?: string, | ||||||
|  |     assignmentNumber?: number, | ||||||
|  |     groupNumber?: number, | ||||||
|  | ) { | ||||||
|  |     const keys = ["group", "group-submissions", "group-questions"]; | ||||||
|  | 
 | ||||||
|  |     for (const key of keys) { | ||||||
|  |         const queryKey = [key, classid, assignmentNumber, groupNumber].filter((arg) => arg !== undefined); | ||||||
|  |         await queryClient.invalidateQueries({ queryKey: queryKey }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     await queryClient.invalidateQueries({ | ||||||
|  |         queryKey: ["groups", classid, assignmentNumber].filter((arg) => arg !== undefined), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function checkEnabled( | ||||||
|  |     classid: string | undefined, | ||||||
|  |     assignmentNumber: number | undefined, | ||||||
|  |     groupNumber: number | undefined, | ||||||
|  | ): boolean { | ||||||
|  |     return Boolean(classid) && !isNaN(Number(groupNumber)) && !isNaN(Number(assignmentNumber)); | ||||||
|  | } | ||||||
|  | function toValues( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean>, | ||||||
|  | ) { | ||||||
|  |     return { cid: toValue(classid), an: toValue(assignmentNumber), gn: toValue(groupNumber), f: toValue(full) }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useGroupsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<GroupsResponse, Error> { | ||||||
|  |     const { cid, an, f } = toValues(classid, assignmentNumber, 1, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => groupsQueryKey(cid!, an!, f)), | ||||||
|  |         queryFn: async () => new GroupController(cid!, an!).getAll(f), | ||||||
|  |         enabled: () => checkEnabled(cid, an, 1), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useGroupQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  | ): UseQueryReturnType<GroupResponse, Error> { | ||||||
|  |     const { cid, an, gn } = toValues(classid, assignmentNumber, groupNumber, true); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => groupQueryKey(cid!, an!, gn!)), | ||||||
|  |         queryFn: async () => new GroupController(cid!, an!).getByNumber(gn!), | ||||||
|  |         enabled: () => checkEnabled(cid, an, gn), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useCreateGroupMutation(): UseMutationReturnType< | ||||||
|  |     GroupResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; an: number; data: GroupDTO }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, an, data }) => new GroupController(cid, an).createGroup(data), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             const cid = typeof response.group.class === "string" ? response.group.class : response.group.class.id; | ||||||
|  |             const an = | ||||||
|  |                 typeof response.group.assignment === "number" | ||||||
|  |                     ? response.group.assignment | ||||||
|  |                     : response.group.assignment.id; | ||||||
|  | 
 | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: groupsQueryKey(cid, an, true) }); | ||||||
|  |             await queryClient.invalidateQueries({ queryKey: groupsQueryKey(cid, an, false) }); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useDeleteGroupMutation(): UseMutationReturnType< | ||||||
|  |     GroupResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; an: number; gn: number }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, an, gn }) => new GroupController(cid, an).deleteGroup(gn), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             const cid = typeof response.group.class === "string" ? response.group.class : response.group.class.id; | ||||||
|  |             const an = | ||||||
|  |                 typeof response.group.assignment === "number" | ||||||
|  |                     ? response.group.assignment | ||||||
|  |                     : response.group.assignment.id; | ||||||
|  |             const gn = response.group.groupNumber; | ||||||
|  | 
 | ||||||
|  |             await invalidateAllGroupKeys(queryClient, cid, an, gn); | ||||||
|  |             await invalidateAllSubmissionKeys(queryClient, cid, an, gn); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useUpdateGroupMutation(): UseMutationReturnType< | ||||||
|  |     GroupResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; an: number; gn: number; data: Partial<GroupDTO> }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, an, gn, data }) => new GroupController(cid, an).updateGroup(gn, data), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             const cid = typeof response.group.class === "string" ? response.group.class : response.group.class.id; | ||||||
|  |             const an = | ||||||
|  |                 typeof response.group.assignment === "number" | ||||||
|  |                     ? response.group.assignment | ||||||
|  |                     : response.group.assignment.id; | ||||||
|  |             const gn = response.group.groupNumber; | ||||||
|  | 
 | ||||||
|  |             await invalidateAllGroupKeys(queryClient, cid, an, gn); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useGroupSubmissionsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<SubmissionsResponse, Error> { | ||||||
|  |     const { cid, an, gn, f } = toValues(classid, assignmentNumber, groupNumber, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => groupSubmissionsQueryKey(cid!, an!, gn!, f)), | ||||||
|  |         queryFn: async () => new GroupController(cid!, an!).getSubmissions(gn!, f), | ||||||
|  |         enabled: () => checkEnabled(cid, an, gn), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useGroupQuestionsQuery( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<QuestionsResponse, Error> { | ||||||
|  |     const { cid, an, gn, f } = toValues(classid, assignmentNumber, groupNumber, full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => groupQuestionsQueryKey(cid!, an!, gn!, f)), | ||||||
|  |         queryFn: async () => new GroupController(cid!, an!).getSubmissions(gn!, f), | ||||||
|  |         enabled: () => checkEnabled(cid, an, gn), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | @ -5,7 +5,7 @@ import { getLearningObjectController } from "@/controllers/controllers.ts"; | ||||||
| import type { LearningObject } from "@/data-objects/learning-objects/learning-object.ts"; | import type { LearningObject } from "@/data-objects/learning-objects/learning-object.ts"; | ||||||
| import type { LearningPath } from "@/data-objects/learning-paths/learning-path.ts"; | import type { LearningPath } from "@/data-objects/learning-paths/learning-path.ts"; | ||||||
| 
 | 
 | ||||||
| const LEARNING_OBJECT_KEY = "learningObject"; | export const LEARNING_OBJECT_KEY = "learningObject"; | ||||||
| const learningObjectController = getLearningObjectController(); | const learningObjectController = getLearningObjectController(); | ||||||
| 
 | 
 | ||||||
| export function useLearningObjectMetadataQuery( | export function useLearningObjectMetadataQuery( | ||||||
|  |  | ||||||
|  | @ -4,19 +4,21 @@ import { useQuery, type UseQueryReturnType } from "@tanstack/vue-query"; | ||||||
| import { getLearningPathController } from "@/controllers/controllers"; | import { getLearningPathController } from "@/controllers/controllers"; | ||||||
| import type { LearningPath } from "@/data-objects/learning-paths/learning-path.ts"; | import type { LearningPath } from "@/data-objects/learning-paths/learning-path.ts"; | ||||||
| 
 | 
 | ||||||
| const LEARNING_PATH_KEY = "learningPath"; | export const LEARNING_PATH_KEY = "learningPath"; | ||||||
| const learningPathController = getLearningPathController(); | const learningPathController = getLearningPathController(); | ||||||
| 
 | 
 | ||||||
| export function useGetLearningPathQuery( | export function useGetLearningPathQuery( | ||||||
|     hruid: MaybeRefOrGetter<string>, |     hruid: MaybeRefOrGetter<string>, | ||||||
|     language: MaybeRefOrGetter<Language>, |     language: MaybeRefOrGetter<Language>, | ||||||
|     options?: MaybeRefOrGetter<{ forGroup?: string; forStudent?: string }>, |     forGroup?: MaybeRefOrGetter<{forGroup: number, assignmentNo: number, classId: string}>, | ||||||
| ): UseQueryReturnType<LearningPath, Error> { | ): UseQueryReturnType<LearningPath, Error> { | ||||||
|     return useQuery({ |     return useQuery({ | ||||||
|         queryKey: [LEARNING_PATH_KEY, "get", hruid, language, options], |         queryKey: [LEARNING_PATH_KEY, "get", toValue(hruid), toValue(language), toValue(forGroup)], | ||||||
|         queryFn: async () => { |         queryFn: async () => { | ||||||
|             const [hruidVal, languageVal, optionsVal] = [toValue(hruid), toValue(language), toValue(options)]; |             console.log("queryKey"); | ||||||
|             return learningPathController.getBy(hruidVal, languageVal, optionsVal); |             console.log([LEARNING_PATH_KEY, "get", toValue(hruid), toValue(language), toValue(forGroup)]); | ||||||
|  |             const [hruidVal, languageVal, forGroupVal] = [toValue(hruid), toValue(language), toValue(forGroup)]; | ||||||
|  |             return learningPathController.getBy(hruidVal, languageVal, forGroupVal); | ||||||
|         }, |         }, | ||||||
|         enabled: () => Boolean(toValue(hruid)) && Boolean(toValue(language)), |         enabled: () => Boolean(toValue(hruid)) && Boolean(toValue(language)), | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
							
								
								
									
										244
									
								
								frontend/src/queries/submissions.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										244
									
								
								frontend/src/queries/submissions.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,244 @@ | ||||||
|  | import { SubmissionController, type SubmissionResponse, type SubmissionsResponse } from "@/controllers/submissions"; | ||||||
|  | import type { SubmissionDTO } from "@dwengo-1/common/interfaces/submission"; | ||||||
|  | import { | ||||||
|  |     QueryClient, | ||||||
|  |     useMutation, | ||||||
|  |     useQuery, | ||||||
|  |     useQueryClient, | ||||||
|  |     type UseMutationReturnType, | ||||||
|  |     type UseQueryReturnType, | ||||||
|  | } from "@tanstack/vue-query"; | ||||||
|  | import { computed, toValue, type MaybeRefOrGetter } from "vue"; | ||||||
|  | import {LEARNING_PATH_KEY} from "@/queries/learning-paths.ts"; | ||||||
|  | import {LEARNING_OBJECT_KEY} from "@/queries/learning-objects.ts"; | ||||||
|  | import type {Language} from "@dwengo-1/common/util/language"; | ||||||
|  | import {getEnvVar} from "@dwengo-1/backend/dist/util/envVars.ts"; | ||||||
|  | 
 | ||||||
|  | function submissionsQueryKey( | ||||||
|  |     hruid: string, | ||||||
|  |     language: Language, | ||||||
|  |     version: number, | ||||||
|  |     classid: string, | ||||||
|  |     assignmentNumber: number, | ||||||
|  |     groupNumber?: number, | ||||||
|  |     full?: boolean | ||||||
|  | ) { | ||||||
|  |     return ["submissions", hruid, language, version, classid, assignmentNumber, groupNumber, full ?? false]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function submissionQueryKey( | ||||||
|  |     hruid: string, | ||||||
|  |     language: Language, | ||||||
|  |     version: number, | ||||||
|  |     classid: string, | ||||||
|  |     assignmentNumber: number, | ||||||
|  |     groupNumber: number, | ||||||
|  |     submissionNumber: number | ||||||
|  | ) { | ||||||
|  |     return ["submission", hruid, language, version, classid, assignmentNumber, groupNumber, submissionNumber]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function invalidateAllSubmissionKeys( | ||||||
|  |     queryClient: QueryClient, | ||||||
|  |     hruid?: string, | ||||||
|  |     language?: Language, | ||||||
|  |     version?: number, | ||||||
|  |     classid?: string, | ||||||
|  |     assignmentNumber?: number, | ||||||
|  |     groupNumber?: number, | ||||||
|  |     submissionNumber?: number, | ||||||
|  | ) { | ||||||
|  |     const keys = ["submission"]; | ||||||
|  | 
 | ||||||
|  |     for (const key of keys) { | ||||||
|  |         const queryKey = [ | ||||||
|  |             key, hruid, language, version, classid, assignmentNumber, groupNumber, submissionNumber | ||||||
|  |         ].filter( | ||||||
|  |             (arg) => arg !== undefined, | ||||||
|  |         ); | ||||||
|  |         await queryClient.invalidateQueries({ queryKey: queryKey }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     await queryClient.invalidateQueries({ | ||||||
|  |         queryKey: ["submissions", hruid, language, version, classid, assignmentNumber, groupNumber] | ||||||
|  |             .filter((arg) => arg !== undefined), | ||||||
|  |     }); | ||||||
|  |     await queryClient.invalidateQueries({ | ||||||
|  |         queryKey: ["group-submissions", hruid, language, version, classid, assignmentNumber, groupNumber] | ||||||
|  |             .filter((arg) => arg !== undefined), | ||||||
|  |     }); | ||||||
|  |     await queryClient.invalidateQueries({ | ||||||
|  |         queryKey: ["assignment-submissions", hruid, language, version,classid, assignmentNumber] | ||||||
|  |             .filter((arg) => arg !== undefined), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function checkEnabled( | ||||||
|  |     classid: string | undefined, | ||||||
|  |     assignmentNumber: number | undefined, | ||||||
|  |     groupNumber: number | undefined, | ||||||
|  |     submissionNumber?: number | undefined, | ||||||
|  |     submissionNumberRequired: boolean = false | ||||||
|  | ): boolean { | ||||||
|  |     return ( | ||||||
|  |         Boolean(classid) && | ||||||
|  |         !isNaN(Number(groupNumber)) && | ||||||
|  |         !isNaN(Number(assignmentNumber)) && | ||||||
|  |         (!isNaN(Number(submissionNumber)) || !submissionNumberRequired) | ||||||
|  |     ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function toValues( | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     submissionNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean>, | ||||||
|  | ) { | ||||||
|  |     return { | ||||||
|  |         cid: toValue(classid), | ||||||
|  |         an: toValue(assignmentNumber), | ||||||
|  |         gn: toValue(groupNumber), | ||||||
|  |         sn: toValue(submissionNumber), | ||||||
|  |         f: toValue(full), | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | export function useSubmissionsQuery( | ||||||
|  |     hruid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     language: MaybeRefOrGetter<Language | undefined>, | ||||||
|  |     version: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     full: MaybeRefOrGetter<boolean> = true, | ||||||
|  | ): UseQueryReturnType<SubmissionsResponse, Error> { | ||||||
|  |     const hruidVal = toValue(hruid); | ||||||
|  |     const languageVal = toValue(language); | ||||||
|  |     const versionVal = toValue(version); | ||||||
|  |     const classIdVal = toValue(classid); | ||||||
|  |     const assignmentNumberVal = toValue(assignmentNumber); | ||||||
|  |     const groupNumberVal = toValue(groupNumber); | ||||||
|  |     const fullVal = toValue(full); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => | ||||||
|  |             submissionsQueryKey( | ||||||
|  |                 hruidVal!, | ||||||
|  |                 languageVal!, | ||||||
|  |                 versionVal!, | ||||||
|  |                 classIdVal!, | ||||||
|  |                 assignmentNumberVal!, | ||||||
|  |                 groupNumberVal, | ||||||
|  |                 fullVal | ||||||
|  |             ) | ||||||
|  |         ), | ||||||
|  |         queryFn: async () => new SubmissionController(hruidVal!).getAll( | ||||||
|  |             languageVal!, versionVal!, classIdVal!, assignmentNumberVal!, groupNumberVal, fullVal | ||||||
|  |         ), | ||||||
|  |         enabled: () => !!hruidVal && !!languageVal && !!versionVal && !!classIdVal && !!assignmentNumberVal, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useSubmissionQuery( | ||||||
|  |     hruid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     language: MaybeRefOrGetter<Language | undefined>, | ||||||
|  |     version: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     classid: MaybeRefOrGetter<string | undefined>, | ||||||
|  |     assignmentNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     groupNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  |     submissionNumber: MaybeRefOrGetter<number | undefined>, | ||||||
|  | ): UseQueryReturnType<SubmissionResponse, Error> { | ||||||
|  |     const { cid, an, gn, sn, f } = toValues(classid, assignmentNumber, groupNumber, submissionNumber, true); | ||||||
|  | 
 | ||||||
|  |     const hruidVal = toValue(hruid); | ||||||
|  |     const languageVal = toValue(language); | ||||||
|  |     const versionVal = toValue(version); | ||||||
|  |     const classIdVal = toValue(classid); | ||||||
|  |     const assignmentNumberVal = toValue(assignmentNumber); | ||||||
|  |     const groupNumberVal = toValue(groupNumber); | ||||||
|  |     const submissionNumberVal = toValue(submissionNumber); | ||||||
|  | 
 | ||||||
|  |     return useQuery({ | ||||||
|  |         queryKey: computed(() => submissionQueryKey( | ||||||
|  |             hruidVal!, languageVal!, versionVal!, classIdVal!, assignmentNumberVal!, groupNumberVal!, submissionNumberVal! | ||||||
|  |         )), | ||||||
|  |         queryFn: async () => new SubmissionController(hruidVal!).getByNumber( | ||||||
|  |             languageVal!, versionVal!, classIdVal!, assignmentNumberVal!, groupNumberVal!, submissionNumberVal! | ||||||
|  |         ), | ||||||
|  |         enabled: () => !!hruidVal && !!languageVal && !!versionVal && !!classIdVal && !!assignmentNumberVal && !!submissionNumber, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useCreateSubmissionMutation(): UseMutationReturnType< | ||||||
|  |     SubmissionResponse, | ||||||
|  |     Error, | ||||||
|  |     { data: SubmissionDTO }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ data }) => new SubmissionController(data.learningObjectIdentifier.hruid).createSubmission(data), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             if (!response.submission.group) { | ||||||
|  |                 await invalidateAllSubmissionKeys(queryClient); | ||||||
|  |             } else { | ||||||
|  |                 const cls = response.submission.group.class; | ||||||
|  |                 const assignment = response.submission.group.assignment; | ||||||
|  | 
 | ||||||
|  |                 const cid = typeof cls === "string" ? cls : cls.id; | ||||||
|  |                 const an = typeof assignment === "number" ? assignment : assignment.id; | ||||||
|  |                 const gn = response.submission.group.groupNumber; | ||||||
|  | 
 | ||||||
|  |                 const {hruid, language, version} = response.submission.learningObjectIdentifier; | ||||||
|  |                 await invalidateAllSubmissionKeys(queryClient, hruid, language, version, cid, an, gn); | ||||||
|  | 
 | ||||||
|  |                 console.log("INVALIDATE"); | ||||||
|  |                 console.log([ | ||||||
|  |                     LEARNING_PATH_KEY, "get", | ||||||
|  |                     response.submission.learningObjectIdentifier.hruid, | ||||||
|  |                 ]); | ||||||
|  |                 await queryClient.invalidateQueries({queryKey: [LEARNING_PATH_KEY, "get"]}); | ||||||
|  | 
 | ||||||
|  |                 await queryClient.invalidateQueries({ | ||||||
|  |                     queryKey: [LEARNING_OBJECT_KEY, "metadata", hruid, language, version] | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useDeleteSubmissionMutation(): UseMutationReturnType< | ||||||
|  |     SubmissionResponse, | ||||||
|  |     Error, | ||||||
|  |     { cid: string; an: number; gn: number; sn: number }, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  | 
 | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async ({ cid, an, gn, sn }) => new SubmissionController(cid).deleteSubmission(sn), | ||||||
|  |         onSuccess: async (response) => { | ||||||
|  |             if (!response.submission.group) { | ||||||
|  |                 await invalidateAllSubmissionKeys(queryClient); | ||||||
|  |             } else { | ||||||
|  |                 const cls = response.submission.group.class; | ||||||
|  |                 const assignment = response.submission.group.assignment; | ||||||
|  | 
 | ||||||
|  |                 const cid = typeof cls === "string" ? cls : cls.id; | ||||||
|  |                 const an = typeof assignment === "number" ? assignment : assignment.id; | ||||||
|  |                 const gn = response.submission.group.groupNumber; | ||||||
|  | 
 | ||||||
|  |                 const {hruid, language, version} = response.submission.learningObjectIdentifier; | ||||||
|  | 
 | ||||||
|  |                 await invalidateAllSubmissionKeys( | ||||||
|  |                     queryClient, | ||||||
|  |                     hruid, | ||||||
|  |                     language, | ||||||
|  |                     version, | ||||||
|  |                     cid, an, gn | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  | } | ||||||
							
								
								
									
										78
									
								
								frontend/src/queries/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								frontend/src/queries/teacher-invitations.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,78 @@ | ||||||
|  | import { useMutation, useQuery, type UseMutationReturnType, type UseQueryReturnType } from "@tanstack/vue-query"; | ||||||
|  | import { computed, toValue } from "vue"; | ||||||
|  | import type { MaybeRefOrGetter } from "vue"; | ||||||
|  | import { | ||||||
|  |     TeacherInvitationController, | ||||||
|  |     type TeacherInvitationResponse, | ||||||
|  |     type TeacherInvitationsResponse, | ||||||
|  | } from "@/controllers/teacher-invitations.ts"; | ||||||
|  | import type { TeacherInvitationData } from "@dwengo-1/common/interfaces/teacher-invitation"; | ||||||
|  | import type { TeacherDTO } from "@dwengo-1/common/interfaces/teacher"; | ||||||
|  | 
 | ||||||
|  | const controller = new TeacherInvitationController(); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |     All the invitations the teacher sent | ||||||
|  | **/ | ||||||
|  | export function useTeacherInvitationsSentQuery( | ||||||
|  |     username: MaybeRefOrGetter<string | undefined>, | ||||||
|  | ): UseQueryReturnType<TeacherInvitationsResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryFn: computed(async () => controller.getAll(toValue(username), true)), | ||||||
|  |         enabled: () => Boolean(toValue(username)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |     All the pending invitations sent to this teacher | ||||||
|  |  */ | ||||||
|  | export function useTeacherInvitationsReceivedQuery( | ||||||
|  |     username: MaybeRefOrGetter<string | undefined>, | ||||||
|  | ): UseQueryReturnType<TeacherInvitationsResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryFn: computed(async () => controller.getAll(toValue(username), false)), | ||||||
|  |         enabled: () => Boolean(toValue(username)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useTeacherInvitationQuery( | ||||||
|  |     data: MaybeRefOrGetter<TeacherInvitationData | undefined>, | ||||||
|  | ): UseQueryReturnType<TeacherInvitationResponse, Error> { | ||||||
|  |     return useQuery({ | ||||||
|  |         queryFn: computed(async () => controller.getBy(toValue(data))), | ||||||
|  |         enabled: () => Boolean(toValue(data)), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useCreateTeacherInvitationMutation(): UseMutationReturnType< | ||||||
|  |     TeacherInvitationResponse, | ||||||
|  |     Error, | ||||||
|  |     TeacherDTO, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async (data: TeacherInvitationData) => controller.create(data), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useRespondTeacherInvitationMutation(): UseMutationReturnType< | ||||||
|  |     TeacherInvitationResponse, | ||||||
|  |     Error, | ||||||
|  |     TeacherDTO, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async (data: TeacherInvitationData) => controller.respond(data), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function useDeleteTeacherInvitationMutation(): UseMutationReturnType< | ||||||
|  |     TeacherInvitationResponse, | ||||||
|  |     Error, | ||||||
|  |     TeacherDTO, | ||||||
|  |     unknown | ||||||
|  | > { | ||||||
|  |     return useMutation({ | ||||||
|  |         mutationFn: async (data: TeacherInvitationData) => controller.remove(data), | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  | @ -276,10 +276,11 @@ | ||||||
|                 <tbody> |                 <tbody> | ||||||
|                     <tr |                     <tr | ||||||
|                         v-for="i in invitations" |                         v-for="i in invitations" | ||||||
|                         :key="(i.class as ClassDTO).id" |                         :key="i.classId" | ||||||
|                     > |                     > | ||||||
|                         <td> |                         <td> | ||||||
|                             {{ (i.class as ClassDTO).displayName }} |                             {{ i.classId }} | ||||||
|  |                             <!-- TODO fetch display name via classId because db only returns classId field --> | ||||||
|                         </td> |                         </td> | ||||||
|                         <td>{{ (i.sender as TeacherDTO).firstName + " " + (i.sender as TeacherDTO).lastName }}</td> |                         <td>{{ (i.sender as TeacherDTO).firstName + " " + (i.sender as TeacherDTO).lastName }}</td> | ||||||
|                         <td class="text-right"> |                         <td class="text-right"> | ||||||
|  |  | ||||||
|  | @ -3,10 +3,24 @@ | ||||||
|     import type { UseQueryReturnType } from "@tanstack/vue-query"; |     import type { UseQueryReturnType } from "@tanstack/vue-query"; | ||||||
|     import { useLearningObjectHTMLQuery } from "@/queries/learning-objects.ts"; |     import { useLearningObjectHTMLQuery } from "@/queries/learning-objects.ts"; | ||||||
|     import UsingQueryResult from "@/components/UsingQueryResult.vue"; |     import UsingQueryResult from "@/components/UsingQueryResult.vue"; | ||||||
|     import {nextTick, onMounted, reactive, watch} from "vue"; |     import {computed, nextTick, onMounted, reactive, watch} from "vue"; | ||||||
|     import {getGiftAdapterForType} from "@/views/learning-paths/gift-adapters/gift-adapters.ts"; |     import {getGiftAdapterForType} from "@/views/learning-paths/gift-adapters/gift-adapters.ts"; | ||||||
|  |     import authService from "@/services/auth/auth-service.ts"; | ||||||
|  |     import {useCreateSubmissionMutation, useSubmissionsQuery} from "@/queries/submissions.ts"; | ||||||
|  |     import type {SubmissionDTO} from "@dwengo-1/common/dist/interfaces/submission.d.ts"; | ||||||
|  |     import type {GroupDTO} from "@dwengo-1/common/interfaces/group"; | ||||||
|  |     import type {StudentDTO} from "@dwengo-1/common/interfaces/student"; | ||||||
|  |     import type {LearningObjectIdentifierDTO} from "@dwengo-1/common/interfaces/learning-content"; | ||||||
|  |     import type {User, UserProfile} from "oidc-client-ts"; | ||||||
| 
 | 
 | ||||||
|     const props = defineProps<{ hruid: string; language: Language; version: number }>(); |     const isStudent = computed(() => authService.authState.activeRole === "student"); | ||||||
|  | 
 | ||||||
|  |     const props = defineProps<{ | ||||||
|  |         hruid: string; | ||||||
|  |         language: Language; | ||||||
|  |         version: number, | ||||||
|  |         group?: {forGroup: number, assignmentNo: number, classId: string} | ||||||
|  |     }>(); | ||||||
| 
 | 
 | ||||||
|     const learningObjectHtmlQueryResult: UseQueryReturnType<Document, Error> = useLearningObjectHTMLQuery( |     const learningObjectHtmlQueryResult: UseQueryReturnType<Document, Error> = useLearningObjectHTMLQuery( | ||||||
|         () => props.hruid, |         () => props.hruid, | ||||||
|  | @ -14,7 +28,61 @@ | ||||||
|         () => props.version, |         () => props.version, | ||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     const currentAnswer = reactive([]); |     const currentAnswer = reactive(<(string | number | object)[]>[]); | ||||||
|  | 
 | ||||||
|  |     const { | ||||||
|  |         isPending: submissionIsPending, | ||||||
|  |         isError: submissionFailed, | ||||||
|  |         error: submissionError, | ||||||
|  |         isSuccess: submissionSuccess, | ||||||
|  |         mutate: submitSolution | ||||||
|  |     } = useCreateSubmissionMutation(); | ||||||
|  | 
 | ||||||
|  |     const { | ||||||
|  |         isPending: existingSubmissionsIsPending, | ||||||
|  |         isError: existingSubmissionsFailed, | ||||||
|  |         error: existingSubmissionsError, | ||||||
|  |         isSuccess: existingSubmissionsSuccess, | ||||||
|  |         data: existingSubmissions | ||||||
|  |     } = useSubmissionsQuery( | ||||||
|  |         props.hruid, | ||||||
|  |         props.language, | ||||||
|  |         props.version, | ||||||
|  |         props.group?.classId, | ||||||
|  |         props.group?.assignmentNo, | ||||||
|  |         props.group?.forGroup, | ||||||
|  |         true | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     function submitCurrentAnswer(): void { | ||||||
|  |         const { forGroup, assignmentNo, classId } = props.group!; | ||||||
|  |         const currentUser: UserProfile = authService.authState.user!.profile; | ||||||
|  |         const learningObjectIdentifier: LearningObjectIdentifierDTO = { | ||||||
|  |             hruid: props.hruid, | ||||||
|  |             language: props.language as Language, | ||||||
|  |             version: props.version | ||||||
|  |         }; | ||||||
|  |         const submitter: StudentDTO = { | ||||||
|  |             id: currentUser.preferred_username!, | ||||||
|  |             username: currentUser.preferred_username!, | ||||||
|  |             firstName: currentUser.given_name!, | ||||||
|  |             lastName: currentUser.family_name! | ||||||
|  |         }; | ||||||
|  |         const group: GroupDTO = { | ||||||
|  |             class: classId, | ||||||
|  |             assignment: assignmentNo, | ||||||
|  |             groupNumber: forGroup | ||||||
|  |         } | ||||||
|  |         const submission: SubmissionDTO = { | ||||||
|  |             learningObjectIdentifier, | ||||||
|  |             submitter, | ||||||
|  |             group, | ||||||
|  |             content: JSON.stringify(currentAnswer) | ||||||
|  |         } | ||||||
|  |         submitSolution({ data: submission }); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     function forEachQuestion( |     function forEachQuestion( | ||||||
|         doAction: (questionIndex: number, questionName: string, questionType: string, questionElement: Element) => void |         doAction: (questionIndex: number, questionName: string, questionType: string, questionElement: Element) => void | ||||||
|  | @ -22,9 +90,9 @@ | ||||||
|         const questions = document.querySelectorAll(".gift-question"); |         const questions = document.querySelectorAll(".gift-question"); | ||||||
|         questions.forEach(question => { |         questions.forEach(question => { | ||||||
|             const name = question.id.match(/gift-q(\d+)/)?.[1] |             const name = question.id.match(/gift-q(\d+)/)?.[1] | ||||||
|             const questionType = question.classList.values() |             const questionType = question.className.split(" ") | ||||||
|                 .find(it => it.startsWith("gift-question-type")) |                 .find(it => it.startsWith("gift-question-type")) | ||||||
|                 .match(/gift-question-type-([^ ]*)/)?.[1]; |                 ?.match(/gift-question-type-([^ ]*)/)?.[1]; | ||||||
| 
 | 
 | ||||||
|             if (!name || isNaN(parseInt(name)) || !questionType) return; |             if (!name || isNaN(parseInt(name)) || !questionType) return; | ||||||
| 
 | 
 | ||||||
|  | @ -46,7 +114,7 @@ | ||||||
|         forEachQuestion((index, name, type, element) => { |         forEachQuestion((index, name, type, element) => { | ||||||
|             getGiftAdapterForType(type)?.setAnswer(element, answers[index]); |             getGiftAdapterForType(type)?.setAnswer(element, answers[index]); | ||||||
|         }); |         }); | ||||||
|         currentAnswer.fill(answers); |         currentAnswer.splice(0, currentAnswer.length, ...answers); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     onMounted(() => nextTick(() => attachQuestionListeners())); |     onMounted(() => nextTick(() => attachQuestionListeners())); | ||||||
|  | @ -68,6 +136,14 @@ | ||||||
|             v-html="learningPathHtml.data.body.innerHTML" |             v-html="learningPathHtml.data.body.innerHTML" | ||||||
|         ></div> |         ></div> | ||||||
|         {{ currentAnswer }} |         {{ currentAnswer }} | ||||||
|  |         <v-btn v-if="isStudent && props.group" | ||||||
|  |                prepend-icon="mdi-check" | ||||||
|  |                variant="elevated" | ||||||
|  |                :loading="submissionIsPending" | ||||||
|  |                @click="submitCurrentAnswer()" | ||||||
|  |         > | ||||||
|  |             Submit | ||||||
|  |         </v-btn> | ||||||
|     </using-query-result> |     </using-query-result> | ||||||
| </template> | </template> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -16,26 +16,31 @@ | ||||||
|     const route = useRoute(); |     const route = useRoute(); | ||||||
|     const { t } = useI18n(); |     const { t } = useI18n(); | ||||||
| 
 | 
 | ||||||
|     const props = defineProps<{ hruid: string; language: Language; learningObjectHruid?: string }>(); |     const props = defineProps<{ | ||||||
|  |         hruid: string; | ||||||
|  |         language: Language; | ||||||
|  |         learningObjectHruid?: string, | ||||||
|  |     }>(); | ||||||
| 
 | 
 | ||||||
|     interface Personalization { |     interface LearningPathPageQuery { | ||||||
|         forStudent?: string; |  | ||||||
|         forGroup?: string; |         forGroup?: string; | ||||||
|  |         assignmentNo?: string; | ||||||
|  |         classId?: string; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     const personalization = computed(() => { |     const query = computed(() => route.query as LearningPathPageQuery); | ||||||
|         if (route.query.forStudent || route.query.forGroup) { | 
 | ||||||
|  |     const forGroup = computed(() => { | ||||||
|  |         if (query.value.forGroup && query.value.assignmentNo && query.value.classId) { | ||||||
|             return { |             return { | ||||||
|                 forStudent: route.query.forStudent, |                 forGroup: parseInt(query.value.forGroup), | ||||||
|                 forGroup: route.query.forGroup, |                 assignmentNo: parseInt(query.value.assignmentNo), | ||||||
|             } as Personalization; |                 classId: query.value.classId | ||||||
|  |             }; | ||||||
|         } |         } | ||||||
|         return { |  | ||||||
|             forStudent: authService.authState.user?.profile?.preferred_username, |  | ||||||
|         } as Personalization; |  | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     const learningPathQueryResult = useGetLearningPathQuery(props.hruid, props.language, personalization); |     const learningPathQueryResult = useGetLearningPathQuery(props.hruid, props.language, forGroup); | ||||||
| 
 | 
 | ||||||
|     const learningObjectListQueryResult = useLearningObjectListForPathQuery(learningPathQueryResult.data); |     const learningObjectListQueryResult = useLearningObjectListForPathQuery(learningPathQueryResult.data); | ||||||
| 
 | 
 | ||||||
|  | @ -184,6 +189,7 @@ | ||||||
|             :hruid="currentNode.learningobjectHruid" |             :hruid="currentNode.learningobjectHruid" | ||||||
|             :language="currentNode.language" |             :language="currentNode.language" | ||||||
|             :version="currentNode.version" |             :version="currentNode.version" | ||||||
|  |             :group="forGroup" | ||||||
|             v-if="currentNode" |             v-if="currentNode" | ||||||
|         ></learning-object-view> |         ></learning-object-view> | ||||||
|         <div class="navigation-buttons-container"> |         <div class="navigation-buttons-container"> | ||||||
|  |  | ||||||
							
								
								
									
										1
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								package-lock.json
									
										
									
										generated
									
									
									
								
							|  | @ -269,6 +269,7 @@ | ||||||
|             "name": "dwengo-1-frontend", |             "name": "dwengo-1-frontend", | ||||||
|             "version": "0.1.1", |             "version": "0.1.1", | ||||||
|             "dependencies": { |             "dependencies": { | ||||||
|  |                 "@dwengo-1/common": "^0.1.1", | ||||||
|                 "@tanstack/react-query": "^5.69.0", |                 "@tanstack/react-query": "^5.69.0", | ||||||
|                 "@tanstack/vue-query": "^5.69.0", |                 "@tanstack/vue-query": "^5.69.0", | ||||||
|                 "axios": "^1.8.2", |                 "axios": "^1.8.2", | ||||||
|  |  | ||||||
		Reference in a new issue
	
	 Gerald Schmittinger
						Gerald Schmittinger