From 5562d95741e1ab5be7d6e5212bab513279bb5fb0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Johannes=20St=C3=B6hr?= <38322605+JohannesStoehr@users.noreply.github.com> Date: Wed, 30 Oct 2024 22:38:09 +0100 Subject: [PATCH] Adaptive learning: Allow editors to edit the competency link weight (#9564) --- .../webapp/app/entities/competency.model.ts | 4 +++ .../competency-selection.component.html | 17 +++++++++++ .../competency-selection.component.ts | 29 ++++++++++++++++++- src/main/webapp/i18n/de/competency.json | 8 ++++- src/main/webapp/i18n/en/competency.json | 8 ++++- .../competency-selection.component.spec.ts | 17 +++++++++++ 6 files changed, 80 insertions(+), 3 deletions(-) diff --git a/src/main/webapp/app/entities/competency.model.ts b/src/main/webapp/app/entities/competency.model.ts index 245cfd71553c..51afaa1b18a3 100644 --- a/src/main/webapp/app/entities/competency.model.ts +++ b/src/main/webapp/app/entities/competency.model.ts @@ -46,7 +46,11 @@ export enum CourseCompetencyType { export const DEFAULT_MASTERY_THRESHOLD = 100; +export const HIGH_COMPETENCY_LINK_WEIGHT = 1; export const MEDIUM_COMPETENCY_LINK_WEIGHT = 0.5; +export const LOW_COMPETENCY_LINK_WEIGHT = 0.25; +export const LOW_COMPETENCY_LINK_WEIGHT_CUT_OFF = 0.375; // halfway between low and medium +export const MEDIUM_COMPETENCY_LINK_WEIGHT_CUT_OFF = 0.75; // halfway between medium and high export abstract class BaseCompetency implements BaseEntity { id?: number; diff --git a/src/main/webapp/app/shared/competency-selection/competency-selection.component.html b/src/main/webapp/app/shared/competency-selection/competency-selection.component.html index b15b8f4cfd8f..7fba47389226 100644 --- a/src/main/webapp/app/shared/competency-selection/competency-selection.component.html +++ b/src/main/webapp/app/shared/competency-selection/competency-selection.component.html @@ -32,6 +32,23 @@ {{ competencyLink.competency.title }} + @if (checkboxStates[competencyLink.competency.id]) { + + } } } diff --git a/src/main/webapp/app/shared/competency-selection/competency-selection.component.ts b/src/main/webapp/app/shared/competency-selection/competency-selection.component.ts index 224abeb3cabe..41f2dcf0e67a 100644 --- a/src/main/webapp/app/shared/competency-selection/competency-selection.component.ts +++ b/src/main/webapp/app/shared/competency-selection/competency-selection.component.ts @@ -1,7 +1,16 @@ import { ChangeDetectorRef, Component, EventEmitter, Input, OnInit, Output, forwardRef } from '@angular/core'; import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; import { faQuestionCircle } from '@fortawesome/free-solid-svg-icons'; -import { CompetencyLearningObjectLink, CourseCompetency, MEDIUM_COMPETENCY_LINK_WEIGHT, getIcon } from 'app/entities/competency.model'; +import { + CompetencyLearningObjectLink, + CourseCompetency, + HIGH_COMPETENCY_LINK_WEIGHT, + LOW_COMPETENCY_LINK_WEIGHT, + LOW_COMPETENCY_LINK_WEIGHT_CUT_OFF, + MEDIUM_COMPETENCY_LINK_WEIGHT, + MEDIUM_COMPETENCY_LINK_WEIGHT_CUT_OFF, + getIcon, +} from 'app/entities/competency.model'; import { ActivatedRoute } from '@angular/router'; import { CourseStorageService } from 'app/course/manage/course-storage.service'; import { finalize } from 'rxjs'; @@ -40,6 +49,12 @@ export class CompetencySelectionComponent implements OnInit, ControlValueAccesso // eslint-disable-next-line @typescript-eslint/no-unused-vars _onChange = (value: any) => {}; + protected readonly HIGH_COMPETENCY_LINK_WEIGHT = HIGH_COMPETENCY_LINK_WEIGHT; + protected readonly MEDIUM_COMPETENCY_LINK_WEIGHT = MEDIUM_COMPETENCY_LINK_WEIGHT; + protected readonly LOW_COMPETENCY_LINK_WEIGHT = LOW_COMPETENCY_LINK_WEIGHT; + protected readonly LOW_COMPETENCY_LINK_WEIGHT_CUT_OFF = LOW_COMPETENCY_LINK_WEIGHT_CUT_OFF; // halfway between low and medium + protected readonly MEDIUM_COMPETENCY_LINK_WEIGHT_CUT_OFF = MEDIUM_COMPETENCY_LINK_WEIGHT_CUT_OFF; // halfway between medium and high + constructor( private route: ActivatedRoute, private courseStorageService: CourseStorageService, @@ -122,7 +137,19 @@ export class CompetencySelectionComponent implements OnInit, ControlValueAccesso } } + updateLinkWeight(link: CompetencyLearningObjectLink, value: number) { + link.weight = value; + + this._onChange(this.selectedCompetencyLinks); + this.valueChange.emit(this.selectedCompetencyLinks); + } + writeValue(value?: CompetencyLearningObjectLink[]): void { + this.competencyLinks?.forEach((link) => { + const selectedLink = value?.find((value) => value.competency?.id === link.competency?.id); + link.weight = selectedLink?.weight ?? MEDIUM_COMPETENCY_LINK_WEIGHT; + }); + if (value && this.competencyLinks) { // Compare the ids of the competencies instead of the whole objects const ids = value.map((el) => el.competency?.id); diff --git a/src/main/webapp/i18n/de/competency.json b/src/main/webapp/i18n/de/competency.json index aaef36b9f34a..f06f02951cda 100644 --- a/src/main/webapp/i18n/de/competency.json +++ b/src/main/webapp/i18n/de/competency.json @@ -117,7 +117,13 @@ "link": { "title": "Verbundene Kompetenzen", "lectureUnit": "Verbinde diese Vorlesungseinheit wahlweise mit einer oder mehreren Kompetenzen.", - "exercise": "Verbinde diese Übung wahlweise mit einer oder mehreren Kompetenzen." + "exercise": "Verbinde diese Übung wahlweise mit einer oder mehreren Kompetenzen.", + "weightTooltip": "Die Relevanz dieses Lernobjektes für die Kompetenz.", + "weightLabels": { + "low": "Niedrig", + "medium": "Mittel", + "high": "Hoch" + } }, "competencyCard": { "connectedLectureUnits": "Verbundene Vorlesungseinheiten", diff --git a/src/main/webapp/i18n/en/competency.json b/src/main/webapp/i18n/en/competency.json index e4aedbe82c77..6cac8ddf0256 100644 --- a/src/main/webapp/i18n/en/competency.json +++ b/src/main/webapp/i18n/en/competency.json @@ -117,7 +117,13 @@ "link": { "title": "Linked Competencies", "lectureUnit": "Optionally link this lecture unit to one or more competencies.", - "exercise": "Optionally link this exercise to one or more competencies." + "exercise": "Optionally link this exercise to one or more competencies.", + "weightTooltip": "The relevance of this learning object for the competency.", + "weightLabels": { + "low": "Low", + "medium": "Medium", + "high": "High" + } }, "competencyCard": { "connectedLectureUnits": "Linked Lecture Units", diff --git a/src/test/javascript/spec/component/shared/competency-selection.component.spec.ts b/src/test/javascript/spec/component/shared/competency-selection.component.spec.ts index a58b665e45cc..13cdf686b9c9 100644 --- a/src/test/javascript/spec/component/shared/competency-selection.component.spec.ts +++ b/src/test/javascript/spec/component/shared/competency-selection.component.spec.ts @@ -14,6 +14,7 @@ import { By } from '@angular/platform-browser'; import { CourseStorageService } from 'app/course/manage/course-storage.service'; import { ChangeDetectorRef } from '@angular/core'; import { CourseCompetencyService } from 'app/course/competencies/course-competency.service'; +import { Prerequisite } from 'app/entities/prerequisite.model'; describe('CompetencySelection', () => { let fixture: ComponentFixture; @@ -122,6 +123,22 @@ describe('CompetencySelection', () => { expect(component.selectedCompetencyLinks?.first()?.competency?.title).toBe('test'); }); + it('should update link weight when value is written', () => { + jest.spyOn(courseStorageService, 'getCourse').mockReturnValue({ + competencies: [{ id: 1, title: 'test' } as Competency, { id: 2, title: 'testAgain' } as Prerequisite, { id: 3, title: 'testMore' } as Competency], + }); + + fixture.detectChanges(); + + component.writeValue([ + new CompetencyLearningObjectLink({ id: 1, title: 'other' } as Competency, 0.5), + new CompetencyLearningObjectLink({ id: 3, title: 'otherMore' } as Competency, 1), + ]); + expect(component.selectedCompetencyLinks).toBeArrayOfSize(2); + expect(component.selectedCompetencyLinks?.first()?.weight).toBe(0.5); + expect(component.selectedCompetencyLinks?.last()?.weight).toBe(1); + }); + it('should trigger change detection after loading competencies', () => { jest.spyOn(courseStorageService, 'getCourse').mockReturnValue({ competencies: undefined }); const changeDetector = fixture.debugElement.injector.get(ChangeDetectorRef);