diff --git a/docs/release-notes-editor.txt b/docs/release-notes-editor.txt index 68f45cee8df985b481b43439d1f5ffe2e8103a27..e54c428d183ab8f41a9e360b6ae18728b063bd9c 100644 --- a/docs/release-notes-editor.txt +++ b/docs/release-notes-editor.txt @@ -1,5 +1,11 @@ Editor ====== +1.35.0 +- Implement hotspot image element + Clickable and non-clickable (read-only) areas can be placed on an image. + The shapes of the areas can be ellipses and rectangles. Their position, + size, rotation, color and frame thickness can be set. The elements can be marked as required. + 1.34.0 - Implement GeoGebra applet element This needs a base64 representation of a unit. diff --git a/docs/release-notes-player.txt b/docs/release-notes-player.txt index 739dea7068895316440c18fe1c12638712c98dd8..473415b0369aa01f2bea9648a4b7d5c58eee841f 100644 --- a/docs/release-notes-player.txt +++ b/docs/release-notes-player.txt @@ -1,5 +1,9 @@ Player ====== +1.20.0 +- Support for hotspot image element + Clickable areas of the image are colored on click. + 1.27.0 - Add new input assistance presets ('Space' and 'Comma'). The preset 'Space' can be used to separate words. diff --git a/projects/common/components/input-elements/hotspot-image.component.ts b/projects/common/components/input-elements/hotspot-image.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..4f09383d2533f2a6779a7abb6dcea3681750fabe --- /dev/null +++ b/projects/common/components/input-elements/hotspot-image.component.ts @@ -0,0 +1,56 @@ +import { Component, Input } from '@angular/core'; +import { HotspotImageElement } from 'common/models/elements/input-elements/hotspot-image'; +import { FormElementComponent } from '../../directives/form-element-component.directive'; + +@Component({ + selector: 'aspect-hotspot-image', + template: ` + <div *ngIf="elementModel.src" + [style.width.%]="100" + [style.height.%]="100" + class="image-container"> + <img [src]="elementModel.src | safeResourceUrl" + [alt]="'imageNotFound' | translate" + tabindex="0" + (focusout)="elementFormControl.markAsTouched()"> + <div *ngFor="let item of elementModel.value; let index = index"> + <div class="hotspot" + [class.active-hotspot]="!item.readOnly" + [class.circle]="item.shape === 'ellipse'" + [class.border]="item.borderWidth > 0" + [style.border-width.px]="item.borderWidth" + [style.border-color]="item.borderColor" + [style.background-color]="(parentForm && elementFormControl.value[index].value) || + (!parentForm && item.value) ? + item.backgroundColor : + 'transparent'" + [style.top.px]="item.top" + [style.left.px]="item.left" + [style.width.px]="item.width" + [style.height.px]="item.height" + [style.transform]="'rotate(' + item.rotation + 'deg)'" + (click)="!item.readOnly && parentForm ? onHotspotClicked(index) : null"> + </div> + </div> + <mat-error *ngIf="elementFormControl.errors && elementFormControl.touched"> + {{elementFormControl.errors | errorTransform: elementModel}} + </mat-error> + </div> + `, + styles: [ + '.circle {border-radius: 50%;}', + '.border {border-color: #000000; border-style: solid}', + '.hotspot {position: absolute; box-sizing: border-box;}', + '.active-hotspot {cursor: pointer;}', + '.image-container {position: relative;}' + ] +}) +export class HotspotImageComponent extends FormElementComponent { + @Input() elementModel!: HotspotImageElement; + + onHotspotClicked(index: number): void { + const actualValue = this.elementFormControl.value; + actualValue[index].value = !actualValue[index].value; + this.elementFormControl.setValue(actualValue); + } +} diff --git a/projects/common/models/elements/element.ts b/projects/common/models/elements/element.ts index e7b072b657b2349f1b8f9ef8baf78bf642059415..ee0b7536e38def1f018cbfd42de01299abc80f98 100644 --- a/projects/common/models/elements/element.ts +++ b/projects/common/models/elements/element.ts @@ -6,11 +6,11 @@ import { ElementFactory } from 'common/util/element.factory'; import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; export type UIElementType = 'text' | 'button' | 'text-field' | 'text-field-simple' | 'text-area' | 'checkbox' -| 'dropdown' | 'radio' | 'image' | 'audio' | 'video' | 'likert' | 'likert-row' | 'radio-group-images' +| 'dropdown' | 'radio' | 'image' | 'audio' | 'video' | 'likert' | 'likert-row' | 'radio-group-images' | 'hotspot-image' | 'drop-list' | 'drop-list-simple' | 'cloze' | 'spell-correct' | 'slider' | 'frame' | 'toggle-button' | 'geometry'; export type UIElementValue = string | number | boolean | undefined | UIElementType | InputElementValue | -TextLabel | TextLabel[] | ClozeDocument | LikertRowElement[] | +TextLabel | TextLabel[] | ClozeDocument | LikertRowElement[] | Hotspot[] | PositionProperties | PlayerProperties | BasicStyles; export type InputAssistancePreset = null | 'french' | 'numbers' | 'numbersAndOperators' | 'numbersAndBasicOperators' @@ -61,7 +61,7 @@ export abstract class UIElement { abstract getElementComponent(): Type<ElementComponent>; } -export type InputElementValue = string[] | string | number | boolean | TextLabel[] | null; +export type InputElementValue = string[] | string | number | boolean | TextLabel[] | null | Hotspot[] | boolean[]; export abstract class InputElement extends UIElement { label: string = 'Beschriftung'; @@ -200,6 +200,20 @@ export interface PlayerProperties { playbackTime: number; } +export interface Hotspot { + top: number; + left: number; + width: number; + height: number; + shape: 'ellipse' | 'rect'; + borderWidth: number; + borderColor: string; + backgroundColor: string; + rotation: number; + value: boolean; + readOnly: boolean +} + export interface ValueChangeElement { id: string; value: InputElementValue; diff --git a/projects/common/models/elements/input-elements/hotspot-image.ts b/projects/common/models/elements/input-elements/hotspot-image.ts new file mode 100644 index 0000000000000000000000000000000000000000..dc4d79671fb67dae6c20a666305e754a1324b8c2 --- /dev/null +++ b/projects/common/models/elements/input-elements/hotspot-image.ts @@ -0,0 +1,53 @@ +import { Type } from '@angular/core'; +import { ElementFactory } from 'common/util/element.factory'; +import { + InputElement, PositionedUIElement, PositionProperties, AnswerScheme, Hotspot, AnswerSchemeValue +} from 'common/models/elements/element'; +import { ElementComponent } from 'common/directives/element-component.directive'; +import { HotspotImageComponent } from 'common/components/input-elements/hotspot-image.component'; + +export class HotspotImageElement extends InputElement implements PositionedUIElement { + value: Hotspot[]; + src: string | null = null; + position: PositionProperties; + + constructor(element: Partial<HotspotImageElement>) { + super({ height: 100, ...element }); + this.value = element.value !== undefined ? [...element.value] : []; + if (element.src) this.src = element.src; + this.position = ElementFactory.initPositionProps(element.position); + } + + hasAnswerScheme(): boolean { + return Boolean(this.getAnswerScheme); + } + + getAnswerScheme(): AnswerScheme { + return { + id: this.id, + type: 'boolean', + format: '', + multiple: true, + nullable: false, + values: this.getAnswerSchemeValues(), + valuesComplete: true + }; + } + + private getAnswerSchemeValues(): AnswerSchemeValue[] { + return this.value + .map((hotspot, index) => ({ + value: (index + 1).toString(), + label: `top: ${hotspot.top}, + left: ${hotspot.left}, + height: ${hotspot.height}, + width: ${hotspot.width}, + shape: ${hotspot.shape}, + value: ${hotspot.value}` + })); + } + + getElementComponent(): Type<ElementComponent> { + return HotspotImageComponent; + } +} diff --git a/projects/common/models/section.ts b/projects/common/models/section.ts index 96ef87fc0b86d0f21190e451fde2bca7aeade9c6..38a12066ca0a746126b7dcdff6c1378f3d7073f6 100644 --- a/projects/common/models/section.ts +++ b/projects/common/models/section.ts @@ -27,6 +27,7 @@ import { SpellCorrectElement } from 'common/models/elements/input-elements/spell import { FrameElement } from 'common/models/elements/frame/frame'; import { ToggleButtonElement } from 'common/models/elements/compound-elements/cloze/cloze-child-elements/toggle-button'; import { GeometryElement } from 'common/models/elements/geometry/geometry'; +import { HotspotImageElement } from 'common/models/elements/input-elements/hotspot-image'; export class Section { [index: string]: unknown; @@ -49,6 +50,7 @@ export class Section { checkbox: CheckboxElement, dropdown: DropdownElement, radio: RadioButtonGroupElement, + 'hotspot-image': HotspotImageElement, image: ImageElement, audio: AudioElement, video: VideoElement, diff --git a/projects/common/shared.module.ts b/projects/common/shared.module.ts index 102cb183d48b3da6f3c640f4b40b1bd13838eb22..161d618b177bdcfb5c384b3820bb9778d04a1c9a 100644 --- a/projects/common/shared.module.ts +++ b/projects/common/shared.module.ts @@ -21,6 +21,7 @@ import { MatDialogModule } from '@angular/material/dialog'; import { MatSliderModule } from '@angular/material/slider'; import { DomSanitizer } from '@angular/platform-browser'; import { HttpClientModule } from '@angular/common/http'; +import { HotspotImageComponent } from 'common/components/input-elements/hotspot-image.component'; import { TextComponent } from './components/text/text.component'; import { ButtonComponent } from './components/button/button.component'; import { TextFieldComponent } from './components/input-elements/text-field.component'; @@ -113,6 +114,7 @@ import { GeometryComponent } from './components/geometry/geometry.component'; RadioGroupImagesComponent, DropListComponent, ClozeComponent, + HotspotImageComponent, DropListSimpleComponent, SliderComponent, SpellCorrectComponent, @@ -158,6 +160,7 @@ import { GeometryComponent } from './components/geometry/geometry.component'; RadioGroupImagesComponent, DropListComponent, ClozeComponent, + HotspotImageComponent, LikertComponent, ButtonComponent, FrameComponent, diff --git a/projects/editor/src/app/app.module.ts b/projects/editor/src/app/app.module.ts index a42abb5f032bd0ffef2251aecc16e7cd61adf413..a2e2e07121540eb8931aae0adee6a511dd830064 100644 --- a/projects/editor/src/app/app.module.ts +++ b/projects/editor/src/app/app.module.ts @@ -21,6 +21,13 @@ import { MatListModule } from '@angular/material/list'; import { APIService, SharedModule } from 'common/shared.module'; import { SectionInsertDialogComponent } from 'editor/src/app/components/dialogs/section-insert-dialog.component'; +import { HotspotListPanelComponent } from 'editor/src/app/components/properties-panel/hotspot-list-panel.component'; +import { VeronaAPIService } from 'editor/src/app/services/verona-api.service'; +import { MatRadioModule } from '@angular/material/radio'; +import { + HotspotFieldSetComponent +} from 'editor/src/app/components/properties-panel/model-properties-tab/input-groups/hotspot-field-set.component'; +import { HotspotEditDialogComponent } from 'editor/src/app/components/dialogs/hotspot-edit-dialog.component'; import { AppComponent } from './app.component'; import { ToolbarComponent } from './components/toolbar/toolbar.component'; import { UiElementToolboxComponent } from @@ -88,7 +95,6 @@ import { LikertRowLabelPipe } from './components/properties-panel/likert-row-lab import { LabelEditDialogComponent } from './components/dialogs/label-edit-dialog.component'; import { BorderPropertiesComponent } from './components/properties-panel/model-properties-tab/input-groups/border-properties.component'; import { GeogebraAppDefinitionDialogComponent } from './components/dialogs/geogebra-app-definition-dialog.component'; -import { VeronaAPIService } from 'editor/src/app/services/verona-api.service'; @NgModule({ declarations: [ @@ -115,6 +121,7 @@ import { VeronaAPIService } from 'editor/src/app/services/verona-api.service'; PlayerEditDialogComponent, LikertRowEditDialogComponent, RichTextEditDialogComponent, + HotspotEditDialogComponent, ElementModelPropertiesComponent, DropListOptionEditDialogComponent, PositionFieldSetComponent, @@ -135,6 +142,8 @@ import { VeronaAPIService } from 'editor/src/app/services/verona-api.service'; InputElementPropertiesComponent, PresetValuePropertiesComponent, OptionListPanelComponent, + HotspotFieldSetComponent, + HotspotListPanelComponent, LikertRowLabelPipe, LabelEditDialogComponent, BorderPropertiesComponent, @@ -162,7 +171,8 @@ import { VeronaAPIService } from 'editor/src/app/services/verona-api.service'; useClass: EditorTranslateLoader } }), - MatListModule + MatListModule, + MatRadioModule ], providers: [ { provide: APIService, useExisting: VeronaAPIService } diff --git a/projects/editor/src/app/components/dialogs/hotspot-edit-dialog.component.ts b/projects/editor/src/app/components/dialogs/hotspot-edit-dialog.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..8acee1882bc50d925f44dffcc1326f524dca2427 --- /dev/null +++ b/projects/editor/src/app/components/dialogs/hotspot-edit-dialog.component.ts @@ -0,0 +1,95 @@ +import { Component, Inject } from '@angular/core'; +import { MAT_DIALOG_DATA } from '@angular/material/dialog'; +import { Hotspot } from 'common/models/elements/element'; + +@Component({ + selector: 'aspect-hotspot-edit-dialog', + template: ` + <mat-dialog-content fxLayout="column" fxLayoutGap="10px"> + <div fxLayout="row" fxLayoutGap="10px"> + <mat-form-field appearance="fill" fxFlex="50"> + <mat-label>{{ 'hotspot.top' | translate }}</mat-label> + <input matInput type="number" min="0" + [(ngModel)]="newHotspot.top"> + </mat-form-field> + <mat-form-field appearance="fill" fxFlex="50"> + <mat-label>{{ 'hotspot.left' | translate }}</mat-label> + <input matInput type="number" min="0" + [(ngModel)]="newHotspot.left"> + </mat-form-field> + </div> + <div fxLayout="row" fxLayoutGap="10px"> + <mat-form-field appearance="fill" fxFlex="50"> + <mat-label>{{ 'hotspot.width' | translate }}</mat-label> + <input matInput type="number" min="0" + [(ngModel)]="newHotspot.width"> + </mat-form-field> + <mat-form-field appearance="fill" fxFlex="50"> + <mat-label>{{ 'hotspot.height' | translate }}</mat-label> + <input matInput type="number" min="0" + [(ngModel)]="newHotspot.height"> + </mat-form-field> + </div> + <div fxLayout="row" fxLayoutGap="10px" fxLayoutAlign="space-between center"> + <mat-radio-group [(ngModel)]="newHotspot.shape" fxFlex="50" fxLayout='column' fxLayoutGap="5px"> + <label>{{'hotspot.shape' | translate}}</label> + <mat-radio-button value='ellipse'>{{'hotspot.ellipse' | translate}}</mat-radio-button> + <mat-radio-button value='rect'>{{'hotspot.rect' | translate}}</mat-radio-button> + </mat-radio-group> + <mat-form-field appearance="fill" fxFlex="50"> + <mat-label>{{ 'hotspot.borderWidth' | translate }}</mat-label> + <input matInput type="number" min="0" + [(ngModel)]="newHotspot.borderWidth"> + </mat-form-field> + </div> + + + <div fxLayout="row" fxLayoutGap="10px"> + <mat-form-field appearance="fill" class="mdInput textsingleline"> + <mat-label>{{'hotspot.backgroundColor' | translate }}</mat-label> + <input matInput type="text" [(ngModel)]="newHotspot.backgroundColor"> + <button mat-icon-button matSuffix (click)="backgroundColorInput.click()"> + <mat-icon>edit</mat-icon> + </button> + </mat-form-field> + <input matInput type="color" hidden #backgroundColorInput [(ngModel)]="newHotspot.backgroundColor"> + + <mat-form-field appearance="fill" class="mdInput textsingleline"> + <mat-label>{{'hotspot.borderColor' | translate }}</mat-label> + <input matInput type="text" [(ngModel)]="newHotspot.borderColor"> + <button mat-icon-button matSuffix (click)="borderColorInput.click()"> + <mat-icon>edit</mat-icon> + </button> + </mat-form-field> + <input matInput type="color" hidden #borderColorInput [(ngModel)]="newHotspot.borderColor"> + </div> + + <div fxLayout="row" fxLayoutGap="10px"> + <mat-form-field appearance="fill" fxFlex="50"> + <mat-label>{{ 'hotspot.rotation' | translate }}</mat-label> + <input matInput type="number" min="0" + [(ngModel)]="newHotspot.rotation"> + </mat-form-field> + <div fxFlex="50"> + <mat-checkbox [checked]="newHotspot.value" + (change)="newHotspot.value = $event.checked"> + {{ 'hotspot.value' | translate }} + </mat-checkbox> + <mat-checkbox [checked]="newHotspot.readOnly" + (change)="newHotspot.readOnly = $event.checked"> + {{ 'hotspot.readOnly' | translate }} + </mat-checkbox> + </div> + </div> + </mat-dialog-content> + <mat-dialog-actions> + <button mat-button [mat-dialog-close]="newHotspot">{{'save' | translate }}</button> + <button mat-button mat-dialog-close>{{'cancel' | translate }}</button> + </mat-dialog-actions> + ` +}) +export class HotspotEditDialogComponent { + constructor(@Inject(MAT_DIALOG_DATA) public data: { hotspot: Hotspot }) { } + + newHotspot = { ...this.data.hotspot }; +} diff --git a/projects/editor/src/app/components/new-ui-element-panel/ui-element-toolbox.component.html b/projects/editor/src/app/components/new-ui-element-panel/ui-element-toolbox.component.html index 60f37bc8b39dec1702917ec9077f6afc79a97c3c..02d738d389b8e59fd70ed354112a2d949d59e71e 100644 --- a/projects/editor/src/app/components/new-ui-element-panel/ui-element-toolbox.component.html +++ b/projects/editor/src/app/components/new-ui-element-panel/ui-element-toolbox.component.html @@ -93,6 +93,11 @@ <mat-icon>linear_scale</mat-icon> {{'toolbox.slider' | translate }} </button> + <button mat-stroked-button (click)="addUIElement('hotspot-image')" + draggable="true" (dragstart)="$event.dataTransfer?.setData('elementType','hotspot-image')"> + <mat-icon>ads_click</mat-icon> + {{'toolbox.hotspot-image' | translate }} + </button> </mat-expansion-panel> <mat-expansion-panel> <mat-expansion-panel-header> diff --git a/projects/editor/src/app/components/properties-panel/element-properties-panel.component.ts b/projects/editor/src/app/components/properties-panel/element-properties-panel.component.ts index 323a3c0ac2c2f22e26633f5a8812b5d430163856..eb06f2b6ebd754f4788238c050e6ed2572f7b22e 100644 --- a/projects/editor/src/app/components/properties-panel/element-properties-panel.component.ts +++ b/projects/editor/src/app/components/properties-panel/element-properties-panel.component.ts @@ -6,7 +6,7 @@ import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { TranslateService } from '@ngx-translate/core'; import { MessageService } from 'common/services/message.service'; -import { TextLabel, UIElement } from 'common/models/elements/element'; +import { Hotspot, TextLabel, UIElement } from 'common/models/elements/element'; import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; import { UnitService } from '../../services/unit.service'; import { SelectionService } from '../../services/selection.service'; @@ -85,7 +85,7 @@ export class ElementPropertiesPanelComponent implements OnInit, OnDestroy { } updateModel(property: string, - value: string | number | boolean | string[] | + value: string | number | boolean | string[] | boolean[] | Hotspot[] | TextLabel | TextLabel[] | LikertRowElement[] | null, isInputValid: boolean | null = true): void { if (isInputValid) { diff --git a/projects/editor/src/app/components/properties-panel/hotspot-list-panel.component.ts b/projects/editor/src/app/components/properties-panel/hotspot-list-panel.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..3f7988c0cd2f98f5f0643bdea93e12e1590cabcc --- /dev/null +++ b/projects/editor/src/app/components/properties-panel/hotspot-list-panel.component.ts @@ -0,0 +1,55 @@ +import { + Component, EventEmitter, Input, Output +} from '@angular/core'; +import { CdkDragDrop } from '@angular/cdk/drag-drop/drag-events'; +import { Hotspot } from 'common/models/elements/element'; + +@Component({ + selector: 'aspect-hotspot-list-panel', + template: ` + <fieldset fxLayout="column"> + <legend>{{title | translate }}</legend> + <button mat-mini-fab matSuffix color="primary" [style.bottom.px]="3" fxFlexAlign="end" + (click)="addListItem();"> + <mat-icon>add</mat-icon> + </button> + + <div class="drop-list" cdkDropList [cdkDropListData]="itemList" + (cdkDropListDropped)="moveListValue($event)"> + <div *ngFor="let item of itemList; let i = index" cdkDrag + class="option-draggable" fxLayout="row"> + <div fxFlex fxFlexAlign="center">{{'hotspot.'+item.shape | translate}}({{i + 1}})</div> + <button mat-icon-button color="primary" + (click)="editItem.emit(i)"> + <mat-icon>build</mat-icon> + </button> + <button mat-icon-button color="primary" + (click)="removeListItem(i)"> + <mat-icon>clear</mat-icon> + </button> + </div> + </div> + </fieldset> + ` +}) +export class HotspotListPanelComponent { + @Input() title!: string; + @Input() textFieldLabel!: string; + @Input() itemList!: Hotspot[]; + @Output() addItem = new EventEmitter<void>(); + @Output() removeItem = new EventEmitter<number>(); + @Output() editItem = new EventEmitter<number>(); + @Output() changedItemOrder = new EventEmitter<{ previousIndex: number, currentIndex: number }>(); + + addListItem(): void { + this.addItem.emit(); + } + + removeListItem(itemIndex: number): void { + this.removeItem.emit(itemIndex); + } + + moveListValue(event: CdkDragDrop<Hotspot[]>): void { + this.changedItemOrder.emit({ previousIndex: event.previousIndex, currentIndex: event.currentIndex }); + } +} diff --git a/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.html b/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.html index 47d9db4b66ea83d88608dafeff028a1c045f8ed2..1c3ca293ce2a0d014a11860393b711410e5e6d51 100644 --- a/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.html +++ b/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.html @@ -31,6 +31,11 @@ (updateModel)="updateModel.emit($event)"> </aspect-options-field-set> + <aspect-hotspot-field-set *ngIf="combinedProperties.type === 'hotspot-image'" + [combinedProperties]="combinedProperties" + (updateModel)="updateModel.emit($event)"> + </aspect-hotspot-field-set> + <aspect-border-properties [combinedProperties]="combinedProperties" (updateModel)="updateModel.emit($event)"></aspect-border-properties> diff --git a/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.ts b/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.ts index b27890edeb611bc49369488b0df06317a8599dc7..3006f14ec8decaf14163944e2d09a27a26de71b3 100644 --- a/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.ts +++ b/projects/editor/src/app/components/properties-panel/model-properties-tab/element-model-properties.component.ts @@ -6,15 +6,15 @@ import { DomSanitizer } from '@angular/platform-browser'; import { CdkDragDrop } from '@angular/cdk/drag-drop/drag-events'; import { moveItemInArray } from '@angular/cdk/drag-drop'; import { - InputElementValue, TextLabel, TextImageLabel, UIElement + InputElementValue, TextLabel, TextImageLabel, UIElement, Hotspot } from 'common/models/elements/element'; import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; import { FileService } from 'common/services/file.service'; import { CombinedProperties } from 'editor/src/app/components/properties-panel/element-properties-panel.component'; +import { firstValueFrom } from 'rxjs'; import { UnitService } from '../../../services/unit.service'; import { SelectionService } from '../../../services/selection.service'; import { DialogService } from '../../../services/dialog.service'; -import { firstValueFrom } from 'rxjs'; @Component({ selector: 'aspect-element-model-properties-component', @@ -26,7 +26,7 @@ export class ElementModelPropertiesComponent { @Input() selectedElements: UIElement[] = []; @Output() updateModel = new EventEmitter<{ property: string; - value: InputElementValue | TextImageLabel[] | LikertRowElement[] | TextLabel[], + value: InputElementValue | TextImageLabel[] | LikertRowElement[] | TextLabel[] | Hotspot[] isInputValid?: boolean | null }>(); diff --git a/projects/editor/src/app/components/properties-panel/model-properties-tab/input-groups/hotspot-field-set.component.ts b/projects/editor/src/app/components/properties-panel/model-properties-tab/input-groups/hotspot-field-set.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..c3e6898687afc9d0a3310ec0364c68dd0530c6c1 --- /dev/null +++ b/projects/editor/src/app/components/properties-panel/model-properties-tab/input-groups/hotspot-field-set.component.ts @@ -0,0 +1,70 @@ +import { + Component, EventEmitter, Input, Output +} from '@angular/core'; +import { Hotspot } from 'common/models/elements/element'; +import { CombinedProperties } from 'editor/src/app/components/properties-panel/element-properties-panel.component'; +import { DialogService } from 'editor/src/app/services/dialog.service'; +import { moveItemInArray } from '@angular/cdk/drag-drop'; + +@Component({ + selector: 'aspect-hotspot-field-set', + template: ` + <aspect-hotspot-list-panel *ngIf="combinedProperties.value !== undefined" + [itemList]="$any(combinedProperties.value)" + [title]="'propertiesPanel.hotspots' | translate" + [textFieldLabel]="'propertiesPanel.newHotspot' | translate" + (changedItemOrder)="moveHotspot($event)" + (addItem)="addHotspot()" + (removeItem)="removeHotspot($event)" + (editItem)="editHotspot($event)"> + </aspect-hotspot-list-panel> + ` +}) +export class HotspotFieldSetComponent { + @Input() combinedProperties!: CombinedProperties; + @Output() updateModel = new EventEmitter<{ property: string; value: Hotspot[] }>(); + + constructor(private dialogService: DialogService) { } + + addHotspot(): void { + const newHotspot: Hotspot = { + top: 10, + left: 10, + width: 20, + height: 20, + shape: 'rect', + borderWidth: 1, + borderColor: '#000000', + backgroundColor: '#000000', + rotation: 0, + value: false, + readOnly: false + }; + (this.combinedProperties.value as Hotspot[]).push(newHotspot); + this.updateModel.emit({ property: 'value', value: this.combinedProperties.value as Hotspot[] }); + } + + removeHotspot(index: number): void { + const valueList = this.combinedProperties.value as Hotspot[]; + valueList.splice(index, 1); + this.updateModel.emit({ property: 'value', value: valueList }); + } + + moveHotspot(indices: { previousIndex: number, currentIndex: number }): void { + moveItemInArray(this.combinedProperties.value as Hotspot[], + indices.previousIndex, + indices.currentIndex); + this.updateModel.emit({ property: 'value', value: this.combinedProperties.value as Hotspot[] }); + } + + async editHotspot(index: number): Promise<void> { + const selectedOption = (this.combinedProperties.value as Hotspot[])[index]; + await this.dialogService.showHotspotEditDialog(selectedOption) + .subscribe((result: Hotspot) => { + if (result) { + (this.combinedProperties.value as Hotspot[])[index] = result; + this.updateModel.emit({ property: 'value', value: (this.combinedProperties.value as Hotspot[]) }); + } + }); + } +} diff --git a/projects/editor/src/app/services/dialog.service.ts b/projects/editor/src/app/services/dialog.service.ts index d7091d01c63e2d857bc9820642aa124866a4fe5f..e43faa42ae769c349a775becf9644ad305c42e08 100644 --- a/projects/editor/src/app/services/dialog.service.ts +++ b/projects/editor/src/app/services/dialog.service.ts @@ -4,13 +4,17 @@ import { MatDialog } from '@angular/material/dialog'; import { DragNDropValueObject, PlayerProperties, - TextImageLabel, Label + TextImageLabel, Label, Hotspot } from 'common/models/elements/element'; import { ClozeDocument } from 'common/models/elements/compound-elements/cloze/cloze'; import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; import { Section } from 'common/models/section'; import { SectionInsertDialogComponent } from 'editor/src/app/components/dialogs/section-insert-dialog.component'; import { LabelEditDialogComponent } from 'editor/src/app/components/dialogs/label-edit-dialog.component'; +import { + GeogebraAppDefinitionDialogComponent +} from 'editor/src/app/components/dialogs/geogebra-app-definition-dialog.component'; +import { HotspotEditDialogComponent } from 'editor/src/app/components/dialogs/hotspot-edit-dialog.component'; import { ConfirmationDialogComponent } from '../components/dialogs/confirmation-dialog.component'; import { TextEditDialogComponent } from '../components/dialogs/text-edit-dialog.component'; import { TextEditMultilineDialogComponent } from '../components/dialogs/text-edit-multiline-dialog.component'; @@ -18,9 +22,6 @@ import { RichTextEditDialogComponent } from '../components/dialogs/rich-text-edi import { PlayerEditDialogComponent } from '../components/dialogs/player-edit-dialog.component'; import { LikertRowEditDialogComponent } from '../components/dialogs/likert-row-edit-dialog.component'; import { DropListOptionEditDialogComponent } from '../components/dialogs/drop-list-option-edit-dialog.component'; -import { - GeogebraAppDefinitionDialogComponent -} from 'editor/src/app/components/dialogs/geogebra-app-definition-dialog.component'; @Injectable({ providedIn: 'root' @@ -103,6 +104,13 @@ export class DialogService { return dialogRef.afterClosed(); } + showHotspotEditDialog(hotspot: Hotspot): Observable<Hotspot> { + const dialogRef = this.dialog.open(HotspotEditDialogComponent, { + data: { hotspot } + }); + return dialogRef.afterClosed(); + } + showSectionInsertDialog(section: Section): Observable<Section> { const dialogRef = this.dialog.open(SectionInsertDialogComponent, { data: { section } diff --git a/projects/editor/src/app/services/id.service.ts b/projects/editor/src/app/services/id.service.ts index c6be2eef012839a5025179c8660ed94b9db5351e..d96646ed78652765a1ebfc4b1ac6a8dda31f3f05 100644 --- a/projects/editor/src/app/services/id.service.ts +++ b/projects/editor/src/app/services/id.service.ts @@ -18,6 +18,7 @@ export class IDService { checkbox: 0, dropdown: 0, radio: 0, + 'hotspot-image': 0, image: 0, audio: 0, video: 0, diff --git a/projects/editor/src/app/services/unit.service.ts b/projects/editor/src/app/services/unit.service.ts index e87713ae2a1de5f74f44efe89491c23133f07359..10ccfb36ef56183e00aa911f0ed56435845d00bf 100644 --- a/projects/editor/src/app/services/unit.service.ts +++ b/projects/editor/src/app/services/unit.service.ts @@ -11,7 +11,7 @@ import { CompoundElement, DragNDropValueObject, InputElement, InputElementValue, TextLabel, PlayerElement, PlayerProperties, PositionedUIElement, - UIElement, UIElementType, UIElementValue + UIElement, UIElementType, UIElementValue, Hotspot } from 'common/models/elements/element'; import { ClozeDocument, ClozeElement } from 'common/models/elements/compound-elements/cloze/cloze'; import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; @@ -109,9 +109,10 @@ export class UnitService { newElement.appDefinition = await firstValueFrom(this.dialogService.showGeogebraAppDefinitionDialog()); if (!newElement.appDefinition) return; // dialog canceled } - if (['audio', 'video', 'image'].includes(elementType)) { + if (['audio', 'video', 'image', 'hotspot-image'].includes(elementType)) { let mediaSrc = ''; switch (elementType) { + case 'hotspot-image': case 'image': mediaSrc = await FileService.loadImage(); break; @@ -225,7 +226,7 @@ export class UnitService { updateElementsProperty(elements: UIElement[], property: string, - value: InputElementValue | LikertRowElement[] | + value: InputElementValue | LikertRowElement[] | Hotspot[] | TextLabel | TextLabel[] | ClozeDocument | null): void { console.log('updateElementProperty', elements, property, value); elements.forEach(element => { diff --git a/projects/editor/src/assets/i18n/de.json b/projects/editor/src/assets/i18n/de.json index a1c997be93f0a1ee6aaa3972d77ddea27b1e8fa8..56ae888e3d6a8333cc175f9b125472aa6a5511cf 100644 --- a/projects/editor/src/assets/i18n/de.json +++ b/projects/editor/src/assets/i18n/de.json @@ -177,7 +177,24 @@ "showFullscreenButton": "Vollbild-Knopf anzeigen", "enableShiftDragZoom": "Bewegen und Zoom erlauben", "customToolBar": "Anpassung Werkzeugleiste", - "customToolbarHelp": "Hier kann definiert werden, welche Elemente auf der Werkzeigleiste angezeigt werden sollen. Für Details bitte die Dokumentation konsultieren." + "customToolbarHelp": "Hier kann definiert werden, welche Elemente auf der Werkzeigleiste angezeigt werden sollen. Für Details bitte die Dokumentation konsultieren.", + "hotspots": "Aktive Bereiche", + "newHotspot": "Neuer Bereich" + }, + "hotspot": { + "top": "Abstand von oben", + "left": "Abstand von links", + "width": "Bereichsbreite", + "height": "Bereichshöhe", + "borderWidth": "Rahmenbreite", + "borderColor": "Rahmenfarbe", + "shape": "Bereichsform", + "rect": "Rechteck", + "ellipse": "Ellipse", + "rotation": "Drehung", + "backgroundColor": "Füllfarbe (bei Aktivierung)", + "value": "Aktivierter Bereich", + "readOnly": "Schreibgeschützt" }, "player": { "autoStart": "Autostart (nicht für Tablets)", @@ -225,7 +242,8 @@ "slider": "Schieberegler", "spell-correct": "Wort korrigieren", "toggle-button": "Optionsfeld", - "geometry": "Geometrie" + "geometry": "Geometrie", + "hotspot-image": "Bildbereiche" }, "section-menu": { "height": "Höhe", diff --git a/projects/player/src/app/components/elements/element-group-selection/element-group-selection.component.ts b/projects/player/src/app/components/elements/element-group-selection/element-group-selection.component.ts index 687252a70892730c4583987431cde758e7bc90d7..e1b3034f699f0f9a98865e8416d1797e4aca7a13 100644 --- a/projects/player/src/app/components/elements/element-group-selection/element-group-selection.component.ts +++ b/projects/player/src/app/components/elements/element-group-selection/element-group-selection.component.ts @@ -1,6 +1,6 @@ import { Component, Input, OnInit } from '@angular/core'; -import { ElementGroup, ElementGroupName } from '../../../models/element-group'; import { UIElement, UIElementType } from 'common/models/elements/element'; +import { ElementGroup, ElementGroupName } from '../../../models/element-group'; @Component({ selector: 'aspect-element-group-selection', @@ -16,7 +16,7 @@ export class ElementGroupSelectionComponent implements OnInit { { name: 'mediaPlayerGroup', types: ['audio', 'video'] }, { name: 'inputGroup', - types: ['checkbox', 'slider', 'drop-list', 'radio', 'radio-group-images', 'dropdown'] + types: ['checkbox', 'slider', 'drop-list', 'radio', 'radio-group-images', 'dropdown', 'hotspot-image'] }, { name: 'compoundGroup', types: ['cloze', 'likert'] }, { name: 'textGroup', types: ['text'] }, diff --git a/projects/player/src/app/components/elements/input-group-element/input-group-element.component.html b/projects/player/src/app/components/elements/input-group-element/input-group-element.component.html index b4ad838ba9b10e449e9a10be7a9f2b95d98fb361..f7be78e173f1677ea3e7436b282ad0d7a877a61a 100644 --- a/projects/player/src/app/components/elements/input-group-element/input-group-element.component.html +++ b/projects/player/src/app/components/elements/input-group-element/input-group-element.component.html @@ -30,6 +30,12 @@ [parentForm]="form" [elementModel]="elementModel | cast: RadioButtonGroupComplexElement"> </aspect-radio-group-images> + <aspect-hotspot-image + *ngIf="elementModel.type === 'hotspot-image'" + #elementComponent + [parentForm]="form" + [elementModel]="elementModel | cast: HotspotImageElement"> + </aspect-hotspot-image> <aspect-dropdown *ngIf="elementModel.type === 'dropdown'" #elementComponent diff --git a/projects/player/src/app/components/elements/input-group-element/input-group-element.component.ts b/projects/player/src/app/components/elements/input-group-element/input-group-element.component.ts index c41bd5b0f4cb4d532f7fb1ae7cb4df5cdc6730c3..4a2942b4b3518bf4c9cecca1b5c394fccc056541 100644 --- a/projects/player/src/app/components/elements/input-group-element/input-group-element.component.ts +++ b/projects/player/src/app/components/elements/input-group-element/input-group-element.component.ts @@ -12,6 +12,7 @@ import { RadioButtonGroupElement } from 'common/models/elements/input-elements/r import { RadioButtonGroupComplexElement } from 'common/models/elements/input-elements/radio-button-group-complex'; import { DropdownElement } from 'common/models/elements/input-elements/dropdown'; import { InputElement } from 'common/models/elements/element'; +import { HotspotImageElement } from 'common/models/elements/input-elements/hotspot-image'; import { ValidationService } from '../../../services/validation.service'; import { ElementFormGroupDirective } from '../../../directives/element-form-group.directive'; import { ElementModelElementCodeMappingService } from '../../../services/element-model-element-code-mapping.service'; @@ -30,6 +31,7 @@ export class InputGroupElementComponent extends ElementFormGroupDirective implem RadioButtonGroupElement!: RadioButtonGroupElement; RadioButtonGroupComplexElement!: RadioButtonGroupComplexElement; DropdownElement!: DropdownElement; + HotspotImageElement!: HotspotImageElement; constructor( public unitStateService: UnitStateService, diff --git a/projects/player/src/app/components/page-scroll-button/page-scroll-button.component.html b/projects/player/src/app/components/page-scroll-button/page-scroll-button.component.html index ba9a935a287c536326fdb06da560d4f08b23210b..88def98ddac31c2f4b340aca63bf8678c6470d4f 100644 --- a/projects/player/src/app/components/page-scroll-button/page-scroll-button.component.html +++ b/projects/player/src/app/components/page-scroll-button/page-scroll-button.component.html @@ -1,5 +1,5 @@ <ng-content></ng-content> -<div fxLayout="column" fxLayoutAlign="center end" +<div fxLayout="column" fxLayoutAlign="center center" class="scroll-button-container"> <button *ngIf="isVisible.value" mat-fab diff --git a/projects/player/src/app/directives/element-form-group.directive.ts b/projects/player/src/app/directives/element-form-group.directive.ts index 79f238c7a172dcf8136c5e296ccd1ac14350e871..87158bf9b976b6a5568322a2cd4e6c0e09d3666c 100644 --- a/projects/player/src/app/directives/element-form-group.directive.ts +++ b/projects/player/src/app/directives/element-form-group.directive.ts @@ -11,6 +11,7 @@ import { VeronaSubscriptionService } from 'player/modules/verona/services/verona import { LogService } from 'player/modules/logging/services/log.service'; import { InputElement, InputElementValue } from 'common/models/elements/element'; import { SliderElement } from 'common/models/elements/input-elements/slider'; +import { hotspotImageRequiredValidator } from 'player/src/app/validators/hotspot-image-required.validator'; import { ValidationService } from '../services/validation.service'; import { ElementGroupDirective } from './element-group.directive'; import { ElementModelElementCodeMappingService } from '../services/element-model-element-code-mapping.service'; @@ -68,6 +69,9 @@ export abstract class ElementFormGroupDirective extends ElementGroupDirective im const validators: ValidatorFn[] = []; if (elementModel.required) { switch (elementModel.type) { + case 'hotspot-image': + validators.push(hotspotImageRequiredValidator()); + break; case 'checkbox': validators.push(Validators.requiredTrue); break; diff --git a/projects/player/src/app/services/element-model-element-code-mapping.service.spec.ts b/projects/player/src/app/services/element-model-element-code-mapping.service.spec.ts index b22e66659f2f8d1170302871c2c977688d946abb..4b4d5e95ef802919f00f1d726d56506bf6542130 100644 --- a/projects/player/src/app/services/element-model-element-code-mapping.service.spec.ts +++ b/projects/player/src/app/services/element-model-element-code-mapping.service.spec.ts @@ -12,6 +12,7 @@ import * as radioGroupImages_130 from 'test-data/element-models/radio-group-imag import * as toggleButton_130 from 'test-data/element-models/toggle-button_130.json'; import * as textArea_130 from 'test-data/element-models/text-area_130.json'; import * as spellCorrect_130 from 'test-data/element-models/spell-correct_130.json'; +import * as hotspotImage_135 from 'test-data/element-models/hotspot-image_135.json'; import * as dragNDropValues_01_130 from 'test-data/values/dragNDropValues_01_130.json'; import * as dragNDropValues_02_130 from 'test-data/values/dragNDropValues_02_130.json'; import { DropListElement } from 'common/models/elements/input-elements/drop-list'; @@ -31,7 +32,8 @@ import { RadioButtonGroupElement } from 'common/models/elements/input-elements/r import { RadioButtonGroupComplexElement } from 'common/models/elements/input-elements/radio-button-group-complex'; import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; import { ToggleButtonElement } from 'common/models/elements/compound-elements/cloze/cloze-child-elements/toggle-button'; -import { DragNDropValueObject } from 'common/models/elements/element'; +import { DragNDropValueObject, Hotspot } from 'common/models/elements/element'; +import { HotspotImageElement } from 'common/models/elements/input-elements/hotspot-image'; import { ElementModelElementCodeMappingService } from './element-model-element-code-mapping.service'; describe('ElementModelElementCodeMappingService', () => { @@ -99,6 +101,46 @@ describe('ElementModelElementCodeMappingService', () => { .toBe(null); }); + it('should map the value of a hotspot image elementModel to its elementCode value', () => { + const hotspots: Hotspot[] = [ + { + top: 10, + left: 10, + width: 20, + height: 20, + shape: 'rect', + borderWidth: 1, + borderColor: '#000000', + backgroundColor: '#000000', + rotation: 0, + readOnly: false, + value: true + }, + { + top: 10, + left: 10, + width: 20, + height: 20, + shape: 'rect', + borderWidth: 1, + borderColor: '#000000', + backgroundColor: '#000000', + rotation: 0, + readOnly: false, + value: false + } + ]; + expect(service.mapToElementCodeValue(hotspots, 'hotspot-image')) + .toEqual([true, false]); + }); + + it('should map the value of a hotspot image elementModel to its elementCode value', () => { + const hotspots: Hotspot[] = []; + + expect(service.mapToElementCodeValue(hotspots, 'hotspot-image')) + .toEqual([]); + }); + it('should map the value of a radio-group-images elementModel to its elementCode value', () => { for (let i = 0; i < 10; i++) { expect(service.mapToElementCodeValue(i, 'radio-group-images')) @@ -496,4 +538,44 @@ describe('ElementModelElementCodeMappingService', () => { expect(service.mapToElementModelValue(undefined, elementModel)) .toEqual(null); }); + + it('should map an elementCode value to hotspot-image elementModel value', () => { + const elementModel: HotspotImageElement = JSON.parse(JSON.stringify(hotspotImage_135)); + expect(service.mapToElementModelValue([true], elementModel)) + .toEqual([ + { + top: 10, + left: 10, + width: 20, + height: 20, + shape: 'rect', + borderWidth: 1, + borderColor: '#000000', + backgroundColor: '#000000', + rotation: 0, + readOnly: false, + value: true + } + ]); + }); + + it('should map an elementCode value to hotspot-image elementModel value', () => { + const elementModel: HotspotImageElement = JSON.parse(JSON.stringify(hotspotImage_135)); + expect(service.mapToElementModelValue([false], elementModel)) + .toEqual([ + { + top: 10, + left: 10, + width: 20, + height: 20, + shape: 'rect', + borderWidth: 1, + borderColor: '#000000', + backgroundColor: '#000000', + rotation: 0, + readOnly: false, + value: false + } + ]); + }); }); diff --git a/projects/player/src/app/services/element-model-element-code-mapping.service.ts b/projects/player/src/app/services/element-model-element-code-mapping.service.ts index d06fa96a1a9cb16263278a949413dd599e054f01..8a49d63cf55dafcf13754a153d810d458018fb5f 100644 --- a/projects/player/src/app/services/element-model-element-code-mapping.service.ts +++ b/projects/player/src/app/services/element-model-element-code-mapping.service.ts @@ -1,7 +1,7 @@ import { Injectable } from '@angular/core'; -import { TextMarkingService } from './text-marking.service'; import { DragNDropValueObject, + Hotspot, InputElement, InputElementValue, UIElement, @@ -12,6 +12,8 @@ import { AudioElement } from 'common/models/elements/media-elements/audio'; import { VideoElement } from 'common/models/elements/media-elements/video'; import { ImageElement } from 'common/models/elements/media-elements/image'; import { GeometryElement } from 'common/models/elements/geometry/geometry'; +import { HotspotImageElement } from 'common/models/elements/input-elements/hotspot-image'; +import { TextMarkingService } from './text-marking.service'; @Injectable({ providedIn: 'root' @@ -28,6 +30,11 @@ export class ElementModelElementCodeMappingService { return (elementCodeValue !== undefined) ? (elementCodeValue as string[]).map(id => this.getDragNDropValueObjectById(id)) as DragNDropValueObject[] : (elementModel as InputElement).value; + case 'hotspot-image': + return (elementCodeValue !== undefined) ? + (elementCodeValue as boolean[]) + .map((v, i) => ({ ...(elementModel as HotspotImageElement).value[i], value: v })) : + (elementModel as HotspotImageElement).value; case 'text': return (elementCodeValue !== undefined) ? TextMarkingService @@ -64,6 +71,8 @@ export class ElementModelElementCodeMappingService { case 'drop-list': case 'drop-list-simple': return (elementModelValue as DragNDropValueObject[]).map(object => object.id); + case 'hotspot-image': + return (elementModelValue as Hotspot[]).map(hotspot => hotspot.value); case 'text': return TextMarkingService.getMarkedTextIndices(elementModelValue as string); case 'radio': diff --git a/projects/player/src/app/validators/hotspot-image-required.validator.ts b/projects/player/src/app/validators/hotspot-image-required.validator.ts new file mode 100644 index 0000000000000000000000000000000000000000..3eead0cc70b4142b0ebd5c6a0d9a6d0583c9f269 --- /dev/null +++ b/projects/player/src/app/validators/hotspot-image-required.validator.ts @@ -0,0 +1,8 @@ +import { AbstractControl, ValidationErrors, ValidatorFn } from '@angular/forms'; +import { Hotspot } from 'common/models/elements/element'; + +export function hotspotImageRequiredValidator(): ValidatorFn { + return (control: AbstractControl): ValidationErrors | null => ( + (control.value as Hotspot[]).some(hotspot => hotspot.value) ? null : { required: true } + ); +} diff --git a/test-data/element-models/hotspot-image_135.json b/test-data/element-models/hotspot-image_135.json new file mode 100644 index 0000000000000000000000000000000000000000..4995ffdde83dc1e60a96d6a040a6a370ae42a321 --- /dev/null +++ b/test-data/element-models/hotspot-image_135.json @@ -0,0 +1,42 @@ +{ + "width": 180, + "height": 100, + "type": "hotspot-image", + "id": "hotspot-image_1", + "label": "Beschriftung", + "value": [ + { + "top": 10, + "left": 10, + "width": 20, + "height": 20, + "shape": "rect", + "borderWidth": 1, + "borderColor": "#000000", + "backgroundColor": "#000000", + "rotation": 0, + "readOnly": false, + "value": true + } + ], + "required": false, + "requiredWarnMessage": "Eingabe erforderlich", + "readOnly": false, + "src": "", + "position": { + "fixedSize": false, + "dynamicPositioning": true, + "xPosition": 0, + "yPosition": 0, + "useMinHeight": false, + "gridColumn": null, + "gridColumnRange": 1, + "gridRow": null, + "gridRowRange": 1, + "marginLeft": 0, + "marginRight": 0, + "marginTop": 0, + "marginBottom": 0, + "zIndex": 0 + } +}