diff --git a/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.html b/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.html index 458366c305d..b7e5dbb860b 100644 --- a/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.html +++ b/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.html @@ -13,8 +13,10 @@ (optionSelected)="itemSelected($event.option.value)" (closed)="closed($event)" > - - {{ workgroup.displayNames }} - + @for (workgroup of filteredWorkgroups | async; track workgroup.displayNames) { + + {{ workgroup.displayNames }} + + } diff --git a/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.ts b/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.ts index b6846a25509..ecf0478938d 100644 --- a/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.ts +++ b/src/app/classroom-monitor/workgroup-select/workgroup-select-autocomplete/workgroup-select-autocomplete.component.ts @@ -1,29 +1,33 @@ -'use strict'; - import { Component, ViewEncapsulation } from '@angular/core'; +import { FormControl, ReactiveFormsModule } from '@angular/forms'; +import { MatAutocompleteModule } from '@angular/material/autocomplete'; +import { MatFormFieldModule } from '@angular/material/form-field'; +import { MatInputModule } from '@angular/material/input'; +import { CommonModule } from '@angular/common'; import { WorkgroupSelectComponent } from '../workgroup-select.component'; -import { FormControl } from '@angular/forms'; import { Observable } from 'rxjs'; import { filter, map, startWith } from 'rxjs/operators'; -import { ConfigService } from '../../../../assets/wise5/services/configService'; -import { TeacherDataService } from '../../../../assets/wise5/services/teacherDataService'; import { copy } from '../../../../assets/wise5/common/object/object'; @Component({ + encapsulation: ViewEncapsulation.None, + imports: [ + CommonModule, + MatAutocompleteModule, + MatFormFieldModule, + MatInputModule, + ReactiveFormsModule + ], selector: 'workgroup-select-autocomplete', - styleUrls: ['workgroup-select-autocomplete.component.scss'], - templateUrl: 'workgroup-select-autocomplete.component.html', - encapsulation: ViewEncapsulation.None + standalone: true, + styleUrl: 'workgroup-select-autocomplete.component.scss', + templateUrl: 'workgroup-select-autocomplete.component.html' }) export class WorkgroupSelectAutocompleteComponent extends WorkgroupSelectComponent { - filteredWorkgroups: Observable; - myControl = new FormControl(); - - constructor(protected configService: ConfigService, protected dataService: TeacherDataService) { - super(configService, dataService); - } + protected filteredWorkgroups: Observable; + protected myControl = new FormControl(); - ngOnInit() { + ngOnInit(): void { super.ngOnInit(); this.updateFilteredWorkgroups(); const currentWorkgroup = this.dataService.getCurrentWorkgroup(); @@ -32,7 +36,7 @@ export class WorkgroupSelectAutocompleteComponent extends WorkgroupSelectCompone } } - private updateFilteredWorkgroups() { + private updateFilteredWorkgroups(): void { this.filteredWorkgroups = this.myControl.valueChanges.pipe( startWith(''), filter((value) => typeof value === 'string'), @@ -40,7 +44,7 @@ export class WorkgroupSelectAutocompleteComponent extends WorkgroupSelectCompone ); } - displayWith(workgroup) { + protected displayWith(workgroup: any): string { return workgroup.displayNames; } @@ -50,11 +54,11 @@ export class WorkgroupSelectAutocompleteComponent extends WorkgroupSelectCompone ); } - currentPeriodChanged() { + protected currentPeriodChanged(): void { this.myControl.setValue(''); } - setWorkgroups() { + protected setWorkgroups(): void { this.filterWorkgroupsBySelectedPeriod(); const students = this.getStudentsFromWorkgroups(); this.workgroups = this.canViewStudentNames @@ -67,7 +71,7 @@ export class WorkgroupSelectAutocompleteComponent extends WorkgroupSelectCompone this.updateWorkgroupDisplay(workgroup); } - getStudentsFromWorkgroups() { + private getStudentsFromWorkgroups(): any[] { const students = []; for (const workgroup of this.workgroups) { const ids = workgroup.userIds; @@ -86,25 +90,21 @@ export class WorkgroupSelectAutocompleteComponent extends WorkgroupSelectCompone return students; } - flipName(name: string) { + private flipName(name: string): string { const names = name.split(' '); return `${names[1]}, ${names[0]}`; } - itemSelected(workgroup: any) { + protected itemSelected(workgroup: any): void { this.setCurrentWorkgroup(workgroup); this.updateWorkgroupDisplay(workgroup); } private updateWorkgroupDisplay(workgroup: any): void { - if (workgroup) { - this.myControl.setValue(workgroup.displayNames); - } else { - this.myControl.setValue(''); - } + this.myControl.setValue(workgroup ? workgroup.displayNames : ''); } - closed(event: any) { + protected closed(event: any): void { if (this.myControl.value === '') { this.itemSelected(null); } diff --git a/src/app/classroom-monitor/workgroup-select/workgroup-select.component.ts b/src/app/classroom-monitor/workgroup-select/workgroup-select.component.ts index 193d7691155..ced2f7f3a59 100644 --- a/src/app/classroom-monitor/workgroup-select/workgroup-select.component.ts +++ b/src/app/classroom-monitor/workgroup-select/workgroup-select.component.ts @@ -1,5 +1,3 @@ -'use strict'; - import { Directive, Input } from '@angular/core'; import { Subscription } from 'rxjs'; import { ConfigService } from '../../../assets/wise5/services/configService'; @@ -8,15 +6,18 @@ import { TeacherDataService } from '../../../assets/wise5/services/teacherDataSe @Directive({ selector: 'workgroup-select' }) export class WorkgroupSelectComponent { @Input() customClass: string; - canViewStudentNames: boolean; - periodId: number; - selectedItem: any; - subscriptions: Subscription = new Subscription(); - workgroups: any; + protected canViewStudentNames: boolean; + protected periodId: number; + protected selectedItem: any; + protected subscriptions: Subscription = new Subscription(); + protected workgroups: any; - constructor(protected configService: ConfigService, protected dataService: TeacherDataService) {} + constructor( + protected configService: ConfigService, + protected dataService: TeacherDataService + ) {} - ngOnInit() { + ngOnInit(): void { this.canViewStudentNames = this.configService.getPermissions().canViewStudentNames; this.periodId = this.dataService.getCurrentPeriod().periodId; this.setWorkgroups(); @@ -37,29 +38,29 @@ export class WorkgroupSelectComponent { ); } - ngOnDestroy() { + ngOnDestroy(): void { this.subscriptions.unsubscribe(); } - setWorkgroups() {} + protected setWorkgroups() {} protected setWorkgroup(workgroup: any): void {} - currentPeriodChanged() {} + protected currentPeriodChanged() {} - sortByField(arr: any[], field: string): any[] { + protected sortByField(arr: any[], field: string): any[] { return arr.sort((workgroup1, workgroup2) => { return workgroup1[field] - workgroup2[field]; }); } - sortByDisplayNames(arr: any[]): any[] { + protected sortByDisplayNames(arr: any[]): any[] { return arr.sort((workgroup1, workgroup2) => { return workgroup1.displayNames.localeCompare(workgroup2.displayNames); }); } - filterWorkgroupsBySelectedPeriod() { + protected filterWorkgroupsBySelectedPeriod(): void { this.workgroups = this.configService.getClassmateUserInfos().filter((workgroup) => { return ( (this.periodId === -1 || workgroup.periodId === this.periodId) && @@ -68,7 +69,7 @@ export class WorkgroupSelectComponent { }); } - setCurrentWorkgroup(workgroup) { + protected setCurrentWorkgroup(workgroup): void { this.dataService.setCurrentWorkgroup(workgroup); } } diff --git a/src/app/student-teacher-common.module.ts b/src/app/student-teacher-common.module.ts index 4e20d967cdf..a40d3e482b4 100644 --- a/src/app/student-teacher-common.module.ts +++ b/src/app/student-teacher-common.module.ts @@ -39,6 +39,7 @@ import { MathModule } from './math/math.module'; import { MatMenuModule } from '@angular/material/menu'; import { MainMenuComponent } from '../assets/wise5/common/main-menu/main-menu.component'; import { SideMenuComponent } from '../assets/wise5/common/side-menu/side-menu.component'; +import { ScrollingModule } from '@angular/cdk/scrolling'; @NgModule({ declarations: [ @@ -82,6 +83,7 @@ import { SideMenuComponent } from '../assets/wise5/common/side-menu/side-menu.co NodeStatusIconComponent, NotebookModule, ReactiveFormsModule, + ScrollingModule, StudentTeacherCommonServicesModule ], exports: [ diff --git a/src/app/teacher/authoring-routing.module.ts b/src/app/teacher/authoring-routing.module.ts index 01ad5a06775..9390f4f4e99 100644 --- a/src/app/teacher/authoring-routing.module.ts +++ b/src/app/teacher/authoring-routing.module.ts @@ -9,7 +9,6 @@ import { ProjectAuthoringComponent } from '../../assets/wise5/authoringTool/proj import { NodeAuthoringComponent } from '../../assets/wise5/authoringTool/node/node-authoring/node-authoring.component'; import { NodeAdvancedAuthoringComponent } from '../../assets/wise5/authoringTool/node/advanced/node-advanced-authoring/node-advanced-authoring.component'; import { NodeAdvancedConstraintAuthoringComponent } from '../../assets/wise5/authoringTool/node/advanced/constraint/node-advanced-constraint-authoring.component'; -import { ChooseComponentLocationComponent } from '../../assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component'; import { AddLessonConfigureComponent } from '../../assets/wise5/authoringTool/addLesson/add-lesson-configure/add-lesson-configure.component'; import { ChooseNewNodeTemplateComponent } from '../../assets/wise5/authoringTool/addNode/choose-new-node-template/choose-new-node-template.component'; import { AddYourOwnNodeComponent } from '../../assets/wise5/authoringTool/addNode/add-your-own-node/add-your-own-node.component'; @@ -153,10 +152,6 @@ const routes: Routes = [ { path: 'rubric', component: EditNodeRubricComponent } ] }, - { - path: 'choose-component-location', - component: ChooseComponentLocationComponent - }, { path: 'import-component', children: [{ path: 'choose-component', component: ChooseImportComponentComponent }] diff --git a/src/app/teacher/authoring-tool.module.ts b/src/app/teacher/authoring-tool.module.ts index 495cc9c3a2b..3095e1714f5 100644 --- a/src/app/teacher/authoring-tool.module.ts +++ b/src/app/teacher/authoring-tool.module.ts @@ -22,7 +22,6 @@ import { WiseTinymceEditorModule } from '../../assets/wise5/directives/wise-tiny import { NotebookAuthoringComponent } from '../../assets/wise5/authoringTool/notebook-authoring/notebook-authoring.component'; import { StructureAuthoringModule } from '../../assets/wise5/authoringTool/structure/structure-authoring.module'; import { MilestonesAuthoringComponent } from '../../assets/wise5/authoringTool/milestones-authoring/milestones-authoring.component'; -import { ChooseComponentLocationComponent } from '../../assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component'; import { TopBarComponent } from '../../assets/wise5/authoringTool/components/top-bar/top-bar.component'; import { ProjectAssetAuthoringModule } from '../../assets/wise5/authoringTool/project-asset-authoring/project-asset-authoring.module'; import { ChooseSimulationComponent } from '../../assets/wise5/authoringTool/addNode/choose-simulation/choose-simulation.component'; @@ -57,7 +56,6 @@ import { TranslatableTextareaComponent } from '../../assets/wise5/authoringTool/ import { TranslatableRichTextEditorComponent } from '../../assets/wise5/authoringTool/components/translatable-rich-text-editor/translatable-rich-text-editor.component'; import { AddStepButtonComponent } from '../../assets/wise5/authoringTool/add-step-button/add-step-button.component'; import { CreateBranchComponent } from '../../assets/wise5/authoringTool/create-branch/create-branch.component'; -import { PreviewComponentButtonComponent } from '../../assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component'; import { EditBranchComponent } from '../../assets/wise5/authoringTool/edit-branch/edit-branch.component'; import { ComponentTypeButtonComponent } from '../../assets/wise5/authoringTool/components/component-type-button/component-type-button.component'; import { MatExpansionModule } from '@angular/material/expansion'; @@ -92,7 +90,6 @@ import { MatExpansionModule } from '@angular/material/expansion'; AddYourOwnNodeComponent, AuthoringToolBarComponent, ChooseAutomatedAssessmentComponent, - ChooseComponentLocationComponent, ChooseCopyNodeLocationComponent, ChooseImportStepComponent, ChooseImportUnitComponent, @@ -116,7 +113,6 @@ import { MatExpansionModule } from '@angular/material/expansion'; NodeAdvancedAuthoringModule, NodeIconAndTitleComponent, NodeWithMoveAfterButtonComponent, - PreviewComponentButtonComponent, ProjectAssetAuthoringModule, ProjectListComponent, RouterModule, diff --git a/src/app/teacher/classroom-monitor.module.ts b/src/app/teacher/classroom-monitor.module.ts index d166238d9d8..5d4e5bf2d52 100644 --- a/src/app/teacher/classroom-monitor.module.ts +++ b/src/app/teacher/classroom-monitor.module.ts @@ -1,6 +1,5 @@ import { NgModule } from '@angular/core'; import { NavItemScoreComponent } from '../../assets/wise5/classroomMonitor/classroomMonitorComponents/nodeProgress/navItemScore/nav-item-score.component'; -import { ViewComponentRevisionsComponent } from '../../assets/wise5/classroomMonitor/classroomMonitorComponents/view-component-revisions/view-component-revisions.component'; import { AlertStatusCornerComponent } from '../classroom-monitor/alert-status-corner/alert-status-corner.component'; import { ComponentNewWorkBadgeComponent } from '../classroom-monitor/component-new-work-badge/component-new-work-badge.component'; import { ComponentSelectComponent } from '../classroom-monitor/component-select/component-select.component'; @@ -33,9 +32,10 @@ import { DataExportModule } from '../../assets/wise5/classroomMonitor/dataExport import { RouterModule } from '@angular/router'; import { SaveIndicatorComponent } from '../../assets/wise5/common/save-indicator/save-indicator.component'; import { PreviewComponentComponent } from '../../assets/wise5/authoringTool/components/preview-component/preview-component.component'; -import { StepToolsComponent } from '../../assets/wise5/common/stepTools/step-tools.component'; import { ComponentGradingComponent } from '../../assets/wise5/classroomMonitor/classroomMonitorComponents/component-grading.component'; import { SelectPeriodComponent } from '../../assets/wise5/classroomMonitor/classroomMonitorComponents/select-period/select-period.component'; +import { GradingStepToolsComponent } from '../../assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component'; +import { GradingNodeService } from '../../assets/wise5/services/gradingNodeService'; @NgModule({ declarations: [ @@ -52,8 +52,7 @@ import { SelectPeriodComponent } from '../../assets/wise5/classroomMonitor/class StudentGradingToolsComponent, StudentProgressComponent, ToolBarComponent, - TopBarComponent, - ViewComponentRevisionsComponent + TopBarComponent ], imports: [ ComponentGradingComponent, @@ -62,6 +61,7 @@ import { SelectPeriodComponent } from '../../assets/wise5/classroomMonitor/class ComponentStudentModule, DataExportModule, GradingCommonModule, + GradingStepToolsComponent, HighchartsChartModule, ManageStudentsModule, MilestoneModule, @@ -75,9 +75,9 @@ import { SelectPeriodComponent } from '../../assets/wise5/classroomMonitor/class SelectPeriodComponent, ShowNodeInfoDialogComponent, StepInfoComponent, - StepToolsComponent, StudentTeacherCommonModule, TeacherSummaryDisplayComponent - ] + ], + providers: [GradingNodeService] }) export class ClassroomMonitorModule {} diff --git a/src/app/teacher/grading-common.module.ts b/src/app/teacher/grading-common.module.ts index f3cafbd0708..5441aee41b7 100644 --- a/src/app/teacher/grading-common.module.ts +++ b/src/app/teacher/grading-common.module.ts @@ -23,11 +23,12 @@ import { ComponentStateInfoComponent } from '../../assets/wise5/common/component StatusIconComponent, StudentTeacherCommonModule, WorkgroupInfoComponent, + WorkgroupItemComponent, WorkgroupComponentGradingComponent, WorkgroupNodeScoreComponent, - WorkgroupNodeStatusComponent + WorkgroupNodeStatusComponent, + WorkgroupSelectAutocompleteComponent ], - declarations: [WorkgroupItemComponent, WorkgroupSelectAutocompleteComponent], exports: [ ComponentGradingComponent, ComponentStateInfoComponent, diff --git a/src/assets/wise5/authoringTool/authoring-tool.component.html b/src/assets/wise5/authoringTool/authoring-tool.component.html index 89fadb859ce..8ba0b3ba39d 100644 --- a/src/assets/wise5/authoringTool/authoring-tool.component.html +++ b/src/assets/wise5/authoringTool/authoring-tool.component.html @@ -23,6 +23,7 @@ class="l-main" [ngClass]="{ 'l-main--with-toolbar': showToolbar }" role="main" + cdkScrollable > diff --git a/src/assets/wise5/authoringTool/components/component-authoring.component.ts b/src/assets/wise5/authoringTool/components/component-authoring.component.ts index 322b7f10a3b..b9c150747d8 100644 --- a/src/assets/wise5/authoringTool/components/component-authoring.component.ts +++ b/src/assets/wise5/authoringTool/components/component-authoring.component.ts @@ -1,45 +1,65 @@ -import { - ApplicationRef, - Component, - ComponentRef, - ElementRef, - EnvironmentInjector, - Input, - ViewChild, - createComponent -} from '@angular/core'; +import { Component, EventEmitter, Input, Output } from '@angular/core'; import { ComponentContent } from '../../common/ComponentContent'; -import { components } from '../../components/Components'; +import { PreviewComponentComponent } from './preview-component/preview-component.component'; +import { EditComponentComponent } from './edit-component/edit-component.component'; +import { ComponentFactory } from '../../common/ComponentFactory'; +import { Component as WISEComponent } from '../../common/Component'; +import { TeacherProjectService } from '../../services/teacherProjectService'; +import { MatTooltipModule } from '@angular/material/tooltip'; @Component({ + imports: [PreviewComponentComponent, EditComponentComponent, MatTooltipModule], selector: 'component-authoring', standalone: true, - template: '
' + styles: [ + ` + preview-component { + display: block; + position: relative; + cursor: pointer; + } + preview-component:hover { + outline: 3px dashed #aaaaaa; + outline-offset: 8px; + } + preview-component:after { + content: ''; + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + } + ` + ], + template: `@if (editing) { + + } @else { + + }` }) export class ComponentAuthoringComponent { - @Input() private componentContent: ComponentContent; - @ViewChild('component') private componentElementRef: ElementRef; - private componentRef: ComponentRef; - @Input() private nodeId: string; + protected component: WISEComponent; + @Input() componentContent: ComponentContent; + @Input() editing: boolean; + @Output() editComponentEvent: EventEmitter = new EventEmitter(); + @Input() nodeId: string; - constructor( - private applicationRef: ApplicationRef, - private injector: EnvironmentInjector - ) {} + constructor(private projectService: TeacherProjectService) {} - ngAfterViewInit(): void { - this.componentRef = createComponent(components[this.componentContent.type].authoring, { - hostElement: this.componentElementRef.nativeElement, - environmentInjector: this.injector - }); - Object.assign(this.componentRef.instance, { - componentContent: this.componentContent, - nodeId: this.nodeId - }); - this.applicationRef.attachView(this.componentRef.hostView); - } - - ngOnDestroy(): void { - this.componentRef.destroy(); + ngOnChanges(): void { + this.component = new ComponentFactory().getComponent( + this.projectService.injectAssetPaths(this.componentContent), + this.nodeId + ); } } diff --git a/src/assets/wise5/authoringTool/components/edit-component/edit-component.component.spec.ts b/src/assets/wise5/authoringTool/components/edit-component/edit-component.component.spec.ts new file mode 100644 index 00000000000..ce7bc1a4c36 --- /dev/null +++ b/src/assets/wise5/authoringTool/components/edit-component/edit-component.component.spec.ts @@ -0,0 +1,75 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { EditComponentComponent } from './edit-component.component'; +import { Component } from '@angular/core'; +import { components } from '../../../components/Components'; +import { ComponentContent } from '../../../common/ComponentContent'; + +@Component({ + selector: 'mock-authoring', + template: '
Mock Authoring Component
' +}) +class MockAuthoringComponent { + componentContent: any; + nodeId: string; +} + +describe('EditComponentComponent', () => { + let component: EditComponentComponent; + let fixture: ComponentFixture; + const mockComponentContent = { + type: 'mockComponent' + }; + const mockNodeId = 'node1'; + + beforeEach(async () => { + components['mockComponent'] = { + authoring: MockAuthoringComponent + }; + + await TestBed.configureTestingModule({ + imports: [EditComponentComponent] + }).compileComponents(); + + fixture = TestBed.createComponent(EditComponentComponent); + component = fixture.componentInstance; + component.componentContent = mockComponentContent as ComponentContent; + component.nodeId = mockNodeId; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); + + it('should create the dynamic component after view init', () => { + component.ngAfterViewInit(); + fixture.detectChanges(); + const componentElement = fixture.nativeElement.querySelector('div'); + expect(componentElement.textContent).toContain('Mock Authoring Component'); + }); + + it('should pass inputs to the dynamic component', () => { + component.ngAfterViewInit(); + fixture.detectChanges(); + const componentInstance = (component as any).componentRef.instance; + expect(componentInstance.componentContent).toBe(mockComponentContent); + expect(componentInstance.nodeId).toBe(mockNodeId); + }); + + it('should destroy the component reference on destroy', () => { + component.ngAfterViewInit(); + const destroySpy = spyOn((component as any).componentRef, 'destroy'); + component.ngOnDestroy(); + expect(destroySpy).toHaveBeenCalled(); + }); + + it('should focus the host element after timeout', (done) => { + component.ngAfterViewInit(); + fixture.detectChanges(); + setTimeout(() => { + const hostElement = fixture.nativeElement.querySelector('div'); + expect(document.activeElement).toBe(hostElement); + done(); + }, 0); + }); +}); diff --git a/src/assets/wise5/authoringTool/components/edit-component/edit-component.component.ts b/src/assets/wise5/authoringTool/components/edit-component/edit-component.component.ts new file mode 100644 index 00000000000..f17d572ce15 --- /dev/null +++ b/src/assets/wise5/authoringTool/components/edit-component/edit-component.component.ts @@ -0,0 +1,47 @@ +import { + ApplicationRef, + Component, + ComponentRef, + ElementRef, + EnvironmentInjector, + Input, + ViewChild, + createComponent +} from '@angular/core'; +import { ComponentContent } from '../../../common/ComponentContent'; +import { components } from '../../../components/Components'; + +@Component({ + selector: 'edit-component', + standalone: true, + template: '
' +}) +export class EditComponentComponent { + @Input() componentContent: ComponentContent; + @ViewChild('component') private componentElementRef: ElementRef; + private componentRef: ComponentRef; + @Input() nodeId: string; + + constructor( + private applicationRef: ApplicationRef, + private injector: EnvironmentInjector + ) {} + + ngAfterViewInit(): void { + const hostElement = this.componentElementRef.nativeElement; + this.componentRef = createComponent(components[this.componentContent.type].authoring, { + hostElement: hostElement, + environmentInjector: this.injector + }); + Object.assign(this.componentRef.instance, { + componentContent: this.componentContent, + nodeId: this.nodeId + }); + this.applicationRef.attachView(this.componentRef.hostView); + setTimeout(() => hostElement.focus()); + } + + ngOnDestroy(): void { + this.componentRef.destroy(); + } +} diff --git a/src/assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component.html b/src/assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component.html deleted file mode 100644 index f454bdaac92..00000000000 --- a/src/assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component.html +++ /dev/null @@ -1,10 +0,0 @@ - diff --git a/src/assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component.ts b/src/assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component.ts deleted file mode 100644 index 7e448c0111b..00000000000 --- a/src/assets/wise5/authoringTool/components/preview-component-button/preview-component-button.component.ts +++ /dev/null @@ -1,31 +0,0 @@ -import { Component, Input } from '@angular/core'; -import { MatDialog } from '@angular/material/dialog'; -import { PreviewComponentDialogComponent } from '../preview-component-dialog/preview-component-dialog.component'; -import { ProjectService } from '../../../services/projectService'; -import { ComponentFactory } from '../../../common/ComponentFactory'; -import { MatIconModule } from '@angular/material/icon'; -import { MatTooltipModule } from '@angular/material/tooltip'; -import { MatButtonModule } from '@angular/material/button'; - -@Component({ - imports: [MatButtonModule, MatIconModule, MatTooltipModule], - selector: 'preview-component-button', - standalone: true, - templateUrl: 'preview-component-button.component.html' -}) -export class PreviewComponentButtonComponent { - @Input() componentId: string; - @Input() nodeId: string; - - constructor(private dialog: MatDialog, private projectService: ProjectService) {} - - protected popUpComponentPreview(): void { - const content = this.projectService.injectAssetPaths( - this.projectService.getComponent(this.nodeId, this.componentId) - ); - this.dialog.open(PreviewComponentDialogComponent, { - data: new ComponentFactory().getComponent(content, this.nodeId), - panelClass: 'dialog-lg' - }); - } -} diff --git a/src/assets/wise5/authoringTool/components/preview-component/preview-component.component.ts b/src/assets/wise5/authoringTool/components/preview-component/preview-component.component.ts index afa07216356..d81d120228c 100644 --- a/src/assets/wise5/authoringTool/components/preview-component/preview-component.component.ts +++ b/src/assets/wise5/authoringTool/components/preview-component/preview-component.component.ts @@ -16,16 +16,20 @@ import { components } from '../../../components/Components'; @Component({ selector: 'preview-component', standalone: true, - template: '
' + template: '
' }) export class PreviewComponentComponent { - @Input() protected component: WISEComponent; + @Input() component: WISEComponent; @ViewChild('component') private componentElementRef: ElementRef; private componentRef: ComponentRef; - @Input() protected periodId: number; - @Output() private starterStateChangedEvent: EventEmitter = new EventEmitter(); + @Input() disabled: boolean; + @Input() periodId: number; + @Output() starterStateChangedEvent: EventEmitter = new EventEmitter(); - constructor(private applicationRef: ApplicationRef, private injector: EnvironmentInjector) {} + constructor( + private applicationRef: ApplicationRef, + private injector: EnvironmentInjector + ) {} ngAfterViewInit(): void { this.renderComponent(); @@ -50,6 +54,7 @@ export class PreviewComponentComponent { component: this.component, mode: 'preview', periodId: this.periodId, + isDisabled: this.disabled, starterStateChangedEvent: this.starterStateChangedEvent }); this.applicationRef.attachView(this.componentRef.hostView); diff --git a/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.html b/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.html index 94957cfaa38..d371844cf07 100644 --- a/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.html +++ b/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.html @@ -1,3 +1,5 @@ -
- {{ message }} -
+@if (message) { +
+ {{ message }} +
+} diff --git a/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.spec.ts b/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.spec.ts index 6e3ae833297..189331916b3 100644 --- a/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.spec.ts +++ b/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.spec.ts @@ -44,9 +44,6 @@ describe('ConcurrentAuthorsMessageComponent', () => { function ngOnInit() { describe('ngOnInit()', () => { - it('set empty message when there are no other authors', () => { - expectMessage('["aa"]', ''); - }); it('set message to author when there is one other author', () => { expectMessage( '["aa","bb"]', diff --git a/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.ts b/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.ts index 413305c4097..bf3eeeed675 100644 --- a/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.ts +++ b/src/assets/wise5/authoringTool/concurrent-authors-message/concurrent-authors-message.component.ts @@ -8,7 +8,8 @@ import { NotifyAuthorService } from '../../services/notifyAuthorService'; @Component({ selector: 'concurrent-authors-message', - templateUrl: 'concurrent-authors-message.component.html' + templateUrl: 'concurrent-authors-message.component.html', + styles: ['.concurrent-authors { padding: 4px; }'] }) export class ConcurrentAuthorsMessageComponent { protected message: string = ''; diff --git a/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.html b/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.html index 727c499b322..b21aa47227b 100644 --- a/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.html +++ b/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.html @@ -1,10 +1,31 @@ - +@if (firstComponent) { + + + + + +} @else { + +} diff --git a/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.spec.ts b/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.spec.ts index 7545cab1030..feefe11c39c 100644 --- a/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.spec.ts +++ b/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.spec.ts @@ -33,7 +33,8 @@ describe('AddComponentButtonComponent', () => { }); fixture = TestBed.createComponent(AddComponentButtonComponent); component = fixture.componentInstance; - component.node = { id: 'node1' } as Node; + component.node = new Node(); + component.node.id = 'node1'; loader = TestbedHarnessEnvironment.loader(fixture); fixture.detectChanges(); }); diff --git a/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.ts b/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.ts index 2f00c1ff18f..0ba5b9c0ab9 100644 --- a/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.ts +++ b/src/assets/wise5/authoringTool/node/add-component-button/add-component-button.component.ts @@ -9,17 +9,31 @@ import { Node } from '../../../common/Node'; import { MatButtonModule } from '@angular/material/button'; import { MatIconModule } from '@angular/material/icon'; import { MatTooltipModule } from '@angular/material/tooltip'; +import { MatMenuModule } from '@angular/material/menu'; +import { CommonModule } from '@angular/common'; @Component({ - imports: [MatButtonModule, MatIconModule, MatTooltipModule], + imports: [CommonModule, MatButtonModule, MatIconModule, MatMenuModule, MatTooltipModule], selector: 'add-component-button', standalone: true, + styles: [ + ` + .rotate-180 { + transform: rotate(180deg); + } + .flip-vertical { + transform: scaleY(-1); + } + ` + ], templateUrl: './add-component-button.component.html' }) export class AddComponentButtonComponent { + protected firstComponent = false; @Input() insertAfterComponentId: string = null; - @Input() node: Node; @Output() newComponentsEvent: EventEmitter = new EventEmitter(); + @Input() node: Node; + protected tooltipText = $localize`Add component`; constructor( private createComponentService: CreateComponentService, @@ -29,10 +43,21 @@ export class AddComponentButtonComponent { private router: Router ) {} - protected addComponent(): void { + ngOnInit(): void { + this.updateUI(); + } + + private updateUI(): void { + this.firstComponent = this.node.getComponentPosition(this.insertAfterComponentId) === 0; + if (this.node.components.length > 0 && !this.firstComponent) { + this.tooltipText = $localize`Add component after`; + } + } + + protected addComponent(afterComponent = this.insertAfterComponentId): void { this.dialog .open(ChooseNewComponent, { - data: this.insertAfterComponentId, + data: afterComponent, width: '80%' }) .afterClosed() @@ -42,17 +67,18 @@ export class AddComponentButtonComponent { this.router.navigate(['import-component/choose-component'], { relativeTo: this.route, state: { - insertAfterComponentId: this.insertAfterComponentId + insertAfterComponentId: afterComponent } }); } else { const component = this.createComponentService.create( this.node.id, componentType, - this.insertAfterComponentId + afterComponent ); this.projectService.saveProject(); this.newComponentsEvent.emit([component]); + this.updateUI(); } }); } diff --git a/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.html b/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.html deleted file mode 100644 index bb73cd85d0b..00000000000 --- a/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.html +++ /dev/null @@ -1,51 +0,0 @@ -
-
-
Choose the new location by clicking one of the buttons below
-
-
-
- - Insert As First Component -
-@for (component of components; track component.id; let componentIndex = $index) { -
-
-
- {{ componentIndex + 1 }}. {{ getComponentTypeLabel(component.type) }} -
-
-
- -
-
-} - -
-
- -
diff --git a/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.spec.ts b/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.spec.ts deleted file mode 100644 index 10c9e8d6969..00000000000 --- a/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.spec.ts +++ /dev/null @@ -1,57 +0,0 @@ -import { TestBed } from '@angular/core/testing'; -import { ChooseComponentLocationComponent } from './choose-component-location.component'; -import { ComponentTypeService } from '../../../services/componentTypeService'; -import { ConfigService } from '../../../services/configService'; -import { TeacherDataService } from '../../../services/teacherDataService'; -import { TeacherProjectService } from '../../../services/teacherProjectService'; -import { CopyTranslationsService } from '../../../services/copyTranslationsService'; -import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'; - -const nodeId = 'node1'; -const components = [ - { id: 'comp1', type: 'OpenResponse' }, - { id: 'comp2', type: 'MultipleChoice' } -]; - -class MockProjectService { - createComponent(componentType) { - return { id: 'comp3', type: componentType }; - } - - getComponents() { - return components; - } - - saveProject() { - return new Promise(() => {}); - } -} - -class MockTeacherDataService { - getCurrentNodeId() { - return nodeId; - } -} - -class MockComponentTypeService {} -let component: ChooseComponentLocationComponent; -describe('ChooseComponentLocationComponent', () => { - beforeEach(() => { - TestBed.configureTestingModule({ - providers: [ - ChooseComponentLocationComponent, - { provide: ComponentTypeService, useClass: MockComponentTypeService }, - ConfigService, - CopyTranslationsService, - { provide: TeacherProjectService, useClass: MockProjectService }, - { provide: TeacherDataService, useClass: MockTeacherDataService }, - provideHttpClient(withInterceptorsFromDi()) - ] - }); - component = TestBed.inject(ChooseComponentLocationComponent); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.ts b/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.ts deleted file mode 100644 index 8972788cdcb..00000000000 --- a/src/assets/wise5/authoringTool/node/chooseComponentLocation/choose-component-location.component.ts +++ /dev/null @@ -1,94 +0,0 @@ -import { Component } from '@angular/core'; -import { CommonModule } from '@angular/common'; -import { RouterModule } from '@angular/router'; -import { MatButtonModule } from '@angular/material/button'; -import { MatIconModule } from '@angular/material/icon'; -import { MatTooltipModule } from '@angular/material/tooltip'; -import { ComponentTypeService } from '../../../services/componentTypeService'; -import { ConfigService } from '../../../services/configService'; -import { TeacherDataService } from '../../../services/teacherDataService'; -import { TeacherProjectService } from '../../../services/teacherProjectService'; -import { Node } from '../../../common/Node'; -import { Router } from '@angular/router'; -import { CopyTranslationsService } from '../../../services/copyTranslationsService'; -import { FlexLayoutModule } from '@angular/flex-layout'; - -@Component({ - standalone: true, - imports: [ - CommonModule, - FlexLayoutModule, - MatButtonModule, - MatIconModule, - MatTooltipModule, - RouterModule - ], - templateUrl: 'choose-component-location.component.html' -}) -export class ChooseComponentLocationComponent { - private action: 'copy' | 'move'; - protected components: any; - private node: Node; - private selectedComponents: any[]; - - constructor( - private componentTypeService: ComponentTypeService, - private copyTranslationsService: CopyTranslationsService, - private configService: ConfigService, - private dataService: TeacherDataService, - private projectService: TeacherProjectService, - private router: Router - ) {} - - ngOnInit(): void { - this.action = history.state.action; - const nodeId = this.dataService.getCurrentNodeId(); - this.node = this.projectService.getNode(nodeId); - this.components = this.projectService.getComponents(nodeId); - this.selectedComponents = history.state.selectedComponents; - } - - protected getComponentTypeLabel(componentType: any): string { - return this.componentTypeService.getComponentTypeLabel(componentType); - } - - protected insertComponentAfter(insertAfterComponentId: string = null): void { - const updatedComponents = - this.action === 'copy' - ? this.copyComponents(insertAfterComponentId) - : this.moveComponents(insertAfterComponentId); - this.projectService.saveProject().then(() => { - this.goToNodeAuthoring(updatedComponents); - }); - } - - private copyComponents(insertAfterComponentId: string = null): any[] { - const newComponents = this.node.copyComponents( - this.selectedComponents.map((c) => c.id), - insertAfterComponentId - ); - this.copyTranslationsService.tryCopyComponents(this.node, newComponents); - return newComponents; - } - - private moveComponents(insertAfterComponentId: string = null): any[] { - this.node.moveComponents( - this.selectedComponents.map((c) => c.id), - insertAfterComponentId - ); - return this.selectedComponents; - } - - protected goToNodeAuthoring(components: any[] = []): void { - this.router.navigate( - ['/teacher/edit/unit', this.configService.getProjectId(), 'node', this.node.id], - { - state: { - projectId: this.configService.getProjectId(), - nodeId: this.node.id, - newComponents: components - } - } - ); - } -} diff --git a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.html b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.html index 558c5e56adb..32193e544ae 100644 --- a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.html +++ b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.html @@ -1,171 +1,131 @@ -
-
- - -
- -
+
+ + +
+
-
- @if (!isGroupNode) { -
-
Components
+
+@if (components.length === 0 && !isGroupNode) { +

This step does not have any components.

+ +} +
+ @for (component of components; track component.id; let i = $index; let last = $last) { +
+ +
+
+ @if (components.length > 1) { + drag_indicator + } + {{ i + 1 }}. {{ getComponentTypeLabel(component.type) }} +
+ +
+ @if (component.id === editingComponentId) { + + } + + +
+
+ +
+ +
+
+
-
- + @if (i > 0) { + } + @if (!last) { -
- -
- - -
+ }
- } -
- @if (components.length === 0 && !isGroupNode) { -

This step does not have any components. Click the + button to add a component.

- } -
- @for (component of components; track component.id; let i = $index) { -
- + {{ i + 1 }}. {{ getComponentTypeLabel(component.type) }} - - - @if (components.length > 1) { - drag_indicator - } - - {{ i + 1 }}. {{ getComponentTypeLabel(component.type) }} - - - - - - - - - -
- @if (componentsToExpanded[component.id]) { - - } -
-
- -
- } -
+ +
+ }
diff --git a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.scss b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.scss index b120e0854ff..74ced36e24a 100644 --- a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.scss +++ b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.scss @@ -1,64 +1,51 @@ -.node-authoring { - .node-authoring-header { - margin: 8px 8px 0; +.node-authoring-header { + margin: 16px 8px 0; } - .node-authoring-controls { - padding: 8px 16px; - background-color: white; - position: sticky; - top: 0; - z-index: 2; - } - - .component { - flex: 1; - - .mat-expansion-panel-header { - padding-inline-start: 8px; - } +.component { + flex: 1; - .mat-expansion-panel-header-title { - flex-grow: 100; - } + &.component-editing { + border-inline-start: 4px solid; - .mat-expansion-panel-body { - padding-left: 16px; - padding-right: 16px; + edit-component-advanced-button { + display: block; } + } +} - &:hover, &:focus-within, &.mat-expanded { - .component-action { - display: block; - } - } +edit-component-advanced-button { + display: none; +} - &.mat-expanded { - edit-component-advanced-button { - display: block; - } - } - } +.component-header { + padding: 0 8px; +} - edit-component-advanced-button { - display: none; - } +component-authoring { + display: block; + margin: 16px; +} - .component-action { - display: none; - } +.mat-divider-horizontal { + margin: 0; +} - .drag-handle { - cursor: move; - margin-right: 4px; - margin-left: 4px; - } +.drag-handle { + cursor: move; + margin-right: 4px; + margin-left: 4px; +} - .cdk-drag-placeholder { - opacity: .4; - } +.cdk-drag-placeholder { + opacity: .4; + background: #ccc; + border: dotted 3px #999; + min-height: 40px; + min-width: 360px; + transition: transform 250ms cubic-bezier(0, 0, 0.2, 1); +} - .mat-icon { - margin: 0; - } +.mat-icon { + margin: 0; } diff --git a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.spec.ts b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.spec.ts index 058b125aa94..5beae635f7f 100644 --- a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.spec.ts +++ b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.spec.ts @@ -27,12 +27,12 @@ import { ProjectLocale } from '../../../../../app/domain/projectLocale'; import { TeacherProjectTranslationService } from '../../../services/teacherProjectTranslationService'; import { ComponentTypeServiceModule } from '../../../services/componentTypeService.module'; import { DeleteTranslationsService } from '../../../services/deleteTranslationsService'; -import { PreviewComponentButtonComponent } from '../../components/preview-component-button/preview-component-button.component'; import { CopyTranslationsService } from '../../../services/copyTranslationsService'; import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'; -import { RouterTestingModule } from '@angular/router/testing'; import { CreateComponentService } from '../../../services/createComponentService'; -import { MatExpansionModule } from '@angular/material/expansion'; +import { VLEProjectService } from '../../../vle/vleProjectService'; +import { NotebookService } from '../../../services/notebookService'; +import { MatDividerModule } from '@angular/material/divider'; let component: NodeAuthoringComponent; let component1: any; @@ -60,11 +60,9 @@ describe('NodeAuthoringComponent', () => { EditNodeTitleComponent, FormsModule, MatCheckboxModule, - MatExpansionModule, + MatDividerModule, MatIconModule, MatInputModule, - PreviewComponentButtonComponent, - RouterTestingModule, StudentTeacherCommonServicesModule, TeacherNodeIconComponent ], @@ -107,8 +105,8 @@ describe('NodeAuthoringComponent', () => { spyOn(document, 'getElementById').and.returnValue(document.createElement('div')); confirmSpy = spyOn(window, 'confirm'); component1 = { id: 'component1', type: 'OpenResponse', showSubmitButton: true }; - component2 = { id: 'component2', type: 'MultipleChoice', showSubmitButton: true }; - component3 = { id: 'component3', type: 'Match', showSubmitButton: true }; + component2 = { id: 'component2', type: 'MultipleChoice', showSubmitButton: true, choices: [] }; + component3 = { id: 'component3', type: 'HTML', showSubmitButton: true, html: '' }; node1Components = [component1, component2, component3]; teacherProjectService = TestBed.inject(TeacherProjectService); const node1 = { components: node1Components }; @@ -127,6 +125,10 @@ describe('NodeAuthoringComponent', () => { new ProjectLocale({ default: 'en-US', supported: ['es'] }) ); spyOn(TestBed.inject(TeacherProjectService), 'isDefaultLocale').and.returnValue(true); + const vleProjectService = TestBed.inject(VLEProjectService); + vleProjectService.project = teacherProjectService.project; + spyOn(TestBed.inject(VLEProjectService), 'getSpeechToTextSettings').and.returnValue(null); + spyOn(TestBed.inject(NotebookService), 'isNotebookEnabled').and.returnValue(false); saveProjectSpy = spyOn(teacherProjectService, 'saveProject').and.returnValue(Promise.resolve()); fixture = TestBed.createComponent(NodeAuthoringComponent); component = fixture.componentInstance; @@ -137,7 +139,6 @@ describe('NodeAuthoringComponent', () => { copyComponent(); deleteComponent(); - deleteComponents(); }); function copyComponent() { @@ -166,7 +167,7 @@ function deleteComponent() { confirmSpy.and.returnValue(true); clickComponentDeleteButton(component2.id); expect(confirmSpy).toHaveBeenCalledWith( - `Are you sure you want to delete this component?\n2. MultipleChoice` + `Are you sure you want to delete this component?\n\n2. MultipleChoice` ); expect(saveProjectSpy).toHaveBeenCalled(); expect(teacherProjectService.idToNode[nodeId1].components).toEqual([component1, component3]); @@ -174,33 +175,6 @@ function deleteComponent() { }); } -function deleteComponents() { - describe('deleteComponents()', () => { - it('should delete components', () => { - clickComponentCheckbox(component1.id); - clickComponentCheckbox(component3.id); - fixture.detectChanges(); - expect(component.components).toEqual(node1Components); - confirmSpy.and.returnValue(true); - clickDeleteComponentsButton(); - expect(confirmSpy).toHaveBeenCalledWith( - `Are you sure you want to delete these components?\n1. OpenResponse\n3. Match` - ); - expect(component.components).toEqual([component2]); - expect(expectCheckboxValue(component1.id)).toBeFalsy(); - expect(expectCheckboxValue(component3.id)).toBeFalsy(); - }); - }); -} - -function expectCheckboxValue(componentId: string): void { - return fixture.debugElement.query(By.css(`#${componentId} mat-checkbox`)).nativeElement.checked; -} - -function clickComponentCheckbox(componentId: string): void { - queryByCssAndClick(`#${componentId} mat-checkbox label`); -} - function clickComponent(componentId: string): void { queryByCssAndClick(`#${componentId}`); } @@ -221,10 +195,6 @@ function clickComponentDeleteButton(componentId: string): void { queryByCssAndClickDelete(`#${componentId} button`); } -function clickDeleteComponentsButton(): void { - queryByCssAndClickDelete('button'); -} - function queryByCssAndClickDelete(css: string): void { clickNativeElement(queryByCssAndInnerText(css, 'delete')); } diff --git a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.ts b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.ts index 77114eeff08..03fd7bb9f1d 100644 --- a/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.ts +++ b/src/assets/wise5/authoringTool/node/node-authoring/node-authoring.component.ts @@ -1,13 +1,4 @@ -import { - Component, - Input, - OnInit, - Signal, - ViewEncapsulation, - WritableSignal, - computed, - signal -} from '@angular/core'; +import { Component, Input, OnInit, ViewChild, ViewEncapsulation } from '@angular/core'; import { Subscription } from 'rxjs'; import { TeacherDataService } from '../../../services/teacherDataService'; import { TeacherProjectService } from '../../../services/teacherProjectService'; @@ -17,29 +8,22 @@ import { Node } from '../../../common/Node'; import { ComponentContent } from '../../../common/ComponentContent'; import { scrollToTopOfPage, temporarilyHighlightElement } from '../../../common/dom/dom'; import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop'; -import { ActivatedRoute, Router } from '@angular/router'; import { TeacherNodeService } from '../../../services/teacherNodeService'; import { DeleteTranslationsService } from '../../../services/deleteTranslationsService'; -import { copy } from '../../../common/object/object'; @Component({ selector: 'node-authoring', templateUrl: './node-authoring.component.html', - styleUrls: ['./node-authoring.component.scss'], - encapsulation: ViewEncapsulation.None + styleUrls: ['./node-authoring.component.scss'] }) export class NodeAuthoringComponent implements OnInit { components: ComponentContent[] = []; - protected componentsToChecked: WritableSignal<{ [key: string]: boolean }> = signal({}); - componentsToExpanded = {}; - protected isAnyComponentSelected: Signal = computed(() => - Object.values(this.componentsToChecked()).some((value) => value) - ); + protected editingComponentId: string; isGroupNode: boolean; node: Node; nodeJson: any; @Input() nodeId?: string; - subscriptions: Subscription = new Subscription(); + private subscriptions: Subscription = new Subscription(); constructor( private componentServiceLookupService: ComponentServiceLookupService, @@ -47,9 +31,7 @@ export class NodeAuthoringComponent implements OnInit { private nodeService: TeacherNodeService, private projectService: TeacherProjectService, private dataService: TeacherDataService, - private deleteTranslationsService: DeleteTranslationsService, - private route: ActivatedRoute, - private router: Router + private deleteTranslationsService: DeleteTranslationsService ) {} ngOnInit(): void { @@ -66,11 +48,10 @@ export class NodeAuthoringComponent implements OnInit { this.isGroupNode = this.projectService.isGroupNode(this.nodeId); this.nodeJson = this.projectService.getNodeById(this.nodeId); this.components = this.projectService.getComponents(this.nodeId); - this.componentsToChecked.set({}); - this.componentsToExpanded = {}; + this.editingComponentId = null; if (history.state.newComponents && history.state.newComponents.length > 0) { - this.highlightAndExpandComponents(history.state.newComponents); + this.highlightComponents(history.state.newComponents); } else { scrollToTopOfPage(); } @@ -140,82 +121,24 @@ export class NodeAuthoringComponent implements OnInit { return this.projectService.saveProject(); } - protected getSelectedComponents(): ComponentContent[] { - return this.components.filter( - (component: ComponentContent) => this.componentsToChecked()[component.id] - ); - } - - /** - * Get the component numbers and component types that have been selected - * @return an array of strings - * example - * [ - * "1. OpenResponse", - * "3. MultipleChoice" - * ] - */ - private getSelectedComponentNumbersAndTypes(): string[] { - const selectedComponents = []; - for (let c = 0; c < this.components.length; c++) { - const component = this.components[c]; - if (this.componentsToChecked()[component.id]) { - const componentNumberAndType = c + 1 + '. ' + component.type; - selectedComponents.push(componentNumberAndType); - } - } - return selectedComponents; - } - - protected chooseComponentLocation(action: string): void { - this.router.navigate(['choose-component-location'], { - relativeTo: this.route, - state: { - action: action, - selectedComponents: this.getSelectedComponents() - } - }); - } - - protected deleteComponents(): void { - scrollToTopOfPage(); - if (this.confirmDeleteComponent(this.getSelectedComponentNumbersAndTypes())) { - this.deleteComponentsOnServer( - this.getSelectedComponents().map((component) => this.node.deleteComponent(component.id)) - ); - } - } - protected deleteComponent( event: any, componentNumber: number, component: ComponentContent ): void { event.stopPropagation(); - if (this.confirmDeleteComponent([`${componentNumber}. ${component.type}`])) { + if ( + confirm( + $localize`Are you sure you want to delete this component?\n\n${componentNumber}. ${component.type}` + ) + ) { this.deleteComponentsOnServer([this.node.deleteComponent(component.id)]); } } - private confirmDeleteComponent(componentLabels: string[]): boolean { - let confirmMessage = - componentLabels.length === 1 - ? $localize`Are you sure you want to delete this component?\n` - : $localize`Are you sure you want to delete these components?\n`; - confirmMessage += `${componentLabels.join('\n')}`; - return confirm(confirmMessage); - } - private deleteComponentsOnServer(components: ComponentContent[]): void { this.checkIfNeedToShowNodeSaveOrNodeSubmitButtons(); this.projectService.saveProject().then(() => { - for (const component of components) { - this.componentsToChecked.update((obj) => { - delete obj[component.id]; - return copy(obj); - }); - delete this.componentsToExpanded[component.id]; - } this.deleteTranslationsService.tryDeleteComponents(components); }); } @@ -240,22 +163,15 @@ export class NodeAuthoringComponent implements OnInit { } /** - * Temporarily highlight the specified components and show the component - * authoring views. Used to bring user's attention to new changes. - * @param components an array of components to highlight and expand + * Temporarily highlight the specified components + * @param components an array of components to highlight */ - protected highlightAndExpandComponents(components: any = []): void { - this.componentsToChecked.set({}); - - // wait for the UI to update and then scroll to the first new component + protected highlightComponents(components: any = []): void { + // wait for the UI to update and then highlight the first component setTimeout(() => { if (components.length > 0) { - const componentElement = $('#' + components[0].id); - $('#content').scrollTop(componentElement.offset().top - 200); - for (const component of components) { - temporarilyHighlightElement(component.id); - this.componentsToExpanded[component.id] = true; - } + document.getElementById(components[0].id).scrollIntoView(); + components.forEach((component) => temporarilyHighlightElement(component.id)); } }, 100); } @@ -264,29 +180,6 @@ export class NodeAuthoringComponent implements OnInit { return this.componentTypeService.getComponentTypeLabel(componentType); } - protected componentCheckboxChanged(componentId: string, checked: boolean): void { - this.componentsToChecked.update((componentsToChecked) => { - componentsToChecked[componentId] = checked; - return copy(componentsToChecked); - }); - } - - protected toggleComponent(componentId: string, expanded: boolean = true): void { - this.componentsToExpanded[componentId] = expanded; - this.projectService.uiChanged(); - } - - protected setAllComponentsIsExpanded(isExpanded: boolean): void { - this.components.forEach((component) => { - this.componentsToExpanded[component.id] = isExpanded; - }); - this.projectService.uiChanged(); - } - - protected getNumberOfComponentsExpanded(): number { - return Object.values(this.componentsToExpanded).filter((value) => value).length; - } - private setShowSaveButtonForAllComponents(node: Node, showSaveButton: boolean): void { node.components .filter((component) => @@ -296,7 +189,22 @@ export class NodeAuthoringComponent implements OnInit { } protected dropComponent(event: CdkDragDrop): void { - moveItemInArray(this.components, event.previousIndex, event.currentIndex); + this.moveComponent(event.previousIndex, event.currentIndex); + } + + protected moveComponent( + previousIndex: number, + currentIndex: number, + scroll: boolean = false + ): void { + moveItemInArray(this.components, previousIndex, currentIndex); + if (scroll) { + this.highlightComponents([this.components[currentIndex]]); + } this.projectService.saveProject(); } + + protected editComponent(componentId: string): void { + this.editingComponentId = componentId; + } } diff --git a/src/assets/wise5/authoringTool/project-authoring-parent/project-authoring-parent.component.scss b/src/assets/wise5/authoringTool/project-authoring-parent/project-authoring-parent.component.scss index 7155e961a66..7e5bae93d20 100644 --- a/src/assets/wise5/authoringTool/project-authoring-parent/project-authoring-parent.component.scss +++ b/src/assets/wise5/authoringTool/project-authoring-parent/project-authoring-parent.component.scss @@ -11,7 +11,6 @@ .concurrent-authors-message { position: sticky; top: 1px; - padding: 4px 0; display: block; z-index: 3; } diff --git a/src/assets/wise5/classroomMonitor/classroom-monitor-testing.module.ts b/src/assets/wise5/classroomMonitor/classroom-monitor-testing.module.ts index 4c43087a0c2..bc446eec3cc 100644 --- a/src/assets/wise5/classroomMonitor/classroom-monitor-testing.module.ts +++ b/src/assets/wise5/classroomMonitor/classroom-monitor-testing.module.ts @@ -14,6 +14,7 @@ import { MilestoneReportService } from '../services/milestoneReportService'; import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'; import { TeacherPauseScreenService } from '../services/teacherPauseScreenService'; import { RunStatusService } from '../services/runStatusService'; +import { GradingNodeService } from '../services/gradingNodeService'; @NgModule({ imports: [ @@ -24,6 +25,7 @@ import { RunStatusService } from '../services/runStatusService'; ], providers: [ ClassroomStatusService, + GradingNodeService, MilestoneService, MilestoneReportService, TeacherDataService, diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component.spec.ts b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component.spec.ts new file mode 100644 index 00000000000..899b356890c --- /dev/null +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component.spec.ts @@ -0,0 +1,22 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; +import { GradingStepToolsComponent } from './grading-step-tools.component'; +import { ClassroomMonitorTestingModule } from '../../classroom-monitor-testing.module'; + +describe('GradingStepToolsComponent', () => { + let component: GradingStepToolsComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + imports: [GradingStepToolsComponent, ClassroomMonitorTestingModule] + }).compileComponents(); + + fixture = TestBed.createComponent(GradingStepToolsComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component.ts b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component.ts new file mode 100644 index 00000000000..5094d56d5a3 --- /dev/null +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/grading-step-tools/grading-step-tools.component.ts @@ -0,0 +1,58 @@ +import { Component, ViewEncapsulation } from '@angular/core'; +import { Directionality } from '@angular/cdk/bidi'; +import { StepToolsComponent } from '../../../common/stepTools/step-tools.component'; +import { MatButtonModule } from '@angular/material/button'; +import { MatFormFieldModule } from '@angular/material/form-field'; +import { MatIconModule } from '@angular/material/icon'; +import { MatSelectModule } from '@angular/material/select'; +import { FlexLayoutModule } from '@angular/flex-layout'; +import { MatTooltipModule } from '@angular/material/tooltip'; +import { CommonModule } from '@angular/common'; +import { FormsModule } from '@angular/forms'; +import { MatInputModule } from '@angular/material/input'; +import { NodeIconComponent } from '../../../vle/node-icon/node-icon.component'; +import { TeacherDataService } from '../../../services/teacherDataService'; +import { GradingNodeService } from '../../../services/gradingNodeService'; +import { TeacherProjectService } from '../../../services/teacherProjectService'; + +@Component({ + encapsulation: ViewEncapsulation.None, + imports: [ + CommonModule, + FlexLayoutModule, + FormsModule, + MatButtonModule, + MatFormFieldModule, + MatIconModule, + MatInputModule, + MatSelectModule, + MatTooltipModule, + NodeIconComponent + ], + selector: 'grading-step-tools', + standalone: true, + templateUrl: '../../../common/stepTools/step-tools.component.html', + styleUrl: '../../../common/stepTools/step-tools.component.scss' +}) +export class GradingStepToolsComponent extends StepToolsComponent { + constructor( + protected dataService: TeacherDataService, + protected dir: Directionality, + protected nodeService: GradingNodeService, + protected projectService: TeacherProjectService + ) { + super(dataService, dir, nodeService, projectService); + } + + protected calculateNodeIds(): void { + this.nodeIds = Object.keys(this.projectService.idToOrder); + this.nodeIds = this.nodeIds.filter((nodeId) => { + return this.isGroupNode(nodeId) || this.projectService.nodeHasWork(nodeId); + }); + this.nodeIds.shift(); // remove the 'group0' master root node from consideration + } + + protected getNextNodeId(): Promise { + return Promise.resolve(this.nodeService.getNextNodeId(this.nodeId)); + } +} diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/milestones/milestone-grading-view/milestone-grading-view.component.html b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/milestones/milestone-grading-view/milestone-grading-view.component.html index 9e50d4d61b1..53eb3f4ef20 100644 --- a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/milestones/milestone-grading-view/milestone-grading-view.component.html +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/milestones/milestone-grading-view/milestone-grading-view.component.html @@ -8,7 +8,7 @@ fxLayoutAlign="center center" fxLayoutGap.lt-md="8px" > - +
- -
- -
-
-

- {{ i + 1 + '. ' + getComponentTypeLabel(component.type) }}  - -

- -
-
-
-
-
+ @if (expanded && !disabled) { + +
+ @for (component of components; track component.id; let i = $index) { + @if (componentIdToIsVisible[component.id]) { +
+
+

+ {{ i + 1 + '. ' + getComponentTypeLabel(component.type) }}  + +

+ +
+
+ } + } +
+
+ }
diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.spec.ts b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.spec.ts index 3ae9af6d290..db074bf87f1 100644 --- a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.spec.ts +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.spec.ts @@ -2,21 +2,17 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ClassroomMonitorTestingModule } from '../../../classroom-monitor-testing.module'; import { WorkgroupItemComponent } from './workgroup-item.component'; import { TeacherProjectService } from '../../../../services/teacherProjectService'; -import { NO_ERRORS_SCHEMA } from '@angular/core'; import { ComponentTypeServiceModule } from '../../../../services/componentTypeService.module'; let component: WorkgroupItemComponent; let fixture: ComponentFixture; let getComponentsSpy: jasmine.Spy; let teacherProjectService: TeacherProjectService; - describe('WorkgroupItemComponent', () => { beforeEach(async () => { await TestBed.configureTestingModule({ - declarations: [WorkgroupItemComponent], - imports: [ClassroomMonitorTestingModule, ComponentTypeServiceModule], - providers: [TeacherProjectService], - schemas: [NO_ERRORS_SCHEMA] + imports: [WorkgroupItemComponent, ClassroomMonitorTestingModule, ComponentTypeServiceModule], + providers: [TeacherProjectService] }).compileComponents(); }); diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.ts b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.ts index 5066a8e97e6..548354991e9 100644 --- a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.ts +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/nodeGrading/workgroup-item/workgroup-item.component.ts @@ -2,30 +2,51 @@ import { Component, EventEmitter, Input, Output, SimpleChanges } from '@angular/ import { ComponentTypeService } from '../../../../services/componentTypeService'; import { TeacherProjectService } from '../../../../services/teacherProjectService'; import { calculateComponentVisibility } from '../../shared/grading-helpers/grading-helpers'; +import { CommonModule } from '@angular/common'; +import { MatButtonModule } from '@angular/material/button'; +import { MatListModule } from '@angular/material/list'; +import { FlexLayoutModule } from '@angular/flex-layout'; +import { WorkgroupInfoComponent } from '../workgroupInfo/workgroup-info.component'; +import { WorkgroupNodeStatusComponent } from '../../../../../../app/classroom-monitor/workgroup-node-status/workgroup-node-status.component'; +import { WorkgroupNodeScoreComponent } from '../../shared/workgroupNodeScore/workgroup-node-score.component'; +import { ComponentNewWorkBadgeComponent } from '../../../../../../app/classroom-monitor/component-new-work-badge/component-new-work-badge.component'; +import { WorkgroupComponentGradingComponent } from '../../workgroup-component-grading/workgroup-component-grading.component'; @Component({ + imports: [ + CommonModule, + MatButtonModule, + MatListModule, + FlexLayoutModule, + WorkgroupInfoComponent, + WorkgroupNodeStatusComponent, + WorkgroupNodeScoreComponent, + ComponentNewWorkBadgeComponent, + WorkgroupComponentGradingComponent + ], selector: 'workgroup-item', - templateUrl: 'workgroup-item.component.html', - styleUrls: ['workgroup-item.component.scss'] + standalone: true, + styleUrl: 'workgroup-item.component.scss', + templateUrl: 'workgroup-item.component.html' }) export class WorkgroupItemComponent { - componentIdToHasWork: { [componentId: string]: boolean } = {}; - componentIdToIsVisible: { [componentId: string]: boolean } = {}; - components: any[] = []; - disabled: boolean; + private componentIdToHasWork: { [componentId: string]: boolean } = {}; + protected componentIdToIsVisible: { [componentId: string]: boolean } = {}; + protected components: any[] = []; + protected disabled: boolean; @Input() expanded: boolean; - hasAlert: boolean; - hasNewAlert: boolean; + protected hasAlert: boolean; + protected hasNewAlert: boolean; @Input() hiddenComponents: string[] = []; @Input() maxScore: number; - nodeHasWork: boolean; + private nodeHasWork: boolean; @Input() nodeId: string; @Output() onUpdateExpand: EventEmitter = new EventEmitter(); - score: any; + protected score: any; @Input() showScore: boolean; - status: any; - statusClass: string; - statusText: string = ''; + private status: any; + protected statusClass: string; + protected statusText: string = ''; @Input() workgroupId: number; @Input() workgroupData: any; @@ -39,7 +60,7 @@ export class WorkgroupItemComponent { this.updateStatus(); } - updateNode(): void { + private updateNode(): void { this.nodeHasWork = this.projectService.nodeHasWork(this.nodeId); this.components = this.projectService.getComponents(this.nodeId); this.componentIdToHasWork = this.projectService.calculateComponentIdToHasWork(this.components); @@ -69,11 +90,11 @@ export class WorkgroupItemComponent { } } - isComponentVisible(componentId: string): boolean { + protected isComponentVisible(componentId: string): boolean { return !this.hiddenComponents.includes(componentId); } - getComponentTypeLabel(componentType): string { + protected getComponentTypeLabel(componentType): string { return this.componentTypeService.getComponentTypeLabel(componentType); } @@ -109,10 +130,9 @@ export class WorkgroupItemComponent { this.disabled = this.status === -1; } - toggleExpand(): void { + protected toggleExpand(): void { if (this.showScore) { - let expand = !this.expanded; - this.onUpdateExpand.emit({ workgroupId: this.workgroupId, value: expand }); + this.onUpdateExpand.emit({ workgroupId: this.workgroupId, value: !this.expanded }); } } } diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/shared/tool-bar/tool-bar.component.html b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/shared/tool-bar/tool-bar.component.html index 22c0c7c0868..7ff75c048a6 100644 --- a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/shared/tool-bar/tool-bar.component.html +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/shared/tool-bar/tool-bar.component.html @@ -13,7 +13,7 @@ {{ viewName }} @if (showStepTools) { - + } diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/studentGrading/student-grading-tools/student-grading-tools.component.html b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/studentGrading/student-grading-tools/student-grading-tools.component.html index 4f4d7f87da7..2f9df33dc01 100644 --- a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/studentGrading/student-grading-tools/student-grading-tools.component.html +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/studentGrading/student-grading-tools/student-grading-tools.component.html @@ -12,7 +12,7 @@ chevron_left account_circle - +
- -
- -
+ } diff --git a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/view-component-revisions/view-component-revisions.component.ts b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/view-component-revisions/view-component-revisions.component.ts index e9a1cb62246..29a1a2083d1 100644 --- a/src/assets/wise5/classroomMonitor/classroomMonitorComponents/view-component-revisions/view-component-revisions.component.ts +++ b/src/assets/wise5/classroomMonitor/classroomMonitorComponents/view-component-revisions/view-component-revisions.component.ts @@ -1,38 +1,54 @@ import { Component, Inject } from '@angular/core'; +import { CommonModule } from '@angular/common'; +import { MatButtonModule } from '@angular/material/button'; +import { MAT_DIALOG_DATA, MatDialogModule } from '@angular/material/dialog'; +import { MatListModule } from '@angular/material/list'; +import { FlexLayoutModule } from '@angular/flex-layout'; import { AnnotationService } from '../../../services/annotationService'; -import { ConfigService } from '../../../services/configService'; -import { TeacherDataService } from '../../../services/teacherDataService'; -import { MAT_DIALOG_DATA } from '@angular/material/dialog'; +import { ConfigService } from '../../..//services/configService'; +import { TeacherDataService } from '../../..//services/teacherDataService'; +import { ComponentGradingComponent } from '../component-grading.component'; +import { ComponentStateInfoComponent } from '../../../common/component-state-info/component-state-info.component'; +import { EditComponentAnnotationsComponent } from '../edit-component-annotations/edit-component-annotations.component'; import { Observable } from 'rxjs'; @Component({ - selector: 'view-component-revisions.component', - styleUrls: ['view-component-revisions.component.scss'], + imports: [ + CommonModule, + MatDialogModule, + MatListModule, + MatButtonModule, + FlexLayoutModule, + ComponentGradingComponent, + ComponentStateInfoComponent, + EditComponentAnnotationsComponent + ], + selector: 'view-component-revisions', + standalone: true, + styleUrl: 'view-component-revisions.component.scss', templateUrl: 'view-component-revisions.component.html' }) export class ViewComponentRevisionsComponent { - componentId: string; - componentStates: any = []; - fromWorkgroupId: number; - increment: number = 5; - nodeId: string; - numRevisionsShown: number = 5; - revisions: any = {}; - revisionsSorted: any[]; - totalRevisions: number; - usernames: string[]; - workgroupId: number; + protected componentId: string; + protected fromWorkgroupId: number; + private increment: number = 5; + protected nodeId: string; + protected numRevisionsShown: number = 5; + private revisions: any = {}; + protected revisionsSorted: any[]; + protected totalRevisions: number; + protected usernames: string[]; + protected workgroupId: number; constructor( private annotationService: AnnotationService, private configService: ConfigService, - private dataService: TeacherDataService, - @Inject(MAT_DIALOG_DATA) public data: any + @Inject(MAT_DIALOG_DATA) public data: any, + private dataService: TeacherDataService ) {} ngOnInit() { this.componentId = this.data.componentId; - this.componentStates = this.data.componentStates; this.fromWorkgroupId = this.data.fromWorkgroupId; this.nodeId = this.data.nodeId; this.workgroupId = this.data.workgroupId; @@ -49,24 +65,22 @@ export class ViewComponentRevisionsComponent { this.revisions = {}; this.totalRevisions = 0; this.getNodeEnteredEvents().subscribe(({ events }) => { - const nodeVisits = events.map((event) => { - return { - serverSaveTime: event.serverSaveTime, - states: [] - }; - }); + const nodeVisits = events.map((event) => ({ + serverSaveTime: event.serverSaveTime, + states: [] + })); this.populateDataHelper(nodeVisits); }); } - populateDataHelper(nodeVisits: any[]) { + private populateDataHelper(nodeVisits: any[]): void { // group all component states by node visit - for (let cStatesIndex = this.componentStates.length - 1; cStatesIndex > -1; cStatesIndex--) { - const componentState = this.componentStates[cStatesIndex]; + for (let i = this.data.componentStates.length - 1; i > -1; i--) { + const componentState = this.data.componentStates[i]; if (nodeVisits.length > 0) { // add state to corresponding node visit - for (let nVisitsIndex = nodeVisits.length - 1; nVisitsIndex > -1; nVisitsIndex--) { - const nodeVisit = nodeVisits[nVisitsIndex]; + for (let j = nodeVisits.length - 1; j > -1; j--) { + const nodeVisit = nodeVisits[j]; if (componentState.serverSaveTime >= nodeVisit.serverSaveTime) { nodeVisit.states.push(componentState); break; @@ -76,46 +90,45 @@ export class ViewComponentRevisionsComponent { // we don't have any node visits, so count all all states as revisions. this.totalRevisions++; this.revisions[componentState.id] = { - clientSaveTime: this.convertToClientTimestamp(componentState.serverSaveTime), + clientSaveTime: this.configService.convertToClientTimestamp( + componentState.serverSaveTime + ), componentState: componentState }; } } // find revisions in each node visit and add to model - for (let visitsIndex = 0; visitsIndex < nodeVisits.length; visitsIndex++) { - const states = nodeVisits[visitsIndex].states; - for (let i = 0; i < states.length; i++) { - const state = states[i]; - let isRevision = false; - if (i === 0) { - isRevision = true; // latest state for a visit always counts as a revision - } else if (state.isSubmit) { - isRevision = true; - } else { - for (const annotation of this.annotationService.getAnnotationsByStudentWorkId(state.id)) { - if (['score', 'autoScore', 'comment', 'autoComment'].includes(annotation.type)) { - isRevision = true; // is revision if there is an annotation for the component - break; - } - } - } - if (isRevision) { + nodeVisits.forEach((nodeVisit) => { + nodeVisit.states + .filter((state, index) => this.isRevision(state, index)) + .forEach((state) => { this.totalRevisions++; this.revisions[state.id] = { - clientSaveTime: this.convertToClientTimestamp(state.serverSaveTime), + clientSaveTime: this.configService.convertToClientTimestamp(state.serverSaveTime), componentState: state }; - } - } - } + }); + }); this.sortRevisions(); } - sortRevisions() { - this.revisionsSorted = Object.values(this.revisions).sort((a: any, b: any) => { - return b.clientSaveTime - a.clientSaveTime; - }); + private isRevision(state: any, stateIndex: number): boolean { + return ( + stateIndex === 0 || + state.isSubmit || + this.annotationService + .getAnnotationsByStudentWorkId(state.id) + .some((annotation) => + ['score', 'autoScore', 'comment', 'autoComment'].includes(annotation.type) + ) + ); + } + + private sortRevisions(): void { + this.revisionsSorted = Object.values(this.revisions).sort( + (a: any, b: any) => b.clientSaveTime - a.clientSaveTime + ); } private getNodeEnteredEvents(): Observable { @@ -130,11 +143,7 @@ export class ViewComponentRevisionsComponent { }); } - convertToClientTimestamp(time: number) { - return this.configService.convertToClientTimestamp(time); - } - - showMore() { + protected showMore(): void { this.numRevisionsShown += this.increment; } } diff --git a/src/assets/wise5/classroomMonitor/notebook-grading/notebook-grading.component.html b/src/assets/wise5/classroomMonitor/notebook-grading/notebook-grading.component.html index 90b52730d3d..89733ce9def 100644 --- a/src/assets/wise5/classroomMonitor/notebook-grading/notebook-grading.component.html +++ b/src/assets/wise5/classroomMonitor/notebook-grading/notebook-grading.component.html @@ -9,7 +9,7 @@ fxLayoutAlign="center center" fxLayoutGap.lt-md="8px" > - +