Merge branch 'dev' into feat/endpoints-beschermen-met-authenticatie-#105
This commit is contained in:
commit
6edb5f144d
46 changed files with 10216 additions and 1054 deletions
|
@ -13,8 +13,10 @@ DWENGO_DB_NAME=":memory:"
|
||||||
DWENGO_DB_UPDATE=true
|
DWENGO_DB_UPDATE=true
|
||||||
|
|
||||||
DWENGO_AUTH_STUDENT_URL=http://localhost:7080/realms/student
|
DWENGO_AUTH_STUDENT_URL=http://localhost:7080/realms/student
|
||||||
|
DWENGO_AUTH_STUDENT_CLIENT_ID=dwengo
|
||||||
DWENGO_AUTH_STUDENT_JWKS_ENDPOINT=http://localhost:7080/realms/student/protocol/openid-connect/certs
|
DWENGO_AUTH_STUDENT_JWKS_ENDPOINT=http://localhost:7080/realms/student/protocol/openid-connect/certs
|
||||||
DWENGO_AUTH_TEACHER_URL=http://localhost:7080/realms/teacher
|
DWENGO_AUTH_TEACHER_URL=http://localhost:7080/realms/teacher
|
||||||
|
DWENGO_AUTH_TEACHER_CLIENT_ID=dwengo
|
||||||
DWENGO_AUTH_TEACHER_JWKS_ENDPOINT=http://localhost:7080/realms/teacher/protocol/openid-connect/certs
|
DWENGO_AUTH_TEACHER_JWKS_ENDPOINT=http://localhost:7080/realms/teacher/protocol/openid-connect/certs
|
||||||
|
|
||||||
DWENGO_CORS_ALLOWED_ORIGINS=http://localhost:5173,http://localhost:3000
|
DWENGO_CORS_ALLOWED_ORIGINS=http://localhost:5173,http://localhost:3000,http://localhost:9876,*
|
||||||
|
|
|
@ -27,7 +27,6 @@
|
||||||
"cross": "^1.0.0",
|
"cross": "^1.0.0",
|
||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
"dotenv": "^16.4.7",
|
"dotenv": "^16.4.7",
|
||||||
"dwengo-1-common": "^0.1.1",
|
|
||||||
"express": "^5.0.1",
|
"express": "^5.0.1",
|
||||||
"express-jwt": "^8.5.1",
|
"express-jwt": "^8.5.1",
|
||||||
"gift-pegjs": "^1.0.2",
|
"gift-pegjs": "^1.0.2",
|
||||||
|
|
|
@ -4,6 +4,7 @@ import {
|
||||||
deleteAssignment,
|
deleteAssignment,
|
||||||
getAllAssignments,
|
getAllAssignments,
|
||||||
getAssignment,
|
getAssignment,
|
||||||
|
getAssignmentsQuestions,
|
||||||
getAssignmentsSubmissions,
|
getAssignmentsSubmissions,
|
||||||
putAssignment,
|
putAssignment,
|
||||||
} from '../services/assignments.js';
|
} from '../services/assignments.js';
|
||||||
|
@ -13,6 +14,19 @@ import { BadRequestException } from '../exceptions/bad-request-exception.js';
|
||||||
import { Assignment } from '../entities/assignments/assignment.entity.js';
|
import { Assignment } from '../entities/assignments/assignment.entity.js';
|
||||||
import { EntityDTO } from '@mikro-orm/core';
|
import { EntityDTO } from '@mikro-orm/core';
|
||||||
|
|
||||||
|
function getAssignmentParams(req: Request): { classid: string; assignmentNumber: number; full: boolean } {
|
||||||
|
const classid = req.params.classid;
|
||||||
|
const assignmentNumber = Number(req.params.id);
|
||||||
|
const full = req.query.full === 'true';
|
||||||
|
requireFields({ assignmentNumber, classid });
|
||||||
|
|
||||||
|
if (isNaN(assignmentNumber)) {
|
||||||
|
throw new BadRequestException('Assignment id should be a number');
|
||||||
|
}
|
||||||
|
|
||||||
|
return { classid, assignmentNumber, full };
|
||||||
|
}
|
||||||
|
|
||||||
export async function getAllAssignmentsHandler(req: Request, res: Response): Promise<void> {
|
export async function getAllAssignmentsHandler(req: Request, res: Response): Promise<void> {
|
||||||
const classId = req.params.classid;
|
const classId = req.params.classid;
|
||||||
const full = req.query.full === 'true';
|
const full = req.query.full === 'true';
|
||||||
|
@ -38,57 +52,42 @@ export async function createAssignmentHandler(req: Request, res: Response): Prom
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getAssignmentHandler(req: Request, res: Response): Promise<void> {
|
export async function getAssignmentHandler(req: Request, res: Response): Promise<void> {
|
||||||
const id = Number(req.params.id);
|
const { classid, assignmentNumber } = getAssignmentParams(req);
|
||||||
const classid = req.params.classid;
|
|
||||||
requireFields({ id, classid });
|
|
||||||
|
|
||||||
if (isNaN(id)) {
|
const assignment = await getAssignment(classid, assignmentNumber);
|
||||||
throw new BadRequestException('Assignment id should be a number');
|
|
||||||
}
|
|
||||||
|
|
||||||
const assignment = await getAssignment(classid, id);
|
|
||||||
|
|
||||||
res.json({ assignment });
|
res.json({ assignment });
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function putAssignmentHandler(req: Request, res: Response): Promise<void> {
|
export async function putAssignmentHandler(req: Request, res: Response): Promise<void> {
|
||||||
const id = Number(req.params.id);
|
const { classid, assignmentNumber } = getAssignmentParams(req);
|
||||||
const classid = req.params.classid;
|
|
||||||
requireFields({ id, classid });
|
|
||||||
|
|
||||||
if (isNaN(id)) {
|
|
||||||
throw new BadRequestException('Assignment id should be a number');
|
|
||||||
}
|
|
||||||
|
|
||||||
const assignmentData = req.body as Partial<EntityDTO<Assignment>>;
|
const assignmentData = req.body as Partial<EntityDTO<Assignment>>;
|
||||||
const assignment = await putAssignment(classid, id, assignmentData);
|
const assignment = await putAssignment(classid, assignmentNumber, assignmentData);
|
||||||
|
|
||||||
res.json({ assignment });
|
res.json({ assignment });
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteAssignmentHandler(req: Request, _res: Response): Promise<void> {
|
export async function deleteAssignmentHandler(req: Request, res: Response): Promise<void> {
|
||||||
const id = Number(req.params.id);
|
const { classid, assignmentNumber } = getAssignmentParams(req);
|
||||||
const classid = req.params.classid;
|
|
||||||
requireFields({ id, classid });
|
|
||||||
|
|
||||||
if (isNaN(id)) {
|
const assignment = await deleteAssignment(classid, assignmentNumber);
|
||||||
throw new BadRequestException('Assignment id should be a number');
|
|
||||||
}
|
|
||||||
|
|
||||||
await deleteAssignment(classid, id);
|
res.json({ assignment });
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getAssignmentsSubmissionsHandler(req: Request, res: Response): Promise<void> {
|
export async function getAssignmentsSubmissionsHandler(req: Request, res: Response): Promise<void> {
|
||||||
const classid = req.params.classid;
|
const { classid, assignmentNumber, full } = getAssignmentParams(req);
|
||||||
const assignmentNumber = Number(req.params.id);
|
|
||||||
const full = req.query.full === 'true';
|
|
||||||
requireFields({ assignmentNumber, classid });
|
|
||||||
|
|
||||||
if (isNaN(assignmentNumber)) {
|
|
||||||
throw new BadRequestException('Assignment id should be a number');
|
|
||||||
}
|
|
||||||
|
|
||||||
const submissions = await getAssignmentsSubmissions(classid, assignmentNumber, full);
|
const submissions = await getAssignmentsSubmissions(classid, assignmentNumber, full);
|
||||||
|
|
||||||
res.json({ submissions });
|
res.json({ submissions });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getAssignmentQuestionsHandler(req: Request, res: Response): Promise<void> {
|
||||||
|
const { classid, assignmentNumber, full } = getAssignmentParams(req);
|
||||||
|
|
||||||
|
const questions = await getAssignmentsQuestions(classid, assignmentNumber, full);
|
||||||
|
|
||||||
|
res.json({ questions });
|
||||||
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { Request, Response } from 'express';
|
import { Request, Response } from 'express';
|
||||||
import { createGroup, deleteGroup, getAllGroups, getGroup, getGroupSubmissions, putGroup } from '../services/groups.js';
|
import { createGroup, deleteGroup, getAllGroups, getGroup, getGroupQuestions, getGroupSubmissions, putGroup } from '../services/groups.js';
|
||||||
import { GroupDTO } from '@dwengo-1/common/interfaces/group';
|
import { GroupDTO } from '@dwengo-1/common/interfaces/group';
|
||||||
import { requireFields } from './error-helper.js';
|
import { requireFields } from './error-helper.js';
|
||||||
import { BadRequestException } from '../exceptions/bad-request-exception.js';
|
import { BadRequestException } from '../exceptions/bad-request-exception.js';
|
||||||
|
@ -84,7 +84,7 @@ export async function createGroupHandler(req: Request, res: Response): Promise<v
|
||||||
res.status(201).json({ group });
|
res.status(201).json({ group });
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getGroupSubmissionsHandler(req: Request, res: Response): Promise<void> {
|
function getGroupParams(req: Request): { classId: string; assignmentId: number; groupId: number; full: boolean } {
|
||||||
const classId = req.params.classid;
|
const classId = req.params.classid;
|
||||||
const assignmentId = Number(req.params.assignmentid);
|
const assignmentId = Number(req.params.assignmentid);
|
||||||
const groupId = Number(req.params.groupid);
|
const groupId = Number(req.params.groupid);
|
||||||
|
@ -100,7 +100,21 @@ export async function getGroupSubmissionsHandler(req: Request, res: Response): P
|
||||||
throw new BadRequestException('Group id must be a number');
|
throw new BadRequestException('Group id must be a number');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return { classId, assignmentId, groupId, full };
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getGroupSubmissionsHandler(req: Request, res: Response): Promise<void> {
|
||||||
|
const { classId, assignmentId, groupId, full } = getGroupParams(req);
|
||||||
|
|
||||||
const submissions = await getGroupSubmissions(classId, assignmentId, groupId, full);
|
const submissions = await getGroupSubmissions(classId, assignmentId, groupId, full);
|
||||||
|
|
||||||
res.json({ submissions });
|
res.json({ submissions });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getGroupQuestionsHandler(req: Request, res: Response): Promise<void> {
|
||||||
|
const { classId, assignmentId, groupId, full } = getGroupParams(req);
|
||||||
|
|
||||||
|
const questions = await getGroupQuestions(classId, assignmentId, groupId, full);
|
||||||
|
|
||||||
|
res.json({ questions });
|
||||||
|
}
|
||||||
|
|
|
@ -62,9 +62,7 @@ export class QuestionRepository extends DwengoEntityRepository<Question> {
|
||||||
|
|
||||||
public async findAllByAssignment(assignment: Assignment): Promise<Question[]> {
|
public async findAllByAssignment(assignment: Assignment): Promise<Question[]> {
|
||||||
return this.find({
|
return this.find({
|
||||||
inGroup: {
|
inGroup: assignment.groups.getItems(),
|
||||||
$contained: assignment.groups,
|
|
||||||
},
|
|
||||||
learningObjectHruid: assignment.learningPathHruid,
|
learningObjectHruid: assignment.learningPathHruid,
|
||||||
learningObjectLanguage: assignment.learningPathLanguage,
|
learningObjectLanguage: assignment.learningPathLanguage,
|
||||||
});
|
});
|
||||||
|
@ -77,6 +75,13 @@ export class QuestionRepository extends DwengoEntityRepository<Question> {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async findAllByGroup(inGroup: Group): Promise<Question[]> {
|
||||||
|
return this.findAll({
|
||||||
|
where: { inGroup },
|
||||||
|
orderBy: { timestamp: 'DESC' },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Looks up all questions for the given learning object which were asked as part of the given assignment.
|
* Looks up all questions for the given learning object which were asked as part of the given assignment.
|
||||||
* When forStudentUsername is set, only the questions within the given user's group are shown.
|
* When forStudentUsername is set, only the questions within the given user's group are shown.
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import { Collection, Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core';
|
import { Cascade, Collection, Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core';
|
||||||
import { Class } from '../classes/class.entity.js';
|
import { Class } from '../classes/class.entity.js';
|
||||||
import { Group } from './group.entity.js';
|
import { Group } from './group.entity.js';
|
||||||
import { Language } from '@dwengo-1/common/util/language';
|
import { Language } from '@dwengo-1/common/util/language';
|
||||||
|
@ -34,6 +34,7 @@ export class Assignment {
|
||||||
@OneToMany({
|
@OneToMany({
|
||||||
entity: () => Group,
|
entity: () => Group,
|
||||||
mappedBy: 'assignment',
|
mappedBy: 'assignment',
|
||||||
|
cascade: [Cascade.ALL],
|
||||||
})
|
})
|
||||||
groups: Collection<Group> = new Collection<Group>(this);
|
groups: Collection<Group> = new Collection<Group>(this);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import { Student } from '../users/student.entity.js';
|
import { Student } from '../users/student.entity.js';
|
||||||
import { Group } from './group.entity.js';
|
import { Group } from './group.entity.js';
|
||||||
import { Entity, Enum, ManyToOne, PrimaryKey, Property } from '@mikro-orm/core';
|
import { Entity, Enum, ManyToOne, PrimaryKey, Property, Cascade } from '@mikro-orm/core';
|
||||||
import { SubmissionRepository } from '../../data/assignments/submission-repository.js';
|
import { SubmissionRepository } from '../../data/assignments/submission-repository.js';
|
||||||
import { Language } from '@dwengo-1/common/util/language';
|
import { Language } from '@dwengo-1/common/util/language';
|
||||||
|
|
||||||
|
@ -21,8 +21,8 @@ export class Submission {
|
||||||
@PrimaryKey({ type: 'numeric', autoincrement: false })
|
@PrimaryKey({ type: 'numeric', autoincrement: false })
|
||||||
learningObjectVersion = 1;
|
learningObjectVersion = 1;
|
||||||
|
|
||||||
@ManyToOne({
|
@ManyToOne(() => Group, {
|
||||||
entity: () => Group,
|
cascade: [Cascade.REMOVE],
|
||||||
})
|
})
|
||||||
onBehalfOf!: Group;
|
onBehalfOf!: Group;
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@ import {
|
||||||
deleteAssignmentHandler,
|
deleteAssignmentHandler,
|
||||||
getAllAssignmentsHandler,
|
getAllAssignmentsHandler,
|
||||||
getAssignmentHandler,
|
getAssignmentHandler,
|
||||||
|
getAssignmentQuestionsHandler,
|
||||||
getAssignmentsSubmissionsHandler,
|
getAssignmentsSubmissionsHandler,
|
||||||
putAssignmentHandler,
|
putAssignmentHandler,
|
||||||
} from '../controllers/assignments.js';
|
} from '../controllers/assignments.js';
|
||||||
|
@ -26,11 +27,7 @@ router.delete('/:id', teachersOnly, onlyAllowIfHasAccessToAssignment, deleteAssi
|
||||||
|
|
||||||
router.get('/:id/submissions', teachersOnly, onlyAllowIfHasAccessToAssignment, getAssignmentsSubmissionsHandler);
|
router.get('/:id/submissions', teachersOnly, onlyAllowIfHasAccessToAssignment, getAssignmentsSubmissionsHandler);
|
||||||
|
|
||||||
router.get('/:id/questions', teachersOnly, onlyAllowIfHasAccessToAssignment, (_req, res) => {
|
router.get('/:id/questions', teachersOnly, onlyAllowIfHasAccessToAssignment, getAssignmentQuestionsHandler);
|
||||||
res.json({
|
|
||||||
questions: ['0'],
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
router.use('/:assignmentid/groups', groupRouter);
|
router.use('/:assignmentid/groups', groupRouter);
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@ import {
|
||||||
deleteGroupHandler,
|
deleteGroupHandler,
|
||||||
getAllGroupsHandler,
|
getAllGroupsHandler,
|
||||||
getGroupHandler,
|
getGroupHandler,
|
||||||
|
getGroupQuestionsHandler,
|
||||||
getGroupSubmissionsHandler,
|
getGroupSubmissionsHandler,
|
||||||
putGroupHandler,
|
putGroupHandler,
|
||||||
} from '../controllers/groups.js';
|
} from '../controllers/groups.js';
|
||||||
|
@ -32,4 +33,6 @@ router.get('/:groupid/questions', onlyAllowIfHasAccessToGroup, (_req, res) => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
router.get('/:groupid/questions', getGroupQuestionsHandler);
|
||||||
|
|
||||||
export default router;
|
export default router;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { EntityDTO } from '@mikro-orm/core';
|
import { EntityDTO } from '@mikro-orm/core';
|
||||||
import { getGroupRepository, getSubmissionRepository } from '../data/repositories.js';
|
import { getGroupRepository, getQuestionRepository, getSubmissionRepository } from '../data/repositories.js';
|
||||||
import { Group } from '../entities/assignments/group.entity.js';
|
import { Group } from '../entities/assignments/group.entity.js';
|
||||||
import { mapToGroupDTO, mapToGroupDTOId } from '../interfaces/group.js';
|
import { mapToGroupDTO, mapToGroupDTOId } from '../interfaces/group.js';
|
||||||
import { mapToSubmissionDTO, mapToSubmissionDTOId } from '../interfaces/submission.js';
|
import { mapToSubmissionDTO, mapToSubmissionDTOId } from '../interfaces/submission.js';
|
||||||
|
@ -12,6 +12,8 @@ import { fetchClass } from './classes.js';
|
||||||
import { BadRequestException } from '../exceptions/bad-request-exception.js';
|
import { BadRequestException } from '../exceptions/bad-request-exception.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 { QuestionDTO, QuestionId } from '@dwengo-1/common/interfaces/question';
|
||||||
|
import { mapToQuestionDTO, mapToQuestionDTOId } from '../interfaces/question.js';
|
||||||
|
|
||||||
async function assertMembersInClass(members: Student[], cls: Class): Promise<void> {
|
async function assertMembersInClass(members: Student[], cls: Class): Promise<void> {
|
||||||
if (!members.every((student) => cls.students.contains(student))) {
|
if (!members.every((student) => cls.students.contains(student))) {
|
||||||
|
@ -130,3 +132,21 @@ export async function getGroupSubmissions(
|
||||||
|
|
||||||
return submissions.map(mapToSubmissionDTOId);
|
return submissions.map(mapToSubmissionDTOId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function getGroupQuestions(
|
||||||
|
classId: string,
|
||||||
|
assignmentNumber: number,
|
||||||
|
groupNumber: number,
|
||||||
|
full: boolean
|
||||||
|
): Promise<QuestionDTO[] | QuestionId[]> {
|
||||||
|
const group = await fetchGroup(classId, assignmentNumber, groupNumber);
|
||||||
|
|
||||||
|
const questionRepository = getQuestionRepository();
|
||||||
|
const questions = await questionRepository.findAllByGroup(group);
|
||||||
|
|
||||||
|
if (full) {
|
||||||
|
return questions.map(mapToQuestionDTO);
|
||||||
|
}
|
||||||
|
|
||||||
|
return questions.map(mapToQuestionDTOId);
|
||||||
|
}
|
||||||
|
|
|
@ -17,6 +17,7 @@ import { ClassRepository } from '../../../src/data/classes/class-repository';
|
||||||
import { Submission } from '../../../src/entities/assignments/submission.entity';
|
import { Submission } from '../../../src/entities/assignments/submission.entity';
|
||||||
import { Class } from '../../../src/entities/classes/class.entity';
|
import { Class } from '../../../src/entities/classes/class.entity';
|
||||||
import { Assignment } from '../../../src/entities/assignments/assignment.entity';
|
import { Assignment } from '../../../src/entities/assignments/assignment.entity';
|
||||||
|
import { testLearningObject01 } from '../../test_assets/content/learning-objects.testdata';
|
||||||
|
|
||||||
describe('SubmissionRepository', () => {
|
describe('SubmissionRepository', () => {
|
||||||
let submissionRepository: SubmissionRepository;
|
let submissionRepository: SubmissionRepository;
|
||||||
|
@ -106,7 +107,7 @@ describe('SubmissionRepository', () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should not find a deleted submission', async () => {
|
it('should not find a deleted submission', async () => {
|
||||||
const id = new LearningObjectIdentifier('id01', Language.English, 1);
|
const id = new LearningObjectIdentifier(testLearningObject01.hruid, testLearningObject01.language, testLearningObject01.version);
|
||||||
await submissionRepository.deleteSubmissionByLearningObjectAndSubmissionNumber(id, 1);
|
await submissionRepository.deleteSubmissionByLearningObjectAndSubmissionNumber(id, 1);
|
||||||
|
|
||||||
const submission = await submissionRepository.findSubmissionByLearningObjectAndSubmissionNumber(id, 1);
|
const submission = await submissionRepository.findSubmissionByLearningObjectAndSubmissionNumber(id, 1);
|
||||||
|
|
|
@ -27,7 +27,7 @@
|
||||||
"oauth2DevicePollingInterval": 5,
|
"oauth2DevicePollingInterval": 5,
|
||||||
"enabled": true,
|
"enabled": true,
|
||||||
"sslRequired": "external",
|
"sslRequired": "external",
|
||||||
"registrationAllowed": false,
|
"registrationAllowed": true,
|
||||||
"registrationEmailAsUsername": false,
|
"registrationEmailAsUsername": false,
|
||||||
"rememberMe": false,
|
"rememberMe": false,
|
||||||
"verifyEmail": false,
|
"verifyEmail": false,
|
||||||
|
|
|
@ -27,7 +27,7 @@
|
||||||
"oauth2DevicePollingInterval": 5,
|
"oauth2DevicePollingInterval": 5,
|
||||||
"enabled": true,
|
"enabled": true,
|
||||||
"sslRequired": "external",
|
"sslRequired": "external",
|
||||||
"registrationAllowed": false,
|
"registrationAllowed": true,
|
||||||
"registrationEmailAsUsername": false,
|
"registrationEmailAsUsername": false,
|
||||||
"rememberMe": false,
|
"rememberMe": false,
|
||||||
"verifyEmail": false,
|
"verifyEmail": false,
|
||||||
|
|
|
@ -52,11 +52,18 @@ npm run test:unit
|
||||||
### Run End-to-End Tests with [Playwright](https://playwright.dev)
|
### Run End-to-End Tests with [Playwright](https://playwright.dev)
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
|
cd frontend
|
||||||
|
|
||||||
# Install browsers for the first run
|
# Install browsers for the first run
|
||||||
npx playwright install
|
npx playwright install
|
||||||
|
# On Ubuntu, you can also use
|
||||||
|
npx playwright install --with-deps
|
||||||
|
# to additionally install the dependencies.
|
||||||
|
|
||||||
# When testing on CI, must build the project first
|
# When testing on CI, must build the project first
|
||||||
|
cd ..
|
||||||
npm run build
|
npm run build
|
||||||
|
cd frontend
|
||||||
|
|
||||||
# Runs the end-to-end tests
|
# Runs the end-to-end tests
|
||||||
npm run test:e2e
|
npm run test:e2e
|
||||||
|
|
86
frontend/e2e/basic-homepage.spec.ts
Normal file
86
frontend/e2e/basic-homepage.spec.ts
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
import { test, expect } from "@playwright/test";
|
||||||
|
|
||||||
|
test("User can pick their language", async ({ page }) => {
|
||||||
|
await page.goto("/");
|
||||||
|
|
||||||
|
await expect(page.getByRole("button", { name: "translate" })).toBeVisible();
|
||||||
|
await page.getByRole("button", { name: "translate" }).click();
|
||||||
|
await page.getByText("Nederlands").click();
|
||||||
|
await expect(page.locator("h1")).toContainText("Onze sterke punten");
|
||||||
|
await expect(page.getByRole("heading", { name: "Innovatief" })).toBeVisible();
|
||||||
|
|
||||||
|
await page.getByRole("heading", { name: "Innovatief" }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole("button", { name: "vertalen" })).toBeVisible();
|
||||||
|
await page.getByRole("button", { name: "vertalen" }).click();
|
||||||
|
await page.getByText("English").click();
|
||||||
|
await expect(page.locator("h1")).toContainText("Our strengths");
|
||||||
|
await expect(page.getByRole("heading", { name: "Innovative" })).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Teacher can sign in", async ({ page }) => {
|
||||||
|
await page.goto("/");
|
||||||
|
await expect(page.getByRole("link", { name: "log in" })).toBeVisible();
|
||||||
|
await page.getByRole("link", { name: "log in" }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole("button", { name: "teacher" })).toBeVisible();
|
||||||
|
await page.getByRole("button", { name: "teacher" }).click();
|
||||||
|
|
||||||
|
await expect(page.getByText("teacher")).toBeVisible();
|
||||||
|
await expect(page.getByRole("button", { name: "Sign In" })).toBeVisible();
|
||||||
|
|
||||||
|
await expect(page).toHaveURL(/\/realms\/teacher\//);
|
||||||
|
|
||||||
|
await page.getByRole("textbox", { name: "Username or email" }).fill("testleerkracht1");
|
||||||
|
await page.getByRole("textbox", { name: "Password" }).fill("password");
|
||||||
|
await page.getByRole("button", { name: "Sign In" }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole("link", { name: "Dwengo logo teacher" })).toBeVisible();
|
||||||
|
await expect(page.getByRole("button").nth(1)).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Student can sign in", async ({ page }) => {
|
||||||
|
await page.goto("/");
|
||||||
|
await expect(page.getByRole("link", { name: "log in" })).toBeVisible();
|
||||||
|
await page.getByRole("link", { name: "log in" }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole("button", { name: "student" })).toBeVisible();
|
||||||
|
await page.getByRole("button", { name: "student" }).click();
|
||||||
|
|
||||||
|
await expect(page).toHaveURL(/\/realms\/student\//);
|
||||||
|
|
||||||
|
await expect(page.getByText("student")).toBeVisible();
|
||||||
|
await expect(page.getByRole("button", { name: "Sign In" })).toBeVisible();
|
||||||
|
|
||||||
|
await page.getByRole("textbox", { name: "Username or email" }).fill("testleerling1");
|
||||||
|
await page.getByRole("textbox", { name: "Password" }).fill("password");
|
||||||
|
await page.getByRole("button", { name: "Sign In" }).click();
|
||||||
|
|
||||||
|
await expect(page.getByRole("link", { name: "Dwengo logo student" })).toBeVisible();
|
||||||
|
await expect(page.getByRole("button").nth(1)).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Cannot sign in with invalid credentials", async ({ page }) => {
|
||||||
|
await page.goto("/");
|
||||||
|
await page.getByRole("link", { name: "log in" }).click();
|
||||||
|
await page.getByRole("button", { name: "teacher" }).click();
|
||||||
|
await page.getByRole("textbox", { name: "Username or email" }).fill("doesnotexist");
|
||||||
|
await page.getByRole("textbox", { name: "Password" }).fill("wrong");
|
||||||
|
await page.getByRole("button", { name: "Sign In" }).click();
|
||||||
|
await expect(page.getByText("Invalid username or password.")).toBeVisible();
|
||||||
|
|
||||||
|
await page.goto("/");
|
||||||
|
await page.getByRole("link", { name: "log in" }).click();
|
||||||
|
await page.getByRole("button", { name: "student" }).click();
|
||||||
|
await page.getByRole("textbox", { name: "Username or email" }).fill("doesnotexist");
|
||||||
|
await page.getByRole("textbox", { name: "Password" }).fill("wrong");
|
||||||
|
await page.getByRole("button", { name: "Sign In" }).click();
|
||||||
|
await expect(page.getByText("Invalid username or password.")).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Cannot skip login", async ({ page }) => {
|
||||||
|
await page.goto("/user");
|
||||||
|
// Should redirect to login
|
||||||
|
await expect(page.getByText("login")).toBeVisible();
|
||||||
|
await expect(page.getByRole("button", { name: "teacher" })).toBeVisible();
|
||||||
|
});
|
12
frontend/e2e/basic-learning.spec.ts
Normal file
12
frontend/e2e/basic-learning.spec.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
import { test, expect } from "./fixtures.js";
|
||||||
|
|
||||||
|
test("Users can filter", async ({ page }) => {
|
||||||
|
await page.goto("/user");
|
||||||
|
|
||||||
|
await page.getByRole("combobox").filter({ hasText: "Select a themeAll" }).locator("i").click();
|
||||||
|
await page.getByText("Nature and climate").click();
|
||||||
|
await page.getByRole("combobox").filter({ hasText: "Select ageAll agesSelect age" }).locator("i").click();
|
||||||
|
await page.getByText("and older").click();
|
||||||
|
|
||||||
|
await expect(page.getByRole("link", { name: "AI and Climate Students in" })).toBeVisible();
|
||||||
|
});
|
5
frontend/e2e/basic-learning.ts
Normal file
5
frontend/e2e/basic-learning.ts
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
import { test, expect } from "./fixtures.js";
|
||||||
|
|
||||||
|
test("myTest", async ({ page }) => {
|
||||||
|
await expect(page).toHaveURL("/");
|
||||||
|
});
|
116
frontend/e2e/fixtures.ts
Normal file
116
frontend/e2e/fixtures.ts
Normal file
|
@ -0,0 +1,116 @@
|
||||||
|
/* eslint-disable no-await-in-loop */
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
import { test as baseTest, expect } from "@playwright/test";
|
||||||
|
import type { Browser } from "playwright-core";
|
||||||
|
import fs from "fs";
|
||||||
|
import path from "path";
|
||||||
|
|
||||||
|
/* Based on https://playwright.dev/docs/auth#moderate-one-account-per-parallel-worker */
|
||||||
|
|
||||||
|
export * from "@playwright/test";
|
||||||
|
export const ROOT_URL = "http://localhost:5173";
|
||||||
|
|
||||||
|
interface Account {
|
||||||
|
username: string;
|
||||||
|
password: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Acquire an account by logging in or creating a new one.
|
||||||
|
* @param id
|
||||||
|
* @param browser
|
||||||
|
*/
|
||||||
|
async function acquireAccount(id: number, browser: Browser): Promise<Account> {
|
||||||
|
const account = {
|
||||||
|
username: `worker${id}`,
|
||||||
|
password: "password",
|
||||||
|
};
|
||||||
|
|
||||||
|
const page = await browser.newPage();
|
||||||
|
await page.goto(ROOT_URL);
|
||||||
|
|
||||||
|
await page.getByRole("link", { name: "log in" }).click();
|
||||||
|
await page.getByRole("button", { name: "student" }).click();
|
||||||
|
|
||||||
|
await page.getByRole("textbox", { name: "Username" }).fill(account.username);
|
||||||
|
await page.getByRole("textbox", { name: "Password", exact: true }).fill(account.password);
|
||||||
|
await page.getByRole("button", { name: "Sign In" }).click();
|
||||||
|
|
||||||
|
let failed = await page.getByText("Invalid username or password.").isVisible();
|
||||||
|
|
||||||
|
if (failed) {
|
||||||
|
await page.getByRole("link", { name: "Register" }).click();
|
||||||
|
}
|
||||||
|
|
||||||
|
const MAX_RETRIES = 5;
|
||||||
|
let retries = 0;
|
||||||
|
while (failed && retries < MAX_RETRIES) {
|
||||||
|
// Retry with a different username, based on Unix timestamp.
|
||||||
|
account.username = `worker${id}-${Date.now()}`;
|
||||||
|
|
||||||
|
await page.getByRole("textbox", { name: "Username" }).fill(account.username);
|
||||||
|
await page.getByRole("textbox", { name: "Password", exact: true }).fill(account.password);
|
||||||
|
await page.getByRole("textbox", { name: "Confirm password" }).fill(account.password);
|
||||||
|
await page.getByRole("textbox", { name: "Email" }).fill(`${account.username}@dwengo.org`);
|
||||||
|
await page.getByRole("textbox", { name: "First name" }).fill("Worker");
|
||||||
|
await page.getByRole("textbox", { name: "Last name" }).fill(id.toString());
|
||||||
|
await page.getByRole("button", { name: "Register" }).click();
|
||||||
|
|
||||||
|
await page.waitForURL(/localhost/);
|
||||||
|
|
||||||
|
failed = await page.getByText("Username already exists.").isVisible();
|
||||||
|
retries += failed ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
await page.waitForURL(/localhost/);
|
||||||
|
await page.close();
|
||||||
|
|
||||||
|
return account;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const test = baseTest.extend<object, { workerStorageState: string }>({
|
||||||
|
// Use the same storage state for all tests in this worker.
|
||||||
|
storageState: async ({ workerStorageState }, use) => use(workerStorageState),
|
||||||
|
|
||||||
|
// Authenticate once per worker with a worker-scoped fixture.
|
||||||
|
workerStorageState: [
|
||||||
|
async ({ browser }, use): Promise<void> => {
|
||||||
|
// Use parallelIndex as a unique identifier for each worker.
|
||||||
|
const id = test.info().parallelIndex;
|
||||||
|
const fileName = path.resolve(test.info().project.outputDir, `.auth/${id}.json`);
|
||||||
|
|
||||||
|
if (fs.existsSync(fileName)) {
|
||||||
|
// Reuse existing authentication state if any.
|
||||||
|
await use(fileName);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Important: make sure we authenticate in a clean environment by unsetting storage state.
|
||||||
|
const page = await browser.newPage({ storageState: undefined });
|
||||||
|
|
||||||
|
// Acquire a unique account by creating a new one.
|
||||||
|
const account = await acquireAccount(id, browser);
|
||||||
|
|
||||||
|
// Perform authentication steps. Replace these actions with your own.
|
||||||
|
await page.goto(ROOT_URL);
|
||||||
|
await page.getByRole("link", { name: "log in" }).click();
|
||||||
|
await page.getByRole("button", { name: "student" }).click();
|
||||||
|
await page.getByRole("textbox", { name: "Username or email" }).fill(account.username);
|
||||||
|
await page.getByRole("textbox", { name: "Password" }).fill(account.password);
|
||||||
|
await page.getByRole("button", { name: "Sign In" }).click();
|
||||||
|
// Wait until the page receives the cookies.
|
||||||
|
//
|
||||||
|
// Sometimes login flow sets cookies in the process of several redirects.
|
||||||
|
// Wait for the final URL to ensure that the cookies are actually set.
|
||||||
|
await page.waitForLoadState("domcontentloaded");
|
||||||
|
// Alternatively, you can wait until the page reaches a state where all cookies are set.
|
||||||
|
|
||||||
|
// End of authentication steps.
|
||||||
|
|
||||||
|
await page.context().storageState({ path: fileName });
|
||||||
|
await page.close();
|
||||||
|
await use(fileName);
|
||||||
|
},
|
||||||
|
{ scope: "worker" },
|
||||||
|
],
|
||||||
|
});
|
|
@ -1,8 +0,0 @@
|
||||||
import { test, expect } from "@playwright/test";
|
|
||||||
|
|
||||||
// See here how to get started:
|
|
||||||
// https://playwright.dev/docs/intro
|
|
||||||
test("visits the app root url", async ({ page }) => {
|
|
||||||
await page.goto("/");
|
|
||||||
await expect(page.locator("h1")).toHaveText("You did it!");
|
|
||||||
});
|
|
|
@ -18,16 +18,18 @@
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@tanstack/react-query": "^5.69.0",
|
"@tanstack/react-query": "^5.69.0",
|
||||||
"@tanstack/vue-query": "^5.69.0",
|
"@tanstack/vue-query": "^5.69.0",
|
||||||
|
"@vueuse/core": "^13.1.0",
|
||||||
"axios": "^1.8.2",
|
"axios": "^1.8.2",
|
||||||
"oidc-client-ts": "^3.1.0",
|
"oidc-client-ts": "^3.1.0",
|
||||||
"uuid": "^11.1.0",
|
"uuid": "^11.1.0",
|
||||||
"vue": "^3.5.13",
|
"vue": "^3.5.13",
|
||||||
"vue-i18n": "^11.1.2",
|
"vue-i18n": "^11.1.2",
|
||||||
"vue-router": "^4.5.0",
|
"vue-router": "^4.5.0",
|
||||||
"vuetify": "^3.7.12"
|
"vuetify": "^3.7.12",
|
||||||
|
"wait-on": "^8.0.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@playwright/test": "^1.50.1",
|
"@playwright/test": "1.50.1",
|
||||||
"@tsconfig/node22": "^22.0.0",
|
"@tsconfig/node22": "^22.0.0",
|
||||||
"@types/jsdom": "^21.1.7",
|
"@types/jsdom": "^21.1.7",
|
||||||
"@types/node": "^22.13.4",
|
"@types/node": "^22.13.4",
|
||||||
|
|
|
@ -24,7 +24,7 @@ export default defineConfig({
|
||||||
/* Fail the build on CI if you accidentally left test.only in the source code. */
|
/* Fail the build on CI if you accidentally left test.only in the source code. */
|
||||||
forbidOnly: Boolean(process.env.CI),
|
forbidOnly: Boolean(process.env.CI),
|
||||||
/* Retry on CI only */
|
/* Retry on CI only */
|
||||||
retries: process.env.CI ? 2 : 0,
|
retries: process.env.CI ? 2 : 1,
|
||||||
/* Opt out of parallel tests on CI. */
|
/* Opt out of parallel tests on CI. */
|
||||||
workers: process.env.CI ? 1 : undefined,
|
workers: process.env.CI ? 1 : undefined,
|
||||||
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
|
/* Reporter to use. See https://playwright.dev/docs/test-reporters */
|
||||||
|
@ -65,18 +65,18 @@ export default defineConfig({
|
||||||
},
|
},
|
||||||
|
|
||||||
/* Test against mobile viewports. */
|
/* Test against mobile viewports. */
|
||||||
// {
|
{
|
||||||
// Name: 'Mobile Chrome',
|
name: "Mobile Chrome",
|
||||||
// Use: {
|
use: {
|
||||||
// ...devices['Pixel 5'],
|
...devices["Pixel 5"],
|
||||||
// },
|
},
|
||||||
// },
|
},
|
||||||
// {
|
{
|
||||||
// Name: 'Mobile Safari',
|
name: "Mobile Safari",
|
||||||
// Use: {
|
use: {
|
||||||
// ...devices['iPhone 12'],
|
...devices["iPhone 12"],
|
||||||
// },
|
},
|
||||||
// },
|
},
|
||||||
|
|
||||||
/* Test against branded browsers. */
|
/* Test against branded browsers. */
|
||||||
// {
|
// {
|
||||||
|
@ -97,14 +97,25 @@ export default defineConfig({
|
||||||
// OutputDir: 'test-results/',
|
// OutputDir: 'test-results/',
|
||||||
|
|
||||||
/* Run your local dev server before starting the tests */
|
/* Run your local dev server before starting the tests */
|
||||||
webServer: {
|
webServer: [
|
||||||
/**
|
// Assuming the idp is already running (because it is slow)
|
||||||
* Use the dev server by default for faster feedback loop.
|
{
|
||||||
* Use the preview server on CI for more realistic testing.
|
/* Frontend */
|
||||||
* Playwright will re-use the local server if there is already a dev-server running.
|
command: `VITE_API_BASE_URL='http://localhost:9876/api' ${process.env.CI ? "npm run preview" : "npm run dev"}`,
|
||||||
*/
|
port: process.env.CI ? 4173 : 5173,
|
||||||
command: process.env.CI ? "npm run preview" : "npm run dev",
|
timeout: 120 * 1000,
|
||||||
port: process.env.CI ? 4173 : 5173,
|
reuseExistingServer: !process.env.CI,
|
||||||
reuseExistingServer: !process.env.CI,
|
},
|
||||||
},
|
{
|
||||||
|
/* Backend */
|
||||||
|
command: `
|
||||||
|
cd .. \
|
||||||
|
&& npx tsc --build common/tsconfig.json \
|
||||||
|
&& cd backend \
|
||||||
|
&& npx tsx --env-file=./.env.test ./tool/startTestApp.ts
|
||||||
|
`,
|
||||||
|
port: 9876,
|
||||||
|
reuseExistingServer: !process.env.CI,
|
||||||
|
},
|
||||||
|
],
|
||||||
});
|
});
|
||||||
|
|
49
frontend/src/assets/assignment.css
Normal file
49
frontend/src/assets/assignment.css
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
.container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 2%;
|
||||||
|
min-height: 100vh;
|
||||||
|
}
|
||||||
|
|
||||||
|
.assignment-card {
|
||||||
|
width: 80%;
|
||||||
|
padding: 2%;
|
||||||
|
border-radius: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.description {
|
||||||
|
margin-top: 2%;
|
||||||
|
line-height: 1.6;
|
||||||
|
font-size: 1.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-right-btn {
|
||||||
|
position: absolute;
|
||||||
|
right: 2%;
|
||||||
|
color: red;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-section {
|
||||||
|
margin-top: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-section h3 {
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-section ul {
|
||||||
|
padding-left: 1.2rem;
|
||||||
|
list-style-type: disc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.subtitle-section {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.assignmentTopTitle {
|
||||||
|
white-space: normal;
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
|
@ -11,7 +11,7 @@
|
||||||
selectedAge: { type: String, required: true },
|
selectedAge: { type: String, required: true },
|
||||||
});
|
});
|
||||||
|
|
||||||
const { locale } = useI18n();
|
const { t, locale } = useI18n();
|
||||||
const language = computed(() => locale.value);
|
const language = computed(() => locale.value);
|
||||||
|
|
||||||
const { data: allThemes, isLoading, error } = useThemeQuery(language);
|
const { data: allThemes, isLoading, error } = useThemeQuery(language);
|
||||||
|
@ -74,6 +74,22 @@
|
||||||
class="fill-height"
|
class="fill-height"
|
||||||
/>
|
/>
|
||||||
</v-col>
|
</v-col>
|
||||||
|
<v-col
|
||||||
|
cols="12"
|
||||||
|
sm="6"
|
||||||
|
md="4"
|
||||||
|
lg="4"
|
||||||
|
class="d-flex"
|
||||||
|
>
|
||||||
|
<ThemeCard
|
||||||
|
path="/learningPath/search"
|
||||||
|
:is-absolute-path="true"
|
||||||
|
:title="t('searchAllLearningPathsTitle')"
|
||||||
|
:description="t('searchAllLearningPathsDescription')"
|
||||||
|
icon="mdi-magnify"
|
||||||
|
class="fill-height"
|
||||||
|
/>
|
||||||
|
</v-col>
|
||||||
</v-row>
|
</v-row>
|
||||||
</v-container>
|
</v-container>
|
||||||
</template>
|
</template>
|
||||||
|
|
|
@ -1,21 +1,26 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { computed } from "vue";
|
||||||
import { useI18n } from "vue-i18n";
|
import { useI18n } from "vue-i18n";
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
defineProps<{
|
const props = defineProps<{
|
||||||
path: string;
|
path: string;
|
||||||
|
isAbsolutePath?: boolean;
|
||||||
title: string;
|
title: string;
|
||||||
description: string;
|
description: string;
|
||||||
image: string;
|
image?: string;
|
||||||
|
icon?: string;
|
||||||
}>();
|
}>();
|
||||||
|
|
||||||
|
const routerLink = computed(() => (props.isAbsolutePath ? props.path : `/theme/${props.path}`));
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<v-card
|
<v-card
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
class="theme-card d-flex flex-column"
|
class="theme-card d-flex flex-column"
|
||||||
:to="`theme/${path}`"
|
:to="routerLink"
|
||||||
link
|
link
|
||||||
>
|
>
|
||||||
<v-card-title class="title-container">
|
<v-card-title class="title-container">
|
||||||
|
@ -27,12 +32,18 @@
|
||||||
contain
|
contain
|
||||||
class="title-image"
|
class="title-image"
|
||||||
></v-img>
|
></v-img>
|
||||||
|
<v-icon
|
||||||
|
v-if="icon"
|
||||||
|
class="title-image"
|
||||||
|
>{{ icon }}</v-icon
|
||||||
|
>
|
||||||
|
|
||||||
<span class="title">{{ title }}</span>
|
<span class="title">{{ title }}</span>
|
||||||
</v-card-title>
|
</v-card-title>
|
||||||
<v-card-text class="description flex-grow-1">{{ description }}</v-card-text>
|
<v-card-text class="description flex-grow-1">{{ description }}</v-card-text>
|
||||||
<v-card-actions>
|
<v-card-actions>
|
||||||
<v-btn
|
<v-btn
|
||||||
:to="`theme/${path}`"
|
:to="routerLink"
|
||||||
variant="text"
|
variant="text"
|
||||||
>
|
>
|
||||||
{{ t("read-more") }}
|
{{ t("read-more") }}
|
||||||
|
|
49
frontend/src/components/assignments/DeadlineSelector.vue
Normal file
49
frontend/src/components/assignments/DeadlineSelector.vue
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, computed } from "vue";
|
||||||
|
import { deadlineRules } from "@/utils/assignment-rules.ts";
|
||||||
|
|
||||||
|
const date = ref("");
|
||||||
|
const time = ref("23:59");
|
||||||
|
const emit = defineEmits(["update:deadline"]);
|
||||||
|
|
||||||
|
const formattedDeadline = computed(() => {
|
||||||
|
if (!date.value || !time.value) return "";
|
||||||
|
return `${date.value} ${time.value}`;
|
||||||
|
});
|
||||||
|
|
||||||
|
function updateDeadline(): void {
|
||||||
|
if (date.value && time.value) {
|
||||||
|
emit("update:deadline", formattedDeadline.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<v-card-text>
|
||||||
|
<v-text-field
|
||||||
|
v-model="date"
|
||||||
|
label="Select Deadline Date"
|
||||||
|
type="date"
|
||||||
|
variant="outlined"
|
||||||
|
density="compact"
|
||||||
|
:rules="deadlineRules"
|
||||||
|
required
|
||||||
|
@update:modelValue="updateDeadline"
|
||||||
|
></v-text-field>
|
||||||
|
</v-card-text>
|
||||||
|
|
||||||
|
<v-card-text>
|
||||||
|
<v-text-field
|
||||||
|
v-model="time"
|
||||||
|
label="Select Deadline Time"
|
||||||
|
type="time"
|
||||||
|
variant="outlined"
|
||||||
|
density="compact"
|
||||||
|
@update:modelValue="updateDeadline"
|
||||||
|
></v-text-field>
|
||||||
|
</v-card-text>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped></style>
|
75
frontend/src/components/assignments/GroupSelector.vue
Normal file
75
frontend/src/components/assignments/GroupSelector.vue
Normal file
|
@ -0,0 +1,75 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref } from "vue";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import UsingQueryResult from "@/components/UsingQueryResult.vue";
|
||||||
|
import type { StudentsResponse } from "@/controllers/students.ts";
|
||||||
|
import { useClassStudentsQuery } from "@/queries/classes.ts";
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
classId: string | undefined;
|
||||||
|
groups: string[][];
|
||||||
|
}>();
|
||||||
|
const emit = defineEmits(["groupCreated"]);
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
const selectedStudents = ref([]);
|
||||||
|
|
||||||
|
const studentQueryResult = useClassStudentsQuery(() => props.classId, true);
|
||||||
|
|
||||||
|
function filterStudents(data: StudentsResponse): { title: string; value: string }[] {
|
||||||
|
const students = data.students;
|
||||||
|
const studentsInGroups = props.groups.flat();
|
||||||
|
|
||||||
|
return students
|
||||||
|
?.map((st) => ({
|
||||||
|
title: `${st.firstName} ${st.lastName}`,
|
||||||
|
value: st.username,
|
||||||
|
}))
|
||||||
|
.filter((student) => !studentsInGroups.includes(student.value));
|
||||||
|
}
|
||||||
|
|
||||||
|
function createGroup(): void {
|
||||||
|
if (selectedStudents.value.length) {
|
||||||
|
// Extract only usernames (student.value)
|
||||||
|
const usernames = selectedStudents.value.map((student) => student.value);
|
||||||
|
emit("groupCreated", usernames);
|
||||||
|
selectedStudents.value = []; // Reset selection after creating group
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<using-query-result
|
||||||
|
:query-result="studentQueryResult"
|
||||||
|
v-slot="{ data }: { data: StudentsResponse }"
|
||||||
|
>
|
||||||
|
<h3>{{ t("create-groups") }}</h3>
|
||||||
|
<v-card-text>
|
||||||
|
<v-combobox
|
||||||
|
v-model="selectedStudents"
|
||||||
|
:items="filterStudents(data)"
|
||||||
|
item-title="title"
|
||||||
|
item-value="value"
|
||||||
|
:label="t('choose-students')"
|
||||||
|
variant="outlined"
|
||||||
|
clearable
|
||||||
|
multiple
|
||||||
|
hide-details
|
||||||
|
density="compact"
|
||||||
|
chips
|
||||||
|
append-inner-icon="mdi-magnify"
|
||||||
|
></v-combobox>
|
||||||
|
|
||||||
|
<v-btn
|
||||||
|
@click="createGroup"
|
||||||
|
color="primary"
|
||||||
|
class="mt-2"
|
||||||
|
size="small"
|
||||||
|
>
|
||||||
|
{{ t("create-group") }}
|
||||||
|
</v-btn>
|
||||||
|
</v-card-text>
|
||||||
|
</using-query-result>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped></style>
|
|
@ -30,4 +30,10 @@ export class LearningPathController extends BaseController {
|
||||||
const dtos = await this.get<LearningPathDTO[]>("/", { theme });
|
const dtos = await this.get<LearningPathDTO[]>("/", { theme });
|
||||||
return dtos.map((dto) => LearningPath.fromDTO(dto));
|
return dtos.map((dto) => LearningPath.fromDTO(dto));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getAllLearningPaths(language: string | null = null): Promise<LearningPath[]> {
|
||||||
|
const query = language ? { language } : undefined;
|
||||||
|
const dtos = await this.get<LearningPathDTO[]>("/", query);
|
||||||
|
return dtos.map((dto) => LearningPath.fromDTO(dto));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,8 +57,21 @@
|
||||||
"legendNotCompletedYet": "Noch nicht fertig",
|
"legendNotCompletedYet": "Noch nicht fertig",
|
||||||
"legendCompleted": "Fertig",
|
"legendCompleted": "Fertig",
|
||||||
"legendTeacherExclusive": "Information für Lehrkräfte",
|
"legendTeacherExclusive": "Information für Lehrkräfte",
|
||||||
|
"new-assignment": "Neue Aufgabe",
|
||||||
|
"edit-assignment": "Zuordnung bearbeiten",
|
||||||
|
"groups": "Gruppen",
|
||||||
|
"learning-path": "Lernpfad",
|
||||||
|
"choose-lp": "Einen lernpfad auswählen",
|
||||||
|
"choose-classes": "Klassen wählen",
|
||||||
|
"create-groups": "Gruppen erstellen",
|
||||||
|
"title": "Titel",
|
||||||
|
"pick-class": "Wählen Sie eine klasse",
|
||||||
|
"choose-students": "Studenten auswählen",
|
||||||
|
"create-group": "Gruppe erstellen",
|
||||||
|
"class": "klasse",
|
||||||
|
"delete": "löschen",
|
||||||
|
"view-assignment": "Auftrag anzeigen",
|
||||||
"code": "code",
|
"code": "code",
|
||||||
"class": "Klasse",
|
|
||||||
"invitations": "Einladungen",
|
"invitations": "Einladungen",
|
||||||
"createClass": "Klasse erstellen",
|
"createClass": "Klasse erstellen",
|
||||||
"createClassInstructions": "Geben Sie einen Namen für Ihre Klasse ein und klicken Sie auf „Erstellen“. Es erscheint ein Fenster mit einem Code, den Sie kopieren können. Geben Sie diesen Code an Ihre Schüler weiter und sie können Ihrer Klasse beitreten.",
|
"createClassInstructions": "Geben Sie einen Namen für Ihre Klasse ein und klicken Sie auf „Erstellen“. Es erscheint ein Fenster mit einem Code, den Sie kopieren können. Geben Sie diesen Code an Ihre Schüler weiter und sie können Ihrer Klasse beitreten.",
|
||||||
|
@ -89,6 +102,11 @@
|
||||||
"noSubmissionsYet": "Noch keine Lösungen eingereicht.",
|
"noSubmissionsYet": "Noch keine Lösungen eingereicht.",
|
||||||
"viewAsGroup": "Fortschritt ansehen von Gruppe...",
|
"viewAsGroup": "Fortschritt ansehen von Gruppe...",
|
||||||
"assignLearningPath": "Als Aufgabe geben",
|
"assignLearningPath": "Als Aufgabe geben",
|
||||||
|
"group": "Gruppe",
|
||||||
|
"description": "Beschreibung",
|
||||||
|
"no-submission": "keine vorlage",
|
||||||
|
"submission": "Einreichung",
|
||||||
|
"progress": "Fortschritte",
|
||||||
"remove": "entfernen",
|
"remove": "entfernen",
|
||||||
"students": "Studenten",
|
"students": "Studenten",
|
||||||
"classJoinRequests": "Beitrittsanfragen",
|
"classJoinRequests": "Beitrittsanfragen",
|
||||||
|
@ -100,5 +118,7 @@
|
||||||
"accepted": "akzeptiert",
|
"accepted": "akzeptiert",
|
||||||
"enterUsername": "Geben Sie den Benutzernamen der Lehrkraft ein, die Sie einladen möchten",
|
"enterUsername": "Geben Sie den Benutzernamen der Lehrkraft ein, die Sie einladen möchten",
|
||||||
"username": "Nutzername",
|
"username": "Nutzername",
|
||||||
"invite": "einladen"
|
"invite": "einladen",
|
||||||
|
"searchAllLearningPathsTitle": "Alle Lernpfade durchsuchen",
|
||||||
|
"searchAllLearningPathsDescription": "Nicht gefunden, was Sie gesucht haben? Klicken Sie hier, um unsere gesamte Lernpfad-Datenbank zu durchsuchen."
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,8 +2,8 @@
|
||||||
"welcome": "Welcome",
|
"welcome": "Welcome",
|
||||||
"student": "student",
|
"student": "student",
|
||||||
"teacher": "teacher",
|
"teacher": "teacher",
|
||||||
"assignments": "assignments",
|
"assignments": "Assignments",
|
||||||
"classes": "classes",
|
"classes": "Classes",
|
||||||
"discussions": "discussions",
|
"discussions": "discussions",
|
||||||
"logout": "log out",
|
"logout": "log out",
|
||||||
"error_title": "Error",
|
"error_title": "Error",
|
||||||
|
@ -33,7 +33,7 @@
|
||||||
"JoinClassExplanation": "Enter the code the teacher has given you to join the class.",
|
"JoinClassExplanation": "Enter the code the teacher has given you to join the class.",
|
||||||
"invalidFormat": "Invalid format.",
|
"invalidFormat": "Invalid format.",
|
||||||
"submitCode": "submit",
|
"submitCode": "submit",
|
||||||
"members": "members",
|
"members": "Members",
|
||||||
"themes": "Themes",
|
"themes": "Themes",
|
||||||
"choose-theme": "Select a theme",
|
"choose-theme": "Select a theme",
|
||||||
"choose-age": "Select age",
|
"choose-age": "Select age",
|
||||||
|
@ -57,8 +57,21 @@
|
||||||
"older": "18 and older"
|
"older": "18 and older"
|
||||||
},
|
},
|
||||||
"read-more": "Read more",
|
"read-more": "Read more",
|
||||||
"code": "code",
|
"new-assignment": "New Assignment",
|
||||||
|
"edit-assignment": "Edit Assignment",
|
||||||
|
"groups": "Groups",
|
||||||
|
"learning-path": "Learning path",
|
||||||
|
"choose-lp": "Select a learning path",
|
||||||
|
"choose-classes": "Select classes",
|
||||||
|
"create-groups": "Create groups",
|
||||||
|
"title": "Title",
|
||||||
|
"pick-class": "Pick a class",
|
||||||
|
"choose-students": "Select students",
|
||||||
|
"create-group": "Create group",
|
||||||
"class": "class",
|
"class": "class",
|
||||||
|
"delete": "delete",
|
||||||
|
"view-assignment": "View assignment",
|
||||||
|
"code": "code",
|
||||||
"invitations": "invitations",
|
"invitations": "invitations",
|
||||||
"createClass": "create class",
|
"createClass": "create class",
|
||||||
"classname": "classname",
|
"classname": "classname",
|
||||||
|
@ -75,7 +88,6 @@
|
||||||
"sent": "sent",
|
"sent": "sent",
|
||||||
"failed": "failed",
|
"failed": "failed",
|
||||||
"wrong": "something went wrong",
|
"wrong": "something went wrong",
|
||||||
"created": "created",
|
|
||||||
"callbackLoading": "You are being logged in...",
|
"callbackLoading": "You are being logged in...",
|
||||||
"loginUnexpectedError": "Login failed",
|
"loginUnexpectedError": "Login failed",
|
||||||
"submitSolution": "Submit solution",
|
"submitSolution": "Submit solution",
|
||||||
|
@ -89,6 +101,12 @@
|
||||||
"noSubmissionsYet": "No submissions yet.",
|
"noSubmissionsYet": "No submissions yet.",
|
||||||
"viewAsGroup": "View progress of group...",
|
"viewAsGroup": "View progress of group...",
|
||||||
"assignLearningPath": "assign",
|
"assignLearningPath": "assign",
|
||||||
|
"group": "Group",
|
||||||
|
"description": "Description",
|
||||||
|
"no-submission": "no submission",
|
||||||
|
"submission": "Submission",
|
||||||
|
"progress": "Progress",
|
||||||
|
"created": "created",
|
||||||
"remove": "remove",
|
"remove": "remove",
|
||||||
"students": "students",
|
"students": "students",
|
||||||
"classJoinRequests": "join requests",
|
"classJoinRequests": "join requests",
|
||||||
|
@ -100,5 +118,7 @@
|
||||||
"rejected": "rejected",
|
"rejected": "rejected",
|
||||||
"enterUsername": "enter the username of the teacher you would like to invite",
|
"enterUsername": "enter the username of the teacher you would like to invite",
|
||||||
"username": "username",
|
"username": "username",
|
||||||
"invite": "invite"
|
"invite": "invite",
|
||||||
|
"searchAllLearningPathsTitle": "Search all learning paths",
|
||||||
|
"searchAllLearningPathsDescription": "You didn't find what you were looking for? Click here to search our whole database of available learning paths."
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,7 +33,7 @@
|
||||||
"JoinClassExplanation": "Entrez le code que l'enseignant vous a donné pour rejoindre la classe.",
|
"JoinClassExplanation": "Entrez le code que l'enseignant vous a donné pour rejoindre la classe.",
|
||||||
"invalidFormat": "Format non valide.",
|
"invalidFormat": "Format non valide.",
|
||||||
"submitCode": "envoyer",
|
"submitCode": "envoyer",
|
||||||
"members": "membres",
|
"members": "Membres",
|
||||||
"themes": "Thèmes",
|
"themes": "Thèmes",
|
||||||
"choose-theme": "Choisis un thème",
|
"choose-theme": "Choisis un thème",
|
||||||
"choose-age": "Choisis un âge",
|
"choose-age": "Choisis un âge",
|
||||||
|
@ -57,8 +57,21 @@
|
||||||
"older": "18 et plus"
|
"older": "18 et plus"
|
||||||
},
|
},
|
||||||
"read-more": "En savoir plus",
|
"read-more": "En savoir plus",
|
||||||
"code": "code",
|
"new-assignment": "Nouveau travail",
|
||||||
|
"edit-assignment": "Modifier le travail",
|
||||||
|
"groups": "Groupes",
|
||||||
|
"learning-path": "Parcours d'apprentissage",
|
||||||
|
"choose-lp": "Choisissez un parcours d'apprentissage",
|
||||||
|
"choose-classes": "Choisissez des classes",
|
||||||
|
"create-groups": "Créer des groupes",
|
||||||
|
"title": "Titre",
|
||||||
|
"pick-class": "Choisissez une classe",
|
||||||
|
"choose-students": "Sélectionnez des élèves",
|
||||||
|
"create-group": "Créer un groupe",
|
||||||
"class": "classe",
|
"class": "classe",
|
||||||
|
"delete": "supprimer",
|
||||||
|
"view-assignment": "Voir le travail",
|
||||||
|
"code": "code",
|
||||||
"invitations": "invitations",
|
"invitations": "invitations",
|
||||||
"createClass": "créer une classe",
|
"createClass": "créer une classe",
|
||||||
"createClassInstructions": "Entrez un nom pour votre classe et cliquez sur créer. Une fenêtre apparaît avec un code que vous pouvez copier. Donnez ce code à vos élèves et ils pourront rejoindre votre classe.",
|
"createClassInstructions": "Entrez un nom pour votre classe et cliquez sur créer. Une fenêtre apparaît avec un code que vous pouvez copier. Donnez ce code à vos élèves et ils pourront rejoindre votre classe.",
|
||||||
|
@ -89,6 +102,11 @@
|
||||||
"noSubmissionsYet": "Pas encore de soumissions.",
|
"noSubmissionsYet": "Pas encore de soumissions.",
|
||||||
"viewAsGroup": "Voir la progression du groupe...",
|
"viewAsGroup": "Voir la progression du groupe...",
|
||||||
"assignLearningPath": "donner comme tâche",
|
"assignLearningPath": "donner comme tâche",
|
||||||
|
"group": "Groupe",
|
||||||
|
"description": "Description",
|
||||||
|
"no-submission": "aucune soumission",
|
||||||
|
"submission": "Soumission",
|
||||||
|
"progress": "Progrès",
|
||||||
"remove": "supprimer",
|
"remove": "supprimer",
|
||||||
"students": "étudiants",
|
"students": "étudiants",
|
||||||
"classJoinRequests": "demandes d'adhésion",
|
"classJoinRequests": "demandes d'adhésion",
|
||||||
|
@ -100,5 +118,7 @@
|
||||||
"rejected": "rejetée",
|
"rejected": "rejetée",
|
||||||
"enterUsername": "entrez le nom d'utilisateur de l'enseignant que vous souhaitez inviter",
|
"enterUsername": "entrez le nom d'utilisateur de l'enseignant que vous souhaitez inviter",
|
||||||
"username": "Nom d'utilisateur",
|
"username": "Nom d'utilisateur",
|
||||||
"invite": "inviter"
|
"invite": "inviter",
|
||||||
|
"searchAllLearningPathsTitle": "Rechercher tous les parcours d'apprentissage",
|
||||||
|
"searchAllLearningPathsDescription": "Vous n'avez pas trouvé ce que vous cherchiez ? Cliquez ici pour rechercher dans toute notre base de données de parcours d'apprentissage disponibles."
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,8 +2,8 @@
|
||||||
"welcome": "Welkom",
|
"welcome": "Welkom",
|
||||||
"student": "leerling",
|
"student": "leerling",
|
||||||
"teacher": "leerkracht",
|
"teacher": "leerkracht",
|
||||||
"assignments": "opdrachten",
|
"assignments": "Opdrachten",
|
||||||
"classes": "klassen",
|
"classes": "Klassen",
|
||||||
"discussions": "discussies",
|
"discussions": "discussies",
|
||||||
"logout": "log uit",
|
"logout": "log uit",
|
||||||
"error_title": "Fout",
|
"error_title": "Fout",
|
||||||
|
@ -33,7 +33,7 @@
|
||||||
"JoinClassExplanation": "Voer de code in die je van de docent hebt gekregen om lid te worden van de klas.",
|
"JoinClassExplanation": "Voer de code in die je van de docent hebt gekregen om lid te worden van de klas.",
|
||||||
"invalidFormat": "Ongeldig formaat.",
|
"invalidFormat": "Ongeldig formaat.",
|
||||||
"submitCode": "verzenden",
|
"submitCode": "verzenden",
|
||||||
"members": "leden",
|
"members": "Leden",
|
||||||
"themes": "Lesthema's",
|
"themes": "Lesthema's",
|
||||||
"choose-theme": "Kies een thema",
|
"choose-theme": "Kies een thema",
|
||||||
"choose-age": "Kies een leeftijd",
|
"choose-age": "Kies een leeftijd",
|
||||||
|
@ -57,8 +57,21 @@
|
||||||
"older": "Hoger onderwijs"
|
"older": "Hoger onderwijs"
|
||||||
},
|
},
|
||||||
"read-more": "Lees meer",
|
"read-more": "Lees meer",
|
||||||
"code": "code",
|
"new-assignment": "Nieuwe opdracht",
|
||||||
|
"edit-assignment": "Opdracht bewerken",
|
||||||
|
"groups": "Groepen",
|
||||||
|
"learning-path": "Leerpad",
|
||||||
|
"choose-lp": "Kies een leerpad",
|
||||||
|
"choose-classes": "Kies klassen",
|
||||||
|
"create-groups": "Groepen maken",
|
||||||
|
"title": "Titel",
|
||||||
|
"pick-class": "Kies een klas",
|
||||||
|
"choose-students": "Studenten selecteren",
|
||||||
|
"create-group": "Groep aanmaken",
|
||||||
"class": "klas",
|
"class": "klas",
|
||||||
|
"delete": "verwijderen",
|
||||||
|
"view-assignment": "Opdracht bekijken",
|
||||||
|
"code": "code",
|
||||||
"invitations": "uitnodigingen",
|
"invitations": "uitnodigingen",
|
||||||
"createClass": "klas aanmaken",
|
"createClass": "klas aanmaken",
|
||||||
"createClassInstructions": "Voer een naam in voor je klas en klik op create. Er verschijnt een venster met een code die je kunt kopiëren. Geef deze code aan je leerlingen en ze kunnen deelnemen aan je klas.",
|
"createClassInstructions": "Voer een naam in voor je klas en klik op create. Er verschijnt een venster met een code die je kunt kopiëren. Geef deze code aan je leerlingen en ze kunnen deelnemen aan je klas.",
|
||||||
|
@ -89,6 +102,11 @@
|
||||||
"noSubmissionsYet": "Nog geen indieningen.",
|
"noSubmissionsYet": "Nog geen indieningen.",
|
||||||
"viewAsGroup": "Vooruitgang bekijken van groep...",
|
"viewAsGroup": "Vooruitgang bekijken van groep...",
|
||||||
"assignLearningPath": "Als opdracht geven",
|
"assignLearningPath": "Als opdracht geven",
|
||||||
|
"group": "Groep",
|
||||||
|
"description": "Beschrijving",
|
||||||
|
"no-submission": "geen indiening",
|
||||||
|
"submission": "Indiening",
|
||||||
|
"progress": "Vooruitgang",
|
||||||
"remove": "verwijder",
|
"remove": "verwijder",
|
||||||
"students": "studenten",
|
"students": "studenten",
|
||||||
"classJoinRequests": "deelname verzoeken",
|
"classJoinRequests": "deelname verzoeken",
|
||||||
|
@ -100,5 +118,7 @@
|
||||||
"rejected": "geweigerd",
|
"rejected": "geweigerd",
|
||||||
"enterUsername": "vul de gebruikersnaam van de leerkracht die je wilt uitnodigen in",
|
"enterUsername": "vul de gebruikersnaam van de leerkracht die je wilt uitnodigen in",
|
||||||
"username": "gebruikersnaam",
|
"username": "gebruikersnaam",
|
||||||
"invite": "uitnodigen"
|
"invite": "uitnodigen",
|
||||||
|
"searchAllLearningPathsTitle": "Alle leerpaden doorzoeken",
|
||||||
|
"searchAllLearningPathsDescription": "Niet gevonden waar je naar op zoek was? Klik hier om onze volledige databank van beschikbare leerpaden te doorzoeken."
|
||||||
}
|
}
|
||||||
|
|
|
@ -104,7 +104,7 @@ export function useAssignmentsQuery(
|
||||||
export function useAssignmentQuery(
|
export function useAssignmentQuery(
|
||||||
classid: MaybeRefOrGetter<string | undefined>,
|
classid: MaybeRefOrGetter<string | undefined>,
|
||||||
assignmentNumber: MaybeRefOrGetter<number | undefined>,
|
assignmentNumber: MaybeRefOrGetter<number | undefined>,
|
||||||
): UseQueryReturnType<AssignmentsResponse, Error> {
|
): UseQueryReturnType<AssignmentResponse, Error> {
|
||||||
const { cid, an } = toValues(classid, assignmentNumber, 1, true);
|
const { cid, an } = toValues(classid, assignmentNumber, 1, true);
|
||||||
|
|
||||||
return useQuery({
|
return useQuery({
|
||||||
|
@ -146,7 +146,7 @@ export function useDeleteAssignmentMutation(): UseMutationReturnType<
|
||||||
|
|
||||||
await invalidateAllAssignmentKeys(queryClient, cid, an);
|
await invalidateAllAssignmentKeys(queryClient, cid, an);
|
||||||
await invalidateAllGroupKeys(queryClient, cid, an);
|
await invalidateAllGroupKeys(queryClient, cid, an);
|
||||||
await invalidateAllSubmissionKeys(queryClient, cid, an);
|
await invalidateAllSubmissionKeys(queryClient, undefined, undefined, undefined, cid, an);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -46,3 +46,16 @@ export function useSearchLearningPathQuery(
|
||||||
enabled: () => Boolean(toValue(query)),
|
enabled: () => Boolean(toValue(query)),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function useGetAllLearningPaths(
|
||||||
|
language: MaybeRefOrGetter<string | undefined>,
|
||||||
|
): UseQueryReturnType<LearningPath[], Error> {
|
||||||
|
return useQuery({
|
||||||
|
queryKey: [LEARNING_PATH_KEY, "getAllLearningPaths", language],
|
||||||
|
queryFn: async () => {
|
||||||
|
const lang = toValue(language);
|
||||||
|
return learningPathController.getAllLearningPaths(lang);
|
||||||
|
},
|
||||||
|
enabled: () => Boolean(toValue(language)),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
import { computed, toValue } from "vue";
|
import { computed, type Ref, toValue } from "vue";
|
||||||
import type { MaybeRefOrGetter } from "vue";
|
import type { MaybeRefOrGetter } from "vue";
|
||||||
import {
|
import {
|
||||||
|
type QueryObserverResult,
|
||||||
useMutation,
|
useMutation,
|
||||||
type UseMutationReturnType,
|
type UseMutationReturnType,
|
||||||
|
useQueries,
|
||||||
useQuery,
|
useQuery,
|
||||||
useQueryClient,
|
useQueryClient,
|
||||||
type UseQueryReturnType,
|
type UseQueryReturnType,
|
||||||
|
@ -70,6 +72,20 @@ export function useStudentQuery(
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function useStudentsByUsernamesQuery(
|
||||||
|
usernames: MaybeRefOrGetter<string[] | undefined>,
|
||||||
|
): Ref<QueryObserverResult<StudentResponse>[]> {
|
||||||
|
const resolvedUsernames = toValue(usernames) ?? [];
|
||||||
|
|
||||||
|
return useQueries({
|
||||||
|
queries: resolvedUsernames?.map((username) => ({
|
||||||
|
queryKey: computed(() => studentQueryKey(toValue(username))),
|
||||||
|
queryFn: async () => studentController.getByUsername(toValue(username)),
|
||||||
|
enabled: Boolean(toValue(username)),
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
export function useStudentClassesQuery(
|
export function useStudentClassesQuery(
|
||||||
username: MaybeRefOrGetter<string | undefined>,
|
username: MaybeRefOrGetter<string | undefined>,
|
||||||
full: MaybeRefOrGetter<boolean> = true,
|
full: MaybeRefOrGetter<boolean> = true,
|
||||||
|
|
|
@ -7,13 +7,13 @@ import CreateAssignment from "@/views/assignments/CreateAssignment.vue";
|
||||||
import CreateDiscussion from "@/views/discussions/CreateDiscussion.vue";
|
import CreateDiscussion from "@/views/discussions/CreateDiscussion.vue";
|
||||||
import CallbackPage from "@/views/CallbackPage.vue";
|
import CallbackPage from "@/views/CallbackPage.vue";
|
||||||
import UserClasses from "@/views/classes/UserClasses.vue";
|
import UserClasses from "@/views/classes/UserClasses.vue";
|
||||||
import UserAssignments from "@/views/classes/UserAssignments.vue";
|
import UserAssignments from "@/views/assignments/UserAssignments.vue";
|
||||||
import authService from "@/services/auth/auth-service.ts";
|
|
||||||
import LearningPathPage from "@/views/learning-paths/LearningPathPage.vue";
|
import LearningPathPage from "@/views/learning-paths/LearningPathPage.vue";
|
||||||
import LearningPathSearchPage from "@/views/learning-paths/LearningPathSearchPage.vue";
|
import LearningPathSearchPage from "@/views/learning-paths/LearningPathSearchPage.vue";
|
||||||
import UserHomePage from "@/views/homepage/UserHomePage.vue";
|
import UserHomePage from "@/views/homepage/UserHomePage.vue";
|
||||||
import SingleTheme from "@/views/SingleTheme.vue";
|
import SingleTheme from "@/views/SingleTheme.vue";
|
||||||
import LearningObjectView from "@/views/learning-paths/learning-object/LearningObjectView.vue";
|
import LearningObjectView from "@/views/learning-paths/learning-object/LearningObjectView.vue";
|
||||||
|
import authService from "@/services/auth/auth-service";
|
||||||
|
|
||||||
const router = createRouter({
|
const router = createRouter({
|
||||||
history: createWebHistory(import.meta.env.BASE_URL),
|
history: createWebHistory(import.meta.env.BASE_URL),
|
||||||
|
@ -72,16 +72,20 @@ const router = createRouter({
|
||||||
meta: { requiresAuth: true },
|
meta: { requiresAuth: true },
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: "/assignment/create",
|
path: "/assignment",
|
||||||
name: "CreateAssigment",
|
|
||||||
component: CreateAssignment,
|
|
||||||
meta: { requiresAuth: true },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
path: "/assignment/:id",
|
|
||||||
name: "SingleAssigment",
|
|
||||||
component: SingleAssignment,
|
|
||||||
meta: { requiresAuth: true },
|
meta: { requiresAuth: true },
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
path: "create",
|
||||||
|
name: "CreateAssigment",
|
||||||
|
component: CreateAssignment,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: ":classId/:id",
|
||||||
|
name: "SingleAssigment",
|
||||||
|
component: SingleAssignment,
|
||||||
|
},
|
||||||
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: "/class/:id",
|
path: "/class/:id",
|
||||||
|
|
76
frontend/src/utils/assignment-rules.ts
Normal file
76
frontend/src/utils/assignment-rules.ts
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
/**
|
||||||
|
* Validation rule for the assignment title.
|
||||||
|
*
|
||||||
|
* Ensures that the title is not empty.
|
||||||
|
*/
|
||||||
|
export const assignmentTitleRules = [
|
||||||
|
(value: string): string | boolean => {
|
||||||
|
if (value?.length >= 1) {
|
||||||
|
return true;
|
||||||
|
} // Title must not be empty
|
||||||
|
return "Title cannot be empty.";
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validation rule for the learning path selection.
|
||||||
|
*
|
||||||
|
* Ensures that a valid learning path is selected.
|
||||||
|
*/
|
||||||
|
export const learningPathRules = [
|
||||||
|
(value: { hruid: string; title: string }): string | boolean => {
|
||||||
|
if (value && value.hruid) {
|
||||||
|
return true; // Valid if hruid is present
|
||||||
|
}
|
||||||
|
return "You must select a learning path.";
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validation rule for the classes selection.
|
||||||
|
*
|
||||||
|
* Ensures that at least one class is selected.
|
||||||
|
*/
|
||||||
|
export const classRules = [
|
||||||
|
(value: string): string | boolean => {
|
||||||
|
if (value) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return "You must select at least one class.";
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validation rule for the deadline field.
|
||||||
|
*
|
||||||
|
* Ensures that a valid deadline is selected and is in the future.
|
||||||
|
*/
|
||||||
|
export const deadlineRules = [
|
||||||
|
(value: string): string | boolean => {
|
||||||
|
if (!value) {
|
||||||
|
return "You must set a deadline.";
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectedDateTime = new Date(value);
|
||||||
|
const now = new Date();
|
||||||
|
|
||||||
|
if (isNaN(selectedDateTime.getTime())) {
|
||||||
|
return "Invalid date or time.";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedDateTime <= now) {
|
||||||
|
return "The deadline must be in the future.";
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const descriptionRules = [
|
||||||
|
(value: string): string | boolean => {
|
||||||
|
if (!value || value.trim() === "") {
|
||||||
|
return "Description cannot be empty.";
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
];
|
|
@ -1,14 +1,258 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { computed, onMounted, ref, watch } from "vue";
|
||||||
|
import GroupSelector from "@/components/assignments/GroupSelector.vue";
|
||||||
|
import { assignmentTitleRules, classRules, descriptionRules, learningPathRules } from "@/utils/assignment-rules.ts";
|
||||||
|
import DeadlineSelector from "@/components/assignments/DeadlineSelector.vue";
|
||||||
|
import auth from "@/services/auth/auth-service.ts";
|
||||||
|
import { useTeacherClassesQuery } from "@/queries/teachers.ts";
|
||||||
|
import { useRouter } from "vue-router";
|
||||||
|
import { useGetAllLearningPaths } from "@/queries/learning-paths.ts";
|
||||||
|
import UsingQueryResult from "@/components/UsingQueryResult.vue";
|
||||||
|
import type { LearningPath } from "@/data-objects/learning-paths/learning-path.ts";
|
||||||
|
import type { ClassesResponse } from "@/controllers/classes.ts";
|
||||||
|
import type { AssignmentDTO } from "@dwengo-1/common/interfaces/assignment";
|
||||||
|
import { useCreateAssignmentMutation } from "@/queries/assignments.ts";
|
||||||
import { useRoute } from "vue-router";
|
import { useRoute } from "vue-router";
|
||||||
|
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const { t, locale } = useI18n();
|
||||||
|
const role = ref(auth.authState.activeRole);
|
||||||
|
const username = ref<string>("");
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
// Redirect student
|
||||||
|
if (role.value === "student") {
|
||||||
|
await router.push("/user");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the user's username
|
||||||
|
const user = await auth.loadUser();
|
||||||
|
username.value = user?.profile?.preferred_username ?? "";
|
||||||
|
});
|
||||||
|
|
||||||
|
const language = computed(() => locale.value);
|
||||||
|
const form = ref();
|
||||||
|
|
||||||
|
//Fetch all learning paths
|
||||||
|
const learningPathsQueryResults = useGetAllLearningPaths(language);
|
||||||
|
|
||||||
|
// Fetch and store all the teacher's classes
|
||||||
|
const classesQueryResults = useTeacherClassesQuery(username, true);
|
||||||
|
|
||||||
|
const selectedClass = ref(undefined);
|
||||||
|
|
||||||
|
const assignmentTitle = ref("");
|
||||||
|
const selectedLearningPath = ref(route.query.hruid || undefined);
|
||||||
|
|
||||||
|
// Disable combobox when learningPath prop is passed
|
||||||
|
const lpIsSelected = route.query.hruid !== undefined;
|
||||||
|
const deadline = ref(null);
|
||||||
|
const description = ref("");
|
||||||
|
const groups = ref<string[][]>([]);
|
||||||
|
|
||||||
|
// New group is added to the list
|
||||||
|
function addGroupToList(students: string[]): void {
|
||||||
|
if (students.length) {
|
||||||
|
groups.value = [...groups.value, students];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(selectedClass, () => {
|
||||||
|
groups.value = [];
|
||||||
|
});
|
||||||
|
|
||||||
|
const { mutate, data, isSuccess } = useCreateAssignmentMutation();
|
||||||
|
|
||||||
|
watch([isSuccess, data], async ([success, newData]) => {
|
||||||
|
if (success && newData?.assignment) {
|
||||||
|
await router.push(`/assignment/${newData.assignment.within}/${newData.assignment.id}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
async function submitFormHandler(): Promise<void> {
|
||||||
|
const { valid } = await form.value.validate();
|
||||||
|
if (!valid) return;
|
||||||
|
|
||||||
|
let lp = selectedLearningPath.value;
|
||||||
|
if (!lpIsSelected) {
|
||||||
|
lp = selectedLearningPath.value?.hruid;
|
||||||
|
}
|
||||||
|
|
||||||
|
const assignmentDTO: AssignmentDTO = {
|
||||||
|
id: 0,
|
||||||
|
within: selectedClass.value?.id || "",
|
||||||
|
title: assignmentTitle.value,
|
||||||
|
description: description.value,
|
||||||
|
learningPath: lp || "",
|
||||||
|
language: language.value,
|
||||||
|
groups: groups.value,
|
||||||
|
};
|
||||||
|
|
||||||
|
mutate({ cid: assignmentDTO.within, data: assignmentDTO });
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<main>
|
<div class="main-container">
|
||||||
Hier zou de pagina staan om een assignment aan te maken voor de leerpad met hruid {{ route.query.hruid }} en
|
<h1 class="title">{{ t("new-assignment") }}</h1>
|
||||||
language {{ route.query.language }}. (Overschrijf dit)
|
<v-card class="form-card">
|
||||||
</main>
|
<v-form
|
||||||
|
ref="form"
|
||||||
|
class="form-container"
|
||||||
|
validate-on="submit lazy"
|
||||||
|
@submit.prevent="submitFormHandler"
|
||||||
|
>
|
||||||
|
<v-container class="step-container">
|
||||||
|
<v-card-text>
|
||||||
|
<v-text-field
|
||||||
|
v-model="assignmentTitle"
|
||||||
|
:label="t('title')"
|
||||||
|
:rules="assignmentTitleRules"
|
||||||
|
density="compact"
|
||||||
|
variant="outlined"
|
||||||
|
clearable
|
||||||
|
required
|
||||||
|
></v-text-field>
|
||||||
|
</v-card-text>
|
||||||
|
|
||||||
|
<using-query-result
|
||||||
|
:query-result="learningPathsQueryResults"
|
||||||
|
v-slot="{ data }: { data: LearningPath[] }"
|
||||||
|
>
|
||||||
|
<v-card-text>
|
||||||
|
<v-combobox
|
||||||
|
v-model="selectedLearningPath"
|
||||||
|
:items="data"
|
||||||
|
:label="t('choose-lp')"
|
||||||
|
:rules="learningPathRules"
|
||||||
|
variant="outlined"
|
||||||
|
clearable
|
||||||
|
hide-details
|
||||||
|
density="compact"
|
||||||
|
append-inner-icon="mdi-magnify"
|
||||||
|
item-title="title"
|
||||||
|
item-value="hruid"
|
||||||
|
required
|
||||||
|
:disabled="lpIsSelected"
|
||||||
|
:filter="
|
||||||
|
(item, query: string) => item.title.toLowerCase().includes(query.toLowerCase())
|
||||||
|
"
|
||||||
|
></v-combobox>
|
||||||
|
</v-card-text>
|
||||||
|
</using-query-result>
|
||||||
|
|
||||||
|
<using-query-result
|
||||||
|
:query-result="classesQueryResults"
|
||||||
|
v-slot="{ data }: { data: ClassesResponse }"
|
||||||
|
>
|
||||||
|
<v-card-text>
|
||||||
|
<v-combobox
|
||||||
|
v-model="selectedClass"
|
||||||
|
:items="data?.classes ?? []"
|
||||||
|
:label="t('pick-class')"
|
||||||
|
:rules="classRules"
|
||||||
|
variant="outlined"
|
||||||
|
clearable
|
||||||
|
hide-details
|
||||||
|
density="compact"
|
||||||
|
append-inner-icon="mdi-magnify"
|
||||||
|
item-title="displayName"
|
||||||
|
item-value="id"
|
||||||
|
required
|
||||||
|
></v-combobox>
|
||||||
|
</v-card-text>
|
||||||
|
</using-query-result>
|
||||||
|
|
||||||
|
<GroupSelector
|
||||||
|
:classId="selectedClass?.id"
|
||||||
|
:groups="groups"
|
||||||
|
@groupCreated="addGroupToList"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- Counter for created groups -->
|
||||||
|
<v-card-text v-if="groups.length">
|
||||||
|
<strong>Created Groups: {{ groups.length }}</strong>
|
||||||
|
</v-card-text>
|
||||||
|
<DeadlineSelector v-model:deadline="deadline" />
|
||||||
|
<v-card-text>
|
||||||
|
<v-textarea
|
||||||
|
v-model="description"
|
||||||
|
:label="t('description')"
|
||||||
|
variant="outlined"
|
||||||
|
density="compact"
|
||||||
|
auto-grow
|
||||||
|
rows="3"
|
||||||
|
:rules="descriptionRules"
|
||||||
|
></v-textarea>
|
||||||
|
</v-card-text>
|
||||||
|
<v-card-text>
|
||||||
|
<v-btn
|
||||||
|
class="mt-2"
|
||||||
|
color="secondary"
|
||||||
|
type="submit"
|
||||||
|
block
|
||||||
|
>{{ t("submit") }}
|
||||||
|
</v-btn>
|
||||||
|
<v-btn
|
||||||
|
to="/user/assignment"
|
||||||
|
color="grey"
|
||||||
|
block
|
||||||
|
>{{ t("cancel") }}
|
||||||
|
</v-btn>
|
||||||
|
</v-card-text>
|
||||||
|
</v-container>
|
||||||
|
</v-form>
|
||||||
|
</v-card>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped></style>
|
<style scoped>
|
||||||
|
.main-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-card {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 55%;
|
||||||
|
/*padding: 1%;*/
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-container {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.step-container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
flex-direction: column;
|
||||||
|
min-height: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 1000px) {
|
||||||
|
.form-card {
|
||||||
|
width: 70%;
|
||||||
|
padding: 1%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.step-container {
|
||||||
|
min-height: 300px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 650px) {
|
||||||
|
.form-card {
|
||||||
|
width: 95%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
|
@ -1,7 +1,75 @@
|
||||||
<script setup lang="ts"></script>
|
<script setup lang="ts">
|
||||||
|
import auth from "@/services/auth/auth-service.ts";
|
||||||
|
import { computed, type Ref, ref, watchEffect } from "vue";
|
||||||
|
import StudentAssignment from "@/views/assignments/StudentAssignment.vue";
|
||||||
|
import TeacherAssignment from "@/views/assignments/TeacherAssignment.vue";
|
||||||
|
import { useRoute } from "vue-router";
|
||||||
|
import type { Language } from "@/data-objects/language.ts";
|
||||||
|
import { useGetLearningPathQuery } from "@/queries/learning-paths.ts";
|
||||||
|
import type { LearningPath } from "@/data-objects/learning-paths/learning-path.ts";
|
||||||
|
import type { GroupDTO } from "@dwengo-1/common/interfaces/group";
|
||||||
|
|
||||||
|
const role = auth.authState.activeRole;
|
||||||
|
const isTeacher = computed(() => role === "teacher");
|
||||||
|
|
||||||
|
const route = useRoute();
|
||||||
|
const classId = ref<string>(route.params.classId as string);
|
||||||
|
const assignmentId = ref(Number(route.params.id));
|
||||||
|
|
||||||
|
function useGroupsWithProgress(
|
||||||
|
groups: Ref<GroupDTO[]>,
|
||||||
|
hruid: Ref<string>,
|
||||||
|
language: Ref<string>,
|
||||||
|
): { groupProgressMap: Map<number, number> } {
|
||||||
|
const groupProgressMap: Map<number, number> = new Map<number, number>();
|
||||||
|
|
||||||
|
watchEffect(() => {
|
||||||
|
// Clear existing entries to avoid stale data
|
||||||
|
groupProgressMap.clear();
|
||||||
|
|
||||||
|
const lang = ref(language.value as Language);
|
||||||
|
|
||||||
|
groups.value.forEach((group) => {
|
||||||
|
const groupKey = group.groupNumber;
|
||||||
|
const forGroup = ref({
|
||||||
|
forGroup: groupKey,
|
||||||
|
assignmentNo: assignmentId,
|
||||||
|
classId: classId,
|
||||||
|
});
|
||||||
|
|
||||||
|
const query = useGetLearningPathQuery(hruid.value, lang, forGroup);
|
||||||
|
|
||||||
|
const data = query.data.value;
|
||||||
|
|
||||||
|
groupProgressMap.set(groupKey, data ? calculateProgress(data) : 0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
groupProgressMap,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function calculateProgress(lp: LearningPath): number {
|
||||||
|
return ((lp.amountOfNodes - lp.amountOfNodesLeft) / lp.amountOfNodes) * 100;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<main></main>
|
<TeacherAssignment
|
||||||
|
:class-id="classId"
|
||||||
|
:assignment-id="assignmentId"
|
||||||
|
:use-groups-with-progress="useGroupsWithProgress"
|
||||||
|
v-if="isTeacher"
|
||||||
|
>
|
||||||
|
</TeacherAssignment>
|
||||||
|
<StudentAssignment
|
||||||
|
:class-id="classId"
|
||||||
|
:assignment-id="assignmentId"
|
||||||
|
:use-groups-with-progress="useGroupsWithProgress"
|
||||||
|
v-else
|
||||||
|
>
|
||||||
|
</StudentAssignment>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
|
167
frontend/src/views/assignments/StudentAssignment.vue
Normal file
167
frontend/src/views/assignments/StudentAssignment.vue
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, computed, type Ref } from "vue";
|
||||||
|
import auth from "@/services/auth/auth-service.ts";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { useAssignmentQuery } from "@/queries/assignments.ts";
|
||||||
|
import UsingQueryResult from "@/components/UsingQueryResult.vue";
|
||||||
|
import type { AssignmentResponse } from "@/controllers/assignments.ts";
|
||||||
|
import { asyncComputed } from "@vueuse/core";
|
||||||
|
import { useStudentsByUsernamesQuery } from "@/queries/students.ts";
|
||||||
|
import { useGroupsQuery } from "@/queries/groups.ts";
|
||||||
|
import { useGetLearningPathQuery } from "@/queries/learning-paths.ts";
|
||||||
|
import type { Language } from "@/data-objects/language.ts";
|
||||||
|
import type { GroupDTO } from "@dwengo-1/common/interfaces/group";
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
classId: string;
|
||||||
|
assignmentId: number;
|
||||||
|
useGroupsWithProgress: (
|
||||||
|
groups: Ref<GroupDTO[]>,
|
||||||
|
hruid: Ref<string>,
|
||||||
|
language: Ref<Language>,
|
||||||
|
) => { groupProgressMap: Map<number, number> };
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const { t, locale } = useI18n();
|
||||||
|
const language = ref<Language>(locale.value as Language);
|
||||||
|
const learningPath = ref();
|
||||||
|
// Get the user's username/id
|
||||||
|
const username = asyncComputed(async () => {
|
||||||
|
const user = await auth.loadUser();
|
||||||
|
return user?.profile?.preferred_username ?? undefined;
|
||||||
|
});
|
||||||
|
|
||||||
|
const assignmentQueryResult = useAssignmentQuery(() => props.classId, props.assignmentId);
|
||||||
|
learningPath.value = assignmentQueryResult.data.value?.assignment?.learningPath;
|
||||||
|
|
||||||
|
const submitted = ref(false); //TODO: update by fetching submissions and check if group submitted
|
||||||
|
|
||||||
|
const lpQueryResult = useGetLearningPathQuery(
|
||||||
|
computed(() => assignmentQueryResult.data.value?.assignment?.learningPath ?? ""),
|
||||||
|
computed(() => language.value),
|
||||||
|
);
|
||||||
|
|
||||||
|
const groupsQueryResult = useGroupsQuery(props.classId, props.assignmentId, true);
|
||||||
|
const group = computed(() =>
|
||||||
|
groupsQueryResult?.data.value?.groups.find((group) =>
|
||||||
|
group.members?.some((m) => m.username === username.value),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
const _groupArray = computed(() => (group.value ? [group.value] : []));
|
||||||
|
const progressValue = ref(0);
|
||||||
|
/* Crashes right now cause api data has inexistent hruid TODO: uncomment later and use it in progress bar
|
||||||
|
Const {groupProgressMap} = props.useGroupsWithProgress(
|
||||||
|
groupArray,
|
||||||
|
learningPath,
|
||||||
|
language
|
||||||
|
);
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Assuming group.value.members is a list of usernames TODO: case when it's StudentDTO's
|
||||||
|
const studentQueries = useStudentsByUsernamesQuery(() => group.value?.members as string[]);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="container">
|
||||||
|
<using-query-result
|
||||||
|
:query-result="assignmentQueryResult"
|
||||||
|
v-slot="{ data }: { data: AssignmentResponse }"
|
||||||
|
>
|
||||||
|
<v-card
|
||||||
|
v-if="data"
|
||||||
|
class="assignment-card"
|
||||||
|
>
|
||||||
|
<div class="top-buttons">
|
||||||
|
<v-btn
|
||||||
|
icon
|
||||||
|
variant="text"
|
||||||
|
class="back-btn"
|
||||||
|
to="/user/assignment"
|
||||||
|
>
|
||||||
|
<v-icon>mdi-arrow-left</v-icon>
|
||||||
|
</v-btn>
|
||||||
|
|
||||||
|
<v-chip
|
||||||
|
v-if="submitted"
|
||||||
|
class="ma-2 top-right-btn"
|
||||||
|
label
|
||||||
|
color="success"
|
||||||
|
>
|
||||||
|
{{ t("submitted") }}
|
||||||
|
</v-chip>
|
||||||
|
</div>
|
||||||
|
<v-card-title class="text-h4 assignmentTopTitle">{{ data.assignment.title }}</v-card-title>
|
||||||
|
|
||||||
|
<v-card-subtitle class="subtitle-section">
|
||||||
|
<using-query-result
|
||||||
|
:query-result="lpQueryResult"
|
||||||
|
v-slot="{ data: lpData }"
|
||||||
|
>
|
||||||
|
<v-btn
|
||||||
|
v-if="lpData"
|
||||||
|
:to="`/learningPath/${lpData.hruid}/${language}/${lpData.startNode.learningobjectHruid}?forGroup=${group?.groupNumber}&assignmentNo=${assignmentId}&classId=${classId}`"
|
||||||
|
variant="tonal"
|
||||||
|
color="primary"
|
||||||
|
>
|
||||||
|
{{ t("learning-path") }}
|
||||||
|
</v-btn>
|
||||||
|
</using-query-result>
|
||||||
|
</v-card-subtitle>
|
||||||
|
|
||||||
|
<v-card-text class="description">
|
||||||
|
{{ data.assignment.description }}
|
||||||
|
</v-card-text>
|
||||||
|
<v-card-text>
|
||||||
|
<v-row
|
||||||
|
align="center"
|
||||||
|
no-gutters
|
||||||
|
>
|
||||||
|
<v-col cols="auto">
|
||||||
|
<span class="progress-label">{{ t("progress") + ": " }}</span>
|
||||||
|
</v-col>
|
||||||
|
<v-col>
|
||||||
|
<v-progress-linear
|
||||||
|
:model-value="progressValue"
|
||||||
|
color="primary"
|
||||||
|
height="20"
|
||||||
|
class="progress-bar"
|
||||||
|
>
|
||||||
|
<template v-slot:default="{ value }">
|
||||||
|
<strong>{{ Math.ceil(value) }}%</strong>
|
||||||
|
</template>
|
||||||
|
</v-progress-linear>
|
||||||
|
</v-col>
|
||||||
|
</v-row>
|
||||||
|
</v-card-text>
|
||||||
|
|
||||||
|
<v-card-text class="group-section">
|
||||||
|
<h3>{{ t("group") }}</h3>
|
||||||
|
<div v-if="studentQueries">
|
||||||
|
<ul>
|
||||||
|
<li
|
||||||
|
v-for="student in group?.members"
|
||||||
|
:key="student.username"
|
||||||
|
>
|
||||||
|
{{ student.firstName + " " + student.lastName }}
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</v-card-text>
|
||||||
|
</v-card>
|
||||||
|
</using-query-result>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
@import "@/assets/assignment.css";
|
||||||
|
|
||||||
|
.progress-label {
|
||||||
|
font-weight: bold;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress-bar {
|
||||||
|
width: 40%;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -1,7 +0,0 @@
|
||||||
<script setup lang="ts"></script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<main></main>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped></style>
|
|
234
frontend/src/views/assignments/TeacherAssignment.vue
Normal file
234
frontend/src/views/assignments/TeacherAssignment.vue
Normal file
|
@ -0,0 +1,234 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, type Ref, ref } from "vue";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { useAssignmentQuery, useDeleteAssignmentMutation } from "@/queries/assignments.ts";
|
||||||
|
import UsingQueryResult from "@/components/UsingQueryResult.vue";
|
||||||
|
import { useGroupsQuery } from "@/queries/groups.ts";
|
||||||
|
import { useGetLearningPathQuery } from "@/queries/learning-paths.ts";
|
||||||
|
import type { Language } from "@/data-objects/language.ts";
|
||||||
|
import type { AssignmentResponse } from "@/controllers/assignments.ts";
|
||||||
|
import type { GroupDTO } from "@dwengo-1/common/interfaces/group";
|
||||||
|
|
||||||
|
const props = defineProps<{
|
||||||
|
classId: string;
|
||||||
|
assignmentId: number;
|
||||||
|
useGroupsWithProgress: (
|
||||||
|
groups: Ref<GroupDTO[]>,
|
||||||
|
hruid: Ref<string>,
|
||||||
|
language: Ref<Language>,
|
||||||
|
) => { groupProgressMap: Map<number, number> };
|
||||||
|
}>();
|
||||||
|
|
||||||
|
const { t, locale } = useI18n();
|
||||||
|
const language = computed(() => locale.value);
|
||||||
|
const groups = ref();
|
||||||
|
const learningPath = ref();
|
||||||
|
|
||||||
|
const assignmentQueryResult = useAssignmentQuery(() => props.classId, props.assignmentId);
|
||||||
|
learningPath.value = assignmentQueryResult.data.value?.assignment?.learningPath;
|
||||||
|
// Get learning path object
|
||||||
|
const lpQueryResult = useGetLearningPathQuery(
|
||||||
|
computed(() => assignmentQueryResult.data.value?.assignment?.learningPath ?? ""),
|
||||||
|
computed(() => language.value as Language),
|
||||||
|
);
|
||||||
|
|
||||||
|
// Get all the groups withing the assignment
|
||||||
|
const groupsQueryResult = useGroupsQuery(props.classId, props.assignmentId, true);
|
||||||
|
groups.value = groupsQueryResult.data.value?.groups;
|
||||||
|
|
||||||
|
/* Crashes right now cause api data has inexistent hruid TODO: uncomment later and use it in progress bar
|
||||||
|
Const {groupProgressMap} = props.useGroupsWithProgress(
|
||||||
|
groups,
|
||||||
|
learningPath,
|
||||||
|
language
|
||||||
|
);
|
||||||
|
*/
|
||||||
|
|
||||||
|
const allGroups = computed(() => {
|
||||||
|
const groups = groupsQueryResult.data.value?.groups;
|
||||||
|
if (!groups) return [];
|
||||||
|
|
||||||
|
return groups.map((group) => ({
|
||||||
|
name: `${t("group")} ${group.groupNumber}`,
|
||||||
|
progress: 0, //GroupProgressMap[group.groupNumber],
|
||||||
|
members: group.members,
|
||||||
|
submitted: false, //TODO: fetch from submission
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
|
||||||
|
const dialog = ref(false);
|
||||||
|
const selectedGroup = ref({});
|
||||||
|
|
||||||
|
function openGroupDetails(group): void {
|
||||||
|
selectedGroup.value = group;
|
||||||
|
dialog.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
const headers = computed(() => [
|
||||||
|
{ title: t("group"), align: "start", key: "name" },
|
||||||
|
{ title: t("progress"), align: "center", key: "progress" },
|
||||||
|
{ title: t("submission"), align: "center", key: "submission" },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const { mutate } = useDeleteAssignmentMutation();
|
||||||
|
|
||||||
|
async function deleteAssignment(num: number, clsId: string): Promise<void> {
|
||||||
|
mutate(
|
||||||
|
{ cid: clsId, an: num },
|
||||||
|
{
|
||||||
|
onSuccess: () => {
|
||||||
|
window.location.href = "/user/assignment";
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="container">
|
||||||
|
<using-query-result
|
||||||
|
:query-result="assignmentQueryResult"
|
||||||
|
v-slot="{ data }: { data: AssignmentResponse }"
|
||||||
|
>
|
||||||
|
<v-card
|
||||||
|
v-if="data"
|
||||||
|
class="assignment-card"
|
||||||
|
>
|
||||||
|
<div class="top-buttons">
|
||||||
|
<v-btn
|
||||||
|
icon
|
||||||
|
variant="text"
|
||||||
|
class="back-btn"
|
||||||
|
to="/user/assignment"
|
||||||
|
>
|
||||||
|
<v-icon>mdi-arrow-left</v-icon>
|
||||||
|
</v-btn>
|
||||||
|
|
||||||
|
<v-btn
|
||||||
|
icon
|
||||||
|
variant="text"
|
||||||
|
class="top-right-btn"
|
||||||
|
@click="deleteAssignment(data.assignment.id, data.assignment.within)"
|
||||||
|
>
|
||||||
|
<v-icon>mdi-delete</v-icon>
|
||||||
|
</v-btn>
|
||||||
|
</div>
|
||||||
|
<v-card-title class="text-h4 assignmentTopTitle">{{ data.assignment.title }}</v-card-title>
|
||||||
|
<v-card-subtitle class="subtitle-section">
|
||||||
|
<using-query-result
|
||||||
|
:query-result="lpQueryResult"
|
||||||
|
v-slot="{ data: lpData }"
|
||||||
|
>
|
||||||
|
<v-btn
|
||||||
|
v-if="lpData"
|
||||||
|
:to="`/learningPath/${lpData.hruid}/${language}/${lpData.startNode.learningobjectHruid}?assignmentNo=${assignmentId}&classId=${classId}`"
|
||||||
|
variant="tonal"
|
||||||
|
color="primary"
|
||||||
|
>
|
||||||
|
{{ t("learning-path") }}
|
||||||
|
</v-btn>
|
||||||
|
</using-query-result>
|
||||||
|
</v-card-subtitle>
|
||||||
|
|
||||||
|
<v-card-text class="description">
|
||||||
|
{{ data.assignment.description }}
|
||||||
|
</v-card-text>
|
||||||
|
|
||||||
|
<v-card-text class="group-section">
|
||||||
|
<h3>{{ t("groups") }}</h3>
|
||||||
|
<div class="table-scroll">
|
||||||
|
<v-data-table
|
||||||
|
:headers="headers"
|
||||||
|
:items="allGroups"
|
||||||
|
item-key="id"
|
||||||
|
class="elevation-1"
|
||||||
|
>
|
||||||
|
<template #[`item.name`]="{ item }">
|
||||||
|
<v-btn
|
||||||
|
@click="openGroupDetails(item)"
|
||||||
|
variant="text"
|
||||||
|
color="primary"
|
||||||
|
>
|
||||||
|
{{ item.name }}
|
||||||
|
</v-btn>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #[`item.progress`]="{ item }">
|
||||||
|
<v-progress-linear
|
||||||
|
:model-value="item.progress"
|
||||||
|
color="blue-grey"
|
||||||
|
height="25"
|
||||||
|
>
|
||||||
|
<template v-slot:default="{ value }">
|
||||||
|
<strong>{{ Math.ceil(value) }}%</strong>
|
||||||
|
</template>
|
||||||
|
</v-progress-linear>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #[`item.submission`]="{ item }">
|
||||||
|
<v-btn
|
||||||
|
:to="item.submitted ? `${props.assignmentId}/submissions/` : undefined"
|
||||||
|
:color="item.submitted ? 'green' : 'red'"
|
||||||
|
variant="text"
|
||||||
|
class="text-capitalize"
|
||||||
|
>
|
||||||
|
{{ item.submitted ? t("see-submission") : t("no-submission") }}
|
||||||
|
</v-btn>
|
||||||
|
</template>
|
||||||
|
</v-data-table>
|
||||||
|
</div>
|
||||||
|
</v-card-text>
|
||||||
|
|
||||||
|
<v-dialog
|
||||||
|
v-model="dialog"
|
||||||
|
max-width="50%"
|
||||||
|
>
|
||||||
|
<v-card>
|
||||||
|
<v-card-title class="headline">{{ t("members") }}</v-card-title>
|
||||||
|
<v-card-text>
|
||||||
|
<v-list>
|
||||||
|
<v-list-item
|
||||||
|
v-for="(member, index) in selectedGroup.members"
|
||||||
|
:key="index"
|
||||||
|
>
|
||||||
|
<v-list-item-content>
|
||||||
|
<v-list-item-title
|
||||||
|
>{{ member.firstName + " " + member.lastName }}
|
||||||
|
</v-list-item-title>
|
||||||
|
</v-list-item-content>
|
||||||
|
</v-list-item>
|
||||||
|
</v-list>
|
||||||
|
</v-card-text>
|
||||||
|
<v-card-actions>
|
||||||
|
<v-btn
|
||||||
|
color="primary"
|
||||||
|
@click="dialog = false"
|
||||||
|
>Close</v-btn
|
||||||
|
>
|
||||||
|
</v-card-actions>
|
||||||
|
</v-card>
|
||||||
|
</v-dialog>
|
||||||
|
<!--
|
||||||
|
<v-card-actions class="justify-end">
|
||||||
|
<v-btn
|
||||||
|
size="large"
|
||||||
|
color="success"
|
||||||
|
variant="text"
|
||||||
|
>
|
||||||
|
{{ t("view-submissions") }}
|
||||||
|
</v-btn>
|
||||||
|
</v-card-actions>
|
||||||
|
-->
|
||||||
|
</v-card>
|
||||||
|
</using-query-result>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
@import "@/assets/assignment.css";
|
||||||
|
|
||||||
|
.table-scroll {
|
||||||
|
overflow-x: auto;
|
||||||
|
-webkit-overflow-scrolling: touch;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -1,7 +0,0 @@
|
||||||
<script setup lang="ts"></script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<main></main>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped></style>
|
|
188
frontend/src/views/assignments/UserAssignments.vue
Normal file
188
frontend/src/views/assignments/UserAssignments.vue
Normal file
|
@ -0,0 +1,188 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, computed, onMounted, watch } from "vue";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
import { useRouter } from "vue-router";
|
||||||
|
import auth from "@/services/auth/auth-service.ts";
|
||||||
|
import { useTeacherClassesQuery } from "@/queries/teachers.ts";
|
||||||
|
import { useStudentClassesQuery } from "@/queries/students.ts";
|
||||||
|
import { ClassController } from "@/controllers/classes.ts";
|
||||||
|
import type { ClassDTO } from "@dwengo-1/common/interfaces/class";
|
||||||
|
import { asyncComputed } from "@vueuse/core";
|
||||||
|
import { useDeleteAssignmentMutation } from "@/queries/assignments.ts";
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const role = ref(auth.authState.activeRole);
|
||||||
|
const username = ref<string>("");
|
||||||
|
|
||||||
|
const isTeacher = computed(() => role.value === "teacher");
|
||||||
|
|
||||||
|
// Fetch and store all the teacher's classes
|
||||||
|
let classesQueryResults = undefined;
|
||||||
|
|
||||||
|
if (isTeacher.value) {
|
||||||
|
classesQueryResults = useTeacherClassesQuery(username, true);
|
||||||
|
} else {
|
||||||
|
classesQueryResults = useStudentClassesQuery(username, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO: remove later
|
||||||
|
const classController = new ClassController();
|
||||||
|
|
||||||
|
//TODO: replace by query that fetches all user's assignment
|
||||||
|
const assignments = asyncComputed(async () => {
|
||||||
|
const classes = classesQueryResults?.data?.value?.classes;
|
||||||
|
if (!classes) return [];
|
||||||
|
const result = await Promise.all(
|
||||||
|
(classes as ClassDTO[]).map(async (cls) => {
|
||||||
|
const { assignments } = await classController.getAssignments(cls.id);
|
||||||
|
return assignments.map((a) => ({
|
||||||
|
id: a.id,
|
||||||
|
class: cls,
|
||||||
|
title: a.title,
|
||||||
|
description: a.description,
|
||||||
|
learningPath: a.learningPath,
|
||||||
|
language: a.language,
|
||||||
|
groups: a.groups,
|
||||||
|
}));
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
return result.flat();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
async function goToCreateAssignment(): Promise<void> {
|
||||||
|
await router.push("/assignment/create");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function goToAssignmentDetails(id: number, clsId: string): Promise<void> {
|
||||||
|
await router.push(`/assignment/${clsId}/${id}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { mutate, data, isSuccess } = useDeleteAssignmentMutation();
|
||||||
|
|
||||||
|
watch([isSuccess, data], async ([success, oldData]) => {
|
||||||
|
if (success && oldData?.assignment) {
|
||||||
|
window.location.reload();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
async function goToDeleteAssignment(num: number, clsId: string): Promise<void> {
|
||||||
|
mutate({ cid: clsId, an: num });
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
const user = await auth.loadUser();
|
||||||
|
username.value = user?.profile?.preferred_username ?? "";
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="assignments-container">
|
||||||
|
<h1>{{ t("assignments") }}</h1>
|
||||||
|
|
||||||
|
<v-btn
|
||||||
|
v-if="isTeacher"
|
||||||
|
color="primary"
|
||||||
|
class="mb-4 center-btn"
|
||||||
|
@click="goToCreateAssignment"
|
||||||
|
>
|
||||||
|
{{ t("new-assignment") }}
|
||||||
|
</v-btn>
|
||||||
|
|
||||||
|
<v-container>
|
||||||
|
<v-row>
|
||||||
|
<v-col
|
||||||
|
v-for="assignment in assignments"
|
||||||
|
:key="assignment.id"
|
||||||
|
cols="12"
|
||||||
|
>
|
||||||
|
<v-card class="assignment-card">
|
||||||
|
<div class="top-content">
|
||||||
|
<div class="assignment-title">{{ assignment.title }}</div>
|
||||||
|
<div class="assignment-class">
|
||||||
|
{{ t("class") }}:
|
||||||
|
<span class="class-name">
|
||||||
|
{{ assignment.class.displayName }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="spacer"></div>
|
||||||
|
|
||||||
|
<div class="button-row">
|
||||||
|
<v-btn
|
||||||
|
color="primary"
|
||||||
|
variant="text"
|
||||||
|
@click="goToAssignmentDetails(assignment.id, assignment.class.id)"
|
||||||
|
>
|
||||||
|
{{ t("view-assignment") }}
|
||||||
|
</v-btn>
|
||||||
|
<v-btn
|
||||||
|
v-if="isTeacher"
|
||||||
|
color="red"
|
||||||
|
variant="text"
|
||||||
|
@click="goToDeleteAssignment(assignment.id, assignment.class.id)"
|
||||||
|
>
|
||||||
|
{{ t("delete") }}
|
||||||
|
</v-btn>
|
||||||
|
</div>
|
||||||
|
</v-card>
|
||||||
|
</v-col>
|
||||||
|
</v-row>
|
||||||
|
</v-container>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.assignments-container {
|
||||||
|
width: 100%;
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: 2% 4%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.center-btn {
|
||||||
|
display: block;
|
||||||
|
margin-left: auto;
|
||||||
|
margin-right: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.assignment-card {
|
||||||
|
padding: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-content {
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spacer {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.button-row {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: 0.5rem;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.assignment-title {
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: 1.5rem;
|
||||||
|
margin-bottom: 0.1rem;
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
.assignment-class {
|
||||||
|
color: #666;
|
||||||
|
font-size: 0.95rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.class-name {
|
||||||
|
font-weight: 500;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
</style>
|
|
@ -1,7 +0,0 @@
|
||||||
<script setup lang="ts"></script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<main></main>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<style scoped></style>
|
|
9424
package-lock.json
generated
9424
package-lock.json
generated
File diff suppressed because it is too large
Load diff
|
@ -41,5 +41,8 @@
|
||||||
"eslint-config-prettier": "^10.0.1",
|
"eslint-config-prettier": "^10.0.1",
|
||||||
"jiti": "^2.4.2",
|
"jiti": "^2.4.2",
|
||||||
"typescript-eslint": "^8.24.1"
|
"typescript-eslint": "^8.24.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"swagger": "^0.7.5"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue