From 4cab13fa5fa7d975c1f5e5a5264cbfc6f99f6fef Mon Sep 17 00:00:00 2001 From: rhenck <richard.henck@iqb.hu-berlin.de> Date: Wed, 1 Nov 2023 19:30:11 +0100 Subject: [PATCH] [editor] Refactor duplication logic --- docs/release-notes-editor.md | 16 +++++--- .../common/models/elements/button/button.ts | 9 ++-- .../cloze-child-elements/text-field-simple.ts | 6 ++- .../cloze-child-elements/toggle-button.ts | 6 ++- .../elements/compound-elements/cloze/cloze.ts | 10 +++-- .../compound-elements/likert/likert-row.ts | 4 ++ .../compound-elements/likert/likert.ts | 10 +++-- projects/common/models/elements/element.ts | 6 ++- .../common/models/elements/frame/frame.ts | 8 +++- .../models/elements/geometry/geometry.ts | 6 ++- .../elements/input-elements/checkbox.ts | 14 +++++-- .../elements/input-elements/drop-list.ts | 6 ++- .../elements/input-elements/dropdown.ts | 8 +++- .../elements/input-elements/hotspot-image.ts | 6 ++- .../elements/input-elements/math-field.ts | 8 +++- .../radio-button-group-complex.ts | 8 +++- .../input-elements/radio-button-group.ts | 8 +++- .../models/elements/input-elements/slider.ts | 8 +++- .../elements/input-elements/spell-correct.ts | 8 +++- .../elements/input-elements/text-area.ts | 8 +++- .../elements/input-elements/text-field.ts | 8 +++- .../models/elements/media-elements/audio.ts | 6 ++- .../models/elements/media-elements/image.ts | 6 ++- .../models/elements/media-elements/video.ts | 6 ++- projects/common/models/elements/text/text.ts | 8 +++- projects/common/models/section.ts | 12 +++--- .../element-properties-panel.component.ts | 6 +-- .../editor/src/app/services/unit.service.ts | 41 +++++++++++-------- 28 files changed, 180 insertions(+), 76 deletions(-) diff --git a/docs/release-notes-editor.md b/docs/release-notes-editor.md index 3933614c7..b5c2bc21b 100644 --- a/docs/release-notes-editor.md +++ b/docs/release-notes-editor.md @@ -1,17 +1,21 @@ Editor ====== -## 2.1.2 +## 2.1.1 ### Änderungen +- Duplizierte Elemente verlieren ihren Wert für Spalte/Zeile (Raster - dynamisches Layout) + Damit liegen duplizierte Elemente nicht mehr deckungsgleich übereinander. + (Gilt nicht für Elemente duplizierter Seitenabschnitte.) - "Zurücksetzen"-Knopf des Geogebra-Elements - Position oberhalb des Elements - mit Beschriftung "neu anlegen" - neues Icon - -## 2.1.1 deleted +### Fehlerbehebungen +- Sichtbarkeitseinstellungen duplizierter Abschnitte repariert + Einstellungen an Kopien werden nicht mehr auf das Ursprungselement angewendet. ## 2.1.0 ### Neue Funktionen -- Für die Bedingungen zur Sichtbarkeit von Abschnitten kann nun die +- Für die Bedingungen zur Sichtbarkeit von Abschnitten kann nun die logische Verknüpfung festgelegt werden: UND / ODER ### Fehlerbehebungen @@ -24,8 +28,8 @@ Editor - Optionsfelder mit Bild: Optionen richten sich wieder nach unten aus ## 2.0.2 -### Änderungen -- Passt das Einbinden des Symbols für den Radiergummi an die Sicherheitseinstellungen des Testcenters an +### Änderungen +- Passt das Einbinden des Symbols für den Radiergummi an die Sicherheitseinstellungen des Testcenters an ## 2.0.1 diff --git a/projects/common/models/elements/button/button.ts b/projects/common/models/elements/button/button.ts index e2d08d341..8a6cedab0 100644 --- a/projects/common/models/elements/button/button.ts +++ b/projects/common/models/elements/button/button.ts @@ -18,10 +18,10 @@ export class ButtonElement extends UIElement implements ButtonProperties { asLink: boolean = false; action: null | ButtonAction = null; actionParam: null | UnitNavParam | number | string = null; - styling: BasicStyles & BorderStyles; tooltipText: string = ''; tooltipPosition: TooltipPosition = 'below'; labelAlignment: 'super' | 'sub' | 'baseline' = 'baseline'; + styling: BasicStyles & BorderStyles; constructor(element?: ButtonProperties) { super(element); @@ -31,10 +31,10 @@ export class ButtonElement extends UIElement implements ButtonProperties { this.asLink = element.asLink; this.action = element.action; this.actionParam = element.actionParam; - this.styling = element.styling; this.tooltipText = element.tooltipText; this.tooltipPosition = element.tooltipPosition; this.labelAlignment = element.labelAlignment; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Button instantiation', element); @@ -47,7 +47,6 @@ export class ButtonElement extends UIElement implements ButtonProperties { if (element?.tooltipText !== undefined) this.tooltipText = element.tooltipText; if (element?.tooltipPosition !== undefined) this.tooltipPosition = element.tooltipPosition; if (element?.labelAlignment !== undefined) this.labelAlignment = element.labelAlignment; - this.position = PropertyGroupGenerators.generatePositionProps(element?.position); this.styling = { ...PropertyGroupGenerators.generateBasicStyleProps(element?.styling), ...PropertyGroupGenerators.generateBorderStylingProps(element?.styling) @@ -55,6 +54,10 @@ export class ButtonElement extends UIElement implements ButtonProperties { } } + getDuplicate(): ButtonElement { + return new ButtonElement(this); + } + getElementComponent(): Type<ElementComponent> { return ButtonComponent; } diff --git a/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/text-field-simple.ts b/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/text-field-simple.ts index abf491992..d4da4fe8b 100644 --- a/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/text-field-simple.ts +++ b/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/text-field-simple.ts @@ -38,7 +38,7 @@ export class TextFieldSimpleElement extends TextInputElement implements TextFiel this.pattern = element.pattern; this.patternWarnMessage = element.patternWarnMessage; this.clearable = element.clearable; - this.styling = element.styling; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at TextFieldSimple instantiation', element); @@ -83,6 +83,10 @@ export class TextFieldSimpleElement extends TextInputElement implements TextFiel getElementComponent(): Type<ElementComponent> { return TextFieldSimpleComponent; } + + getDuplicate(): TextFieldSimpleElement { + return new TextFieldSimpleElement(this); + } } export interface TextFieldSimpleProperties extends TextInputElementProperties { diff --git a/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/toggle-button.ts b/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/toggle-button.ts index 061504e93..3b327fb9b 100644 --- a/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/toggle-button.ts +++ b/projects/common/models/elements/compound-elements/cloze/cloze-child-elements/toggle-button.ts @@ -32,7 +32,7 @@ export class ToggleButtonElement extends InputElement implements ToggleButtonPro this.strikeOtherOptions = element.strikeOtherOptions; this.strikeSelectedOption = element.strikeSelectedOption; this.verticalOrientation = element.verticalOrientation; - this.styling = element.styling; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at ToggleButton instantiation', element); @@ -84,6 +84,10 @@ export class ToggleButtonElement extends InputElement implements ToggleButtonPro getNewOptionLabel(optionText: string): TextLabel { return UIElement.createOptionLabel(optionText) as TextLabel; } + + getDuplicate(): ToggleButtonElement { + return new ToggleButtonElement(this); + } } export interface ToggleButtonProperties extends InputElementProperties { diff --git a/projects/common/models/elements/compound-elements/cloze/cloze.ts b/projects/common/models/elements/compound-elements/cloze/cloze.ts index 0423a13cc..3f53cc7b6 100644 --- a/projects/common/models/elements/compound-elements/cloze/cloze.ts +++ b/projects/common/models/elements/compound-elements/cloze/cloze.ts @@ -35,10 +35,10 @@ export class ClozeElement extends CompoundElement implements PositionedUIElement super(element); if (element && isValid(element)) { this.columnCount = element.columnCount; - this.document = element.document; + this.document = { ...element.document }; this.instantiateChildElements(); - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Cloze instantiation', element); @@ -111,6 +111,10 @@ export class ClozeElement extends CompoundElement implements PositionedUIElement return ClozeComponent; } + getDuplicate(): ClozeElement { + return new ClozeElement(this); + } + getChildElements(): UIElement[] { return ClozeElement.getCustomNodes(this.document.content).map(el => el.attrs.model); } diff --git a/projects/common/models/elements/compound-elements/likert/likert-row.ts b/projects/common/models/elements/compound-elements/likert/likert-row.ts index be2750013..de636e5b8 100644 --- a/projects/common/models/elements/compound-elements/likert/likert-row.ts +++ b/projects/common/models/elements/compound-elements/likert/likert-row.ts @@ -66,6 +66,10 @@ export class LikertRowElement extends InputElement implements LikertRowPropertie getElementComponent(): Type<ElementComponent> { return LikertRadioButtonGroupComponent; } + + getDuplicate(): LikertRowElement { + return new LikertRowElement(this); + } } export interface LikertRowProperties extends InputElementProperties { diff --git a/projects/common/models/elements/compound-elements/likert/likert.ts b/projects/common/models/elements/compound-elements/likert/likert.ts index c2166d0a7..1cf31804e 100644 --- a/projects/common/models/elements/compound-elements/likert/likert.ts +++ b/projects/common/models/elements/compound-elements/likert/likert.ts @@ -31,14 +31,14 @@ export class LikertElement extends CompoundElement implements PositionedUIElemen constructor(element?: LikertProperties) { super(element); if (element && isValid(element)) { - this.options = element.options; + this.options = [...element.options]; this.firstColumnSizeRatio = element.firstColumnSizeRatio; this.rows = element.rows.map(row => new LikertRowElement(row)); this.label = element.label; this.label2 = element.label2; this.stickyHeader = element.stickyHeader; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Likert instantiation', element); @@ -107,6 +107,10 @@ export class LikertElement extends CompoundElement implements PositionedUIElemen getChildElements(): UIElement[] { return this.rows; } + + getDuplicate(): LikertElement { + return new LikertElement(this); + } } export interface LikertProperties extends UIElementProperties { diff --git a/projects/common/models/elements/element.ts b/projects/common/models/elements/element.ts index 6d1b826d4..ba5d7e565 100644 --- a/projects/common/models/elements/element.ts +++ b/projects/common/models/elements/element.ts @@ -71,8 +71,8 @@ export abstract class UIElement implements UIElementProperties { this.id = element.id; this.isRelevantForPresentationComplete = element.isRelevantForPresentationComplete; this.dimensions = element.dimensions; - this.position = element.position; - this.styling = element.styling; + if (element.position) this.position = { ...element.position }; + if (element.styling) this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at UIElement instantiation', element); @@ -129,6 +129,8 @@ export abstract class UIElement implements UIElementProperties { imgPosition: addImg ? 'above' : undefined }; } + + abstract getDuplicate(): UIElement; } export type InputElementValue = string[] | string | number | boolean | TextLabel[] | null | Hotspot[] | boolean[]; diff --git a/projects/common/models/elements/frame/frame.ts b/projects/common/models/elements/frame/frame.ts index 8284d7743..955eef5ed 100644 --- a/projects/common/models/elements/frame/frame.ts +++ b/projects/common/models/elements/frame/frame.ts @@ -26,8 +26,8 @@ export class FrameElement extends UIElement implements PositionedUIElement, Fram this.hasBorderBottom = element.hasBorderBottom; this.hasBorderLeft = element.hasBorderLeft; this.hasBorderRight = element.hasBorderRight; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Frame instantiation', element); @@ -53,6 +53,10 @@ export class FrameElement extends UIElement implements PositionedUIElement, Fram getElementComponent(): Type<ElementComponent> { return FrameComponent; } + + getDuplicate(): FrameElement { + return new FrameElement(this); + } } export interface FrameProperties extends UIElementProperties { diff --git a/projects/common/models/elements/geometry/geometry.ts b/projects/common/models/elements/geometry/geometry.ts index 22093c0a0..2af924257 100644 --- a/projects/common/models/elements/geometry/geometry.ts +++ b/projects/common/models/elements/geometry/geometry.ts @@ -35,7 +35,7 @@ export class GeometryElement extends UIElement implements PositionedUIElement, G this.showZoomButtons = element.showZoomButtons; this.showFullscreenButton = element.showFullscreenButton; this.customToolbar = element.customToolbar; - this.position = element.position; + this.position = { ...element.position }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Geometry instantiation', element); @@ -76,6 +76,10 @@ export class GeometryElement extends UIElement implements PositionedUIElement, G getElementComponent(): Type<ElementComponent> { return GeometryComponent; } + + getDuplicate(): GeometryElement { + return new GeometryElement(this); + } } export interface GeometryProperties extends UIElementProperties { diff --git a/projects/common/models/elements/input-elements/checkbox.ts b/projects/common/models/elements/input-elements/checkbox.ts index 4ed705803..dfcee227f 100644 --- a/projects/common/models/elements/input-elements/checkbox.ts +++ b/projects/common/models/elements/input-elements/checkbox.ts @@ -6,7 +6,7 @@ import { ElementComponent } from 'common/directives/element-component.directive' import { CheckboxComponent } from 'common/components/input-elements/checkbox.component'; import { AnswerScheme, AnswerSchemeValue } from 'common/models/elements/answer-scheme-interfaces'; import { - BasicStyles, PropertyGroupGenerators, PropertyGroupValidators + BasicStyles, PositionProperties, PropertyGroupGenerators, PropertyGroupValidators } from 'common/models/elements/property-group-interfaces'; import { environment } from 'common/environment'; import { InstantiationEror } from 'common/util/errors'; @@ -14,14 +14,15 @@ import { InstantiationEror } from 'common/util/errors'; export class CheckboxElement extends InputElement implements CheckboxProperties { type: UIElementType = 'checkbox'; crossOutChecked: boolean = false; + position: PositionProperties; styling: BasicStyles; constructor(element?: CheckboxProperties) { super(element); if (element && isValid(element)) { this.crossOutChecked = element.crossOutChecked; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Checkbox instantiation', element); @@ -31,10 +32,15 @@ export class CheckboxElement extends InputElement implements CheckboxProperties width: 215, ...element?.dimensions }); + this.position = PropertyGroupGenerators.generatePositionProps(element?.position); this.styling = PropertyGroupGenerators.generateBasicStyleProps(element?.styling); } } + getDuplicate(): CheckboxElement { + return new CheckboxElement(this); + } + hasAnswerScheme(): boolean { return Boolean(this.getAnswerScheme); } @@ -65,11 +71,13 @@ export class CheckboxElement extends InputElement implements CheckboxProperties export interface CheckboxProperties extends InputElementProperties { crossOutChecked: boolean; + position: PositionProperties; styling: BasicStyles; } function isValid(blueprint?: CheckboxProperties): boolean { if (!blueprint) return false; return blueprint.crossOutChecked !== undefined && + PropertyGroupValidators.isValidPosition(blueprint.position) && PropertyGroupValidators.isValidBasicStyles(blueprint.styling); } diff --git a/projects/common/models/elements/input-elements/drop-list.ts b/projects/common/models/elements/input-elements/drop-list.ts index d56bc2320..e8be7f772 100644 --- a/projects/common/models/elements/input-elements/drop-list.ts +++ b/projects/common/models/elements/input-elements/drop-list.ts @@ -38,7 +38,7 @@ export class DropListElement extends InputElement implements DropListProperties this.orientation = element.orientation; this.highlightReceivingDropList = element.highlightReceivingDropList; this.highlightReceivingDropListColor = element.highlightReceivingDropListColor; - this.styling = element.styling; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at DropList instantiation', element); @@ -71,6 +71,10 @@ export class DropListElement extends InputElement implements DropListProperties } } + getDuplicate(): DropListElement { + return new DropListElement(this); + } + /* Set originListID and originListIndex if applicable. */ setProperty(property: string, value: UIElementValue): void { super.setProperty(property, value); diff --git a/projects/common/models/elements/input-elements/dropdown.ts b/projects/common/models/elements/input-elements/dropdown.ts index 341f0e397..384c155fa 100644 --- a/projects/common/models/elements/input-elements/dropdown.ts +++ b/projects/common/models/elements/input-elements/dropdown.ts @@ -24,8 +24,8 @@ export class DropdownElement extends InputElement implements PositionedUIElement if (element && isValid(element)) { this.options = element.options; this.allowUnset = element.allowUnset; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Dropdown instantiation', element); @@ -42,6 +42,10 @@ export class DropdownElement extends InputElement implements PositionedUIElement } } + getDuplicate(): DropdownElement { + return new DropdownElement(this); + } + hasAnswerScheme(): boolean { return Boolean(this.getAnswerScheme); } diff --git a/projects/common/models/elements/input-elements/hotspot-image.ts b/projects/common/models/elements/input-elements/hotspot-image.ts index 4a18bf4b3..255dbf756 100644 --- a/projects/common/models/elements/input-elements/hotspot-image.ts +++ b/projects/common/models/elements/input-elements/hotspot-image.ts @@ -39,7 +39,7 @@ export class HotspotImageElement extends InputElement implements PositionedUIEle if (element && isValid(element)) { this.value = element.value; this.src = element.src; - this.position = element.position; + this.position = { ...element.position }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at HotspotImage instantiation', element); @@ -54,6 +54,10 @@ export class HotspotImageElement extends InputElement implements PositionedUIEle } } + getDuplicate(): HotspotImageElement { + return new HotspotImageElement(this); + } + hasAnswerScheme(): boolean { return Boolean(this.getAnswerScheme); } diff --git a/projects/common/models/elements/input-elements/math-field.ts b/projects/common/models/elements/input-elements/math-field.ts index 4713b930d..35af41021 100644 --- a/projects/common/models/elements/input-elements/math-field.ts +++ b/projects/common/models/elements/input-elements/math-field.ts @@ -23,8 +23,8 @@ export class MathFieldElement extends InputElement implements MathFieldPropertie super(element); if (element && isValid(element)) { this.enableModeSwitch = element.enableModeSwitch; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Mathfield instantiation', element); @@ -38,6 +38,10 @@ export class MathFieldElement extends InputElement implements MathFieldPropertie } } + getDuplicate(): MathFieldElement { + return new MathFieldElement(this); + } + hasAnswerScheme(): boolean { return Boolean(this.getAnswerScheme); } diff --git a/projects/common/models/elements/input-elements/radio-button-group-complex.ts b/projects/common/models/elements/input-elements/radio-button-group-complex.ts index b7d2facde..f0f36ddeb 100644 --- a/projects/common/models/elements/input-elements/radio-button-group-complex.ts +++ b/projects/common/models/elements/input-elements/radio-button-group-complex.ts @@ -25,8 +25,8 @@ export class RadioButtonGroupComplexElement extends InputElement if (element && isValid(element)) { this.options = element.options; this.itemsPerRow = element.itemsPerRow; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at RadioButtonGroupComplex instantiation', element); @@ -44,6 +44,10 @@ export class RadioButtonGroupComplexElement extends InputElement } } + getDuplicate(): RadioButtonGroupComplexElement { + return new RadioButtonGroupComplexElement(this); + } + hasAnswerScheme(): boolean { return Boolean(this.getAnswerScheme); } diff --git a/projects/common/models/elements/input-elements/radio-button-group.ts b/projects/common/models/elements/input-elements/radio-button-group.ts index baddbf3e2..705696534 100644 --- a/projects/common/models/elements/input-elements/radio-button-group.ts +++ b/projects/common/models/elements/input-elements/radio-button-group.ts @@ -29,8 +29,8 @@ export class RadioButtonGroupElement extends InputElement this.options = element.options; this.alignment = element.alignment; this.strikeOtherOptions = element.strikeOtherOptions; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at RadioButtonGroupElement instantiation', element); @@ -50,6 +50,10 @@ export class RadioButtonGroupElement extends InputElement } } + getDuplicate(): RadioButtonGroupElement { + return new RadioButtonGroupElement(this); + } + hasAnswerScheme(): boolean { return Boolean(this.getAnswerScheme); } diff --git a/projects/common/models/elements/input-elements/slider.ts b/projects/common/models/elements/input-elements/slider.ts index 7c4333c38..dc66dd07b 100644 --- a/projects/common/models/elements/input-elements/slider.ts +++ b/projects/common/models/elements/input-elements/slider.ts @@ -31,8 +31,8 @@ export class SliderElement extends InputElement implements PositionedUIElement, this.showValues = element.showValues; this.barStyle = element.barStyle; this.thumbLabel = element.thumbLabel; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Slider instantiation', element); @@ -75,6 +75,10 @@ export class SliderElement extends InputElement implements PositionedUIElement, getElementComponent(): Type<ElementComponent> { return SliderComponent; } + + getDuplicate(): SliderElement { + return new SliderElement(this); + } } export interface SliderProperties extends InputElementProperties { diff --git a/projects/common/models/elements/input-elements/spell-correct.ts b/projects/common/models/elements/input-elements/spell-correct.ts index 3b2d337a2..ba27df0b3 100644 --- a/projects/common/models/elements/input-elements/spell-correct.ts +++ b/projects/common/models/elements/input-elements/spell-correct.ts @@ -19,8 +19,8 @@ export class SpellCorrectElement extends TextInputElement implements PositionedU constructor(element?: SpellCorrectProperties) { super(element); if (element && isValid(element)) { - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at SpellCorrect instantiation', element); @@ -54,6 +54,10 @@ export class SpellCorrectElement extends TextInputElement implements PositionedU getElementComponent(): Type<ElementComponent> { return SpellCorrectComponent; } + + getDuplicate(): SpellCorrectElement { + return new SpellCorrectElement(this); + } } export interface SpellCorrectProperties extends TextInputElementProperties { diff --git a/projects/common/models/elements/input-elements/text-area.ts b/projects/common/models/elements/input-elements/text-area.ts index 2a3cb9369..4131c66d9 100644 --- a/projects/common/models/elements/input-elements/text-area.ts +++ b/projects/common/models/elements/input-elements/text-area.ts @@ -37,8 +37,8 @@ export class TextAreaElement extends TextInputElement implements PositionedUIEle this.expectedCharactersCount = element.expectedCharactersCount; this.hasReturnKey = element.hasReturnKey; this.hasKeyboardIcon = element.hasKeyboardIcon; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at TextArea instantiation', element); @@ -82,6 +82,10 @@ export class TextAreaElement extends TextInputElement implements PositionedUIEle getElementComponent(): Type<ElementComponent> { return TextAreaComponent; } + + getDuplicate(): TextAreaElement { + return new TextAreaElement(this); + } } export interface TextAreaProperties extends TextInputElementProperties { diff --git a/projects/common/models/elements/input-elements/text-field.ts b/projects/common/models/elements/input-elements/text-field.ts index ccd15aadc..4132d7187 100644 --- a/projects/common/models/elements/input-elements/text-field.ts +++ b/projects/common/models/elements/input-elements/text-field.ts @@ -41,8 +41,8 @@ export class TextFieldElement extends TextInputElement implements PositionedUIEl this.patternWarnMessage = element.patternWarnMessage; this.clearable = element.clearable; this.hasKeyboardIcon = element.hasKeyboardIcon; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at TextField instantiation', element); @@ -89,6 +89,10 @@ export class TextFieldElement extends TextInputElement implements PositionedUIEl getElementComponent(): Type<ElementComponent> { return TextFieldComponent; } + + getDuplicate(): TextFieldElement { + return new TextFieldElement(this); + } } export interface TextFieldProperties extends TextInputElementProperties { diff --git a/projects/common/models/elements/media-elements/audio.ts b/projects/common/models/elements/media-elements/audio.ts index c93d72581..d5dde5b28 100644 --- a/projects/common/models/elements/media-elements/audio.ts +++ b/projects/common/models/elements/media-elements/audio.ts @@ -19,7 +19,7 @@ export class AudioElement extends PlayerElement implements PositionedUIElement, super(element); if (element && isValid(element)) { this.src = element.src; - this.position = element.position; + this.position = { ...element.position }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Audio instantiation', element); @@ -34,6 +34,10 @@ export class AudioElement extends PlayerElement implements PositionedUIElement, } } + getDuplicate(): AudioElement { + return new AudioElement(this); + } + getElementComponent(): Type<ElementComponent> { return AudioComponent; } diff --git a/projects/common/models/elements/media-elements/image.ts b/projects/common/models/elements/media-elements/image.ts index 99e36f825..46731dc65 100644 --- a/projects/common/models/elements/media-elements/image.ts +++ b/projects/common/models/elements/media-elements/image.ts @@ -33,7 +33,7 @@ export class ImageElement extends UIElement implements PositionedUIElement, Imag this.magnifierSize = element.magnifierSize; this.magnifierZoom = element.magnifierZoom; this.magnifierUsed = element.magnifierUsed; - this.position = element.position; + this.position = { ...element.position }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Image instantiation', element); @@ -72,6 +72,10 @@ export class ImageElement extends UIElement implements PositionedUIElement, Imag valuesComplete: true }; } + + getDuplicate(): ImageElement { + return new ImageElement(this); + } } export interface ImageProperties extends UIElementProperties { diff --git a/projects/common/models/elements/media-elements/video.ts b/projects/common/models/elements/media-elements/video.ts index 6a6e5848e..d26ed22c3 100644 --- a/projects/common/models/elements/media-elements/video.ts +++ b/projects/common/models/elements/media-elements/video.ts @@ -21,7 +21,7 @@ export class VideoElement extends PlayerElement implements PositionedUIElement, if (element && isValid(element)) { this.src = element.src; this.scale = element.scale; - this.position = element.position; + this.position = { ...element.position }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Video instantiation', element); @@ -40,6 +40,10 @@ export class VideoElement extends PlayerElement implements PositionedUIElement, getElementComponent(): Type<ElementComponent> { return VideoComponent; } + + getDuplicate(): VideoElement { + return new VideoElement(this); + } } export interface VideoProperties extends PlayerElementBlueprint { diff --git a/projects/common/models/elements/text/text.ts b/projects/common/models/elements/text/text.ts index 32970bba8..b33abbdde 100644 --- a/projects/common/models/elements/text/text.ts +++ b/projects/common/models/elements/text/text.ts @@ -35,8 +35,8 @@ export class TextElement extends UIElement implements PositionedUIElement, TextP this.highlightableYellow = element.highlightableYellow; this.hasSelectionPopup = element.hasSelectionPopup; this.columnCount = element.columnCount; - this.position = element.position; - this.styling = element.styling; + this.position = { ...element.position }; + this.styling = { ...element.styling }; } else { if (environment.strictInstantiation) { throw new InstantiationEror('Error at Text instantiation', element); @@ -59,6 +59,10 @@ export class TextElement extends UIElement implements PositionedUIElement, TextP } } + getDuplicate(): TextElement { + return new TextElement(this); + } + private isHighlightable(): boolean { return this.highlightableYellow || this.highlightableTurquoise || diff --git a/projects/common/models/section.ts b/projects/common/models/section.ts index b460deef6..d888d1992 100644 --- a/projects/common/models/section.ts +++ b/projects/common/models/section.ts @@ -37,13 +37,13 @@ export class Section { this.dynamicPositioning = section.dynamicPositioning; this.autoColumnSize = section.autoColumnSize; this.autoRowSize = section.autoRowSize; - this.gridColumnSizes = section.gridColumnSizes; - this.gridRowSizes = section.gridRowSizes; + this.gridColumnSizes = { ...section.gridColumnSizes }; + this.gridRowSizes = { ...section.gridRowSizes }; this.visibilityDelay = section.visibilityDelay; this.animatedVisibility = section.animatedVisibility; this.enableReHide = section.enableReHide; this.logicalConnectiveOfRules = section.logicalConnectiveOfRules; - this.visibilityRules = section.visibilityRules; + this.visibilityRules = section.visibilityRules.map(rule => ({ ...rule })); this.elements = section.elements .map(element => ElementFactory.createElement(element)) as PositionedUIElement[]; } else { @@ -55,13 +55,13 @@ export class Section { if (section?.dynamicPositioning !== undefined) this.dynamicPositioning = section.dynamicPositioning; if (section?.autoColumnSize !== undefined) this.autoColumnSize = section.autoColumnSize; if (section?.autoRowSize !== undefined) this.autoRowSize = section.autoRowSize; - if (section?.gridColumnSizes !== undefined) this.gridColumnSizes = section.gridColumnSizes; - if (section?.gridRowSizes !== undefined) this.gridRowSizes = section.gridRowSizes; + if (section?.gridColumnSizes !== undefined) this.gridColumnSizes = { ...section.gridColumnSizes }; + if (section?.gridRowSizes !== undefined) this.gridRowSizes = { ...section.gridRowSizes }; if (section?.visibilityDelay !== undefined) this.visibilityDelay = section.visibilityDelay; if (section?.animatedVisibility !== undefined) this.animatedVisibility = section.animatedVisibility; if (section?.enableReHide !== undefined) this.enableReHide = section.enableReHide; if (section?.logicalConnectiveOfRules !== undefined) this.logicalConnectiveOfRules = section.logicalConnectiveOfRules; - if (section?.visibilityRules !== undefined) this.visibilityRules = section.visibilityRules; + if (section?.visibilityRules !== undefined) this.visibilityRules = section.visibilityRules.map(rule => ({ ...rule })); this.elements = section?.elements !== undefined ? section.elements.map(element => ElementFactory.createElement(element)) as PositionedUIElement[] : []; 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 cf9ec2ad4..47f770c32 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 @@ -125,11 +125,7 @@ export class ElementPropertiesPanelComponent implements OnInit, OnDestroy { } duplicateElement(): void { - this.unitService.duplicateElementsInSection( - this.selectedElements, - this.selectionService.selectedPageIndex, - this.selectionService.selectedPageSectionIndex - ); + this.unitService.duplicateSelectedElements(); } ngOnDestroy(): void { diff --git a/projects/editor/src/app/services/unit.service.ts b/projects/editor/src/app/services/unit.service.ts index 6b863ee96..dc4c86438 100644 --- a/projects/editor/src/app/services/unit.service.ts +++ b/projects/editor/src/app/services/unit.service.ts @@ -7,7 +7,11 @@ import { FileService } from 'common/services/file.service'; import { MessageService } from 'common/services/message.service'; import { ArrayUtils } from 'common/util/array'; import { Unit, UnitProperties } from 'common/models/unit'; -import { PlayerProperties, PositionProperties } from 'common/models/elements/property-group-interfaces'; +import { + PlayerProperties, + PositionProperties, + PropertyGroupGenerators +} from 'common/models/elements/property-group-interfaces'; import { DragNDropValueObject, TextLabel } from 'common/models/elements/label-interfaces'; import { Hotspot } from 'common/models/elements/input-elements/hotspot-image'; import { @@ -209,6 +213,7 @@ export class UnitService { section.addElement(ElementFactory.createElement({ type: elementType, + position: PropertyGroupGenerators.generatePositionProps(newElementProperties.position), ...newElementProperties, id: this.idService.getAndRegisterNewID(elementType) }) as PositionedUIElement); @@ -274,33 +279,35 @@ export class UnitService { this.veronaApiService.sendVoeDefinitionChangedNotification(this.unit); } - duplicateElementsInSection(elements: UIElement[], pageIndex: number, sectionIndex: number): void { - const section = this.unit.pages[pageIndex].sections[sectionIndex]; - elements.forEach((element: UIElement) => { - section.elements.push(this.duplicateElement(element) as PositionedUIElement); + duplicateSelectedElements(): void { + const selectedSection = + this.unit.pages[this.selectionService.selectedPageIndex].sections[this.selectionService.selectedPageSectionIndex]; + this.selectionService.getSelectedElements().forEach((element: UIElement) => { + selectedSection.elements.push(this.duplicateElement(element, true) as PositionedUIElement); }); this.veronaApiService.sendVoeDefinitionChangedNotification(this.unit); } - private duplicateElement(element: UIElement): UIElement { - const newElement = ElementFactory.createElement(JSON.parse(JSON.stringify(element))); - newElement.id = this.idService.getAndRegisterNewID(newElement.type); + /* - Also changes position of the element to not cover copied element. + - Also changes and registers all copied IDs. */ + private duplicateElement(element: UIElement, adjustPosition: boolean = false): UIElement { + const newElement = element.getDuplicate(); - if (newElement.position) { + if (newElement.position && adjustPosition) { newElement.position.xPosition += 10; newElement.position.yPosition += 10; + newElement.position.gridRow = null; + newElement.position.gridColumn = null; } - if (newElement.type === 'likert') { // replace row Ids with fresh ones (likert) - (newElement.rows as LikertRowElement[]).forEach((rowObject: { id: string }) => { - rowObject.id = this.idService.getAndRegisterNewID('likert-row'); - }); - } - if (newElement.type === 'cloze') { - ClozeElement.getDocumentChildElements((newElement as ClozeElement).document).forEach(clozeChild => { - clozeChild.id = this.idService.getAndRegisterNewID(clozeChild.type); + newElement.id = this.idService.getAndRegisterNewID(newElement.type); + if (newElement instanceof CompoundElement) { + newElement.getChildElements().forEach((child: UIElement) => { + child.id = this.idService.getAndRegisterNewID(child.type); }); } + + // Special care with DropLists as they are no CompoundElement yet still have children with IDs if (newElement.type === 'drop-list') { (newElement.value as DragNDropValueObject[]).forEach(valueObject => { valueObject.id = this.idService.getAndRegisterNewID('value'); -- GitLab