diff --git a/projects/editor/src/app/app.module.ts b/projects/editor/src/app/app.module.ts index 019bd19aa3caac239887caf70167bb84dada2eb7..8c6fbc5d93e5d8f254c34bd9111c6a5ad96b24b7 100644 --- a/projects/editor/src/app/app.module.ts +++ b/projects/editor/src/app/app.module.ts @@ -130,6 +130,8 @@ import { RadioWizardDialogComponent } from 'editor/src/app/components/dialogs/wi import { TextWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/text.dialog.component'; import { LikertWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/likert.dialog.component'; import { InputWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/input.dialog.component'; +import { RadioImagesWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/radio2.dialog.component'; +import { Text2WizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/text2.dialog.component'; /** Custom options the configure the tooltip's default show/hide delays. */ export const myCustomTooltipDefaults: MatTooltipDefaultOptions = { @@ -194,7 +196,9 @@ export const myCustomTooltipDefaults: MatTooltipDefaultOptions = { GetValidAudioVideoIDsPipe, InputAssistancePropertiesComponent, RadioWizardDialogComponent, + RadioImagesWizardDialogComponent, TextWizardDialogComponent, + Text2WizardDialogComponent, LikertWizardDialogComponent, InputWizardDialogComponent ], diff --git a/projects/editor/src/app/components/dialogs/wizards/likert.dialog.component.ts b/projects/editor/src/app/components/dialogs/wizards/likert.dialog.component.ts index 72eaa7e614049b427579fc21984a458a43e46f99..a0f80e1267dd8abf1dbbfef94afc6de4f1860f53 100644 --- a/projects/editor/src/app/components/dialogs/wizards/likert.dialog.component.ts +++ b/projects/editor/src/app/components/dialogs/wizards/likert.dialog.component.ts @@ -1,18 +1,32 @@ import { Component } from '@angular/core'; +import { TextImageLabel } from 'common/models/elements/label-interfaces'; +import { LikertRowElement } from 'common/models/elements/compound-elements/likert/likert-row'; @Component({ selector: 'aspect-editor-likert-wizard-dialog', template: ` <div mat-dialog-title>Richtig/Falsch-Assistent</div> <div mat-dialog-content> - Text + <h3>Text</h3> <aspect-rich-text-editor [(content)]="text1"></aspect-rich-text-editor> - Satzanfang + <h3>Satzanfang</h3> <mat-form-field appearance="fill"> <textarea matInput type="text" [(ngModel)]="text2"></textarea> </mat-form-field> + <h3>Optionen</h3> + <aspect-option-list-panel class="options" [textFieldLabel]="'Neue Option'" + [itemList]="options" + [localMode]="true"> + </aspect-option-list-panel> + + <h3>Zeilen</h3> + <aspect-option-list-panel class="options" [textFieldLabel]="'Neue Zeile'" + [itemList]="rows" + [localMode]="true"> + </aspect-option-list-panel> + </div> <div mat-dialog-actions> @@ -22,6 +36,12 @@ import { Component } from '@angular/core'; ` }) export class LikertWizardDialogComponent { - text1: string = 'Testtext 1'; + text1: string = 'Hier steht die Frage der fünften Teilaufgabe mit Complex-Multiple-Choice (CMC). ' + + '[Optionentabelle] Bei einer Tabelle mit vielen' + + 'Zeilen kann ein Häkchen bei „haftende Kopfzeile“ gesetzt werden, damit die Spaltenüberschriften beim ' + + 'Scrollen sichtbar bleiben.'; + text2: string = 'Testtext 2'; + options: TextImageLabel[] = []; + rows: TextImageLabel[] = []; } diff --git a/projects/editor/src/app/components/dialogs/wizards/radio.dialog.component.ts b/projects/editor/src/app/components/dialogs/wizards/radio.dialog.component.ts index cd660876c88b14e12ae8d285a9758870e2f41089..481708a61372fa19a5056f3c91305429fe3ea4f5 100644 --- a/projects/editor/src/app/components/dialogs/wizards/radio.dialog.component.ts +++ b/projects/editor/src/app/components/dialogs/wizards/radio.dialog.component.ts @@ -12,7 +12,7 @@ import { Label } from 'common/models/elements/label-interfaces'; <mat-divider></mat-divider> - <h3>Satzanfang</h3> + <h3>Satzanfang (optional)</h3> <mat-form-field appearance="fill"> <textarea matInput type="text" [(ngModel)]="label2"></textarea> </mat-form-field> @@ -38,6 +38,6 @@ import { Label } from 'common/models/elements/label-interfaces'; }) export class RadioWizardDialogComponent { label1: string = 'Hier steht die Frage der Teilaufgabe mit Multiple Choice (MC).'; - label2: string = 'Hier könnte ein Satzanfang stehen ...'; + label2: string = ''; options: Label[] = []; } diff --git a/projects/editor/src/app/components/dialogs/wizards/radio2.dialog.component.ts b/projects/editor/src/app/components/dialogs/wizards/radio2.dialog.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..9511625d526b8d48bc179964f564f77d57d01d9f --- /dev/null +++ b/projects/editor/src/app/components/dialogs/wizards/radio2.dialog.component.ts @@ -0,0 +1,42 @@ +import { Component } from '@angular/core'; +import { TextImageLabel } from 'common/models/elements/label-interfaces'; + +@Component({ + selector: 'aspect-editor-radio-images-wizard-dialog', + template: ` + <div mat-dialog-title>Assistent: MC mit Bild</div> + <div mat-dialog-content> + <h3>Frage</h3> + <aspect-rich-text-editor class="input1" [(content)]="label1" [showReducedControls]="true"> + </aspect-rich-text-editor> + + <mat-divider></mat-divider> + + <h3>Optionen</h3> + <aspect-option-list-panel class="options" [textFieldLabel]="'Neue Option'" + [itemList]="options" + [showImageButton]="true" + [localMode]="true"> + </aspect-option-list-panel> + + <h3>Bilder je Zeile</h3> + <mat-form-field class="align-start"> + <input matInput type="number" min="1" maxlength="1" [(ngModel)]="itemsPerRow"> + </mat-form-field> + </div> + <div mat-dialog-actions> + <button mat-button [mat-dialog-close]="{ label1, options, itemsPerRow }">{{'confirm' | translate }}</button> + <button mat-button mat-dialog-close>{{'cancel' | translate }}</button> + </div> + `, + styles: ` + .mat-mdc-dialog-content {display: flex; flex-direction: column;} + /*.mat-mdc-dialog-content *:not(h3, mat-divider) {padding-left: 30px;}*/ + .input1 {min-height: 200px;} + ` +}) +export class RadioImagesWizardDialogComponent { + label1: string = 'Hier steht die Frage der Teilaufgabe mit Multiple Choice (MC).'; + options: TextImageLabel[] = []; + itemsPerRow: number = 4; +} diff --git a/projects/editor/src/app/components/dialogs/wizards/text.dialog.component.ts b/projects/editor/src/app/components/dialogs/wizards/text.dialog.component.ts index 6bba087b842b1773baa84efff76a747184062dd5..0c13d553e59cc07dad71fe3a123b6e734b68e2d3 100644 --- a/projects/editor/src/app/components/dialogs/wizards/text.dialog.component.ts +++ b/projects/editor/src/app/components/dialogs/wizards/text.dialog.component.ts @@ -48,7 +48,7 @@ import { Component } from '@angular/core'; }) export class TextWizardDialogComponent { text1: string = 'Lorem ipsum dolor sit amet'; - text2: string = 'Lorem ipsum dolor sit amet'; + text2: string = 'Platzhalter Quelle'; highlightableOrange: boolean = false; highlightableTurquoise: boolean = false; highlightableYellow: boolean = false; diff --git a/projects/editor/src/app/components/dialogs/wizards/text2.dialog.component.ts b/projects/editor/src/app/components/dialogs/wizards/text2.dialog.component.ts new file mode 100644 index 0000000000000000000000000000000000000000..abf279e1e0f72609937d716b8c140a34f4b7214c --- /dev/null +++ b/projects/editor/src/app/components/dialogs/wizards/text2.dialog.component.ts @@ -0,0 +1,34 @@ +import { Component } from '@angular/core'; + +@Component({ + selector: 'aspect-editor-text2-wizard-dialog', + template: ` + <div mat-dialog-title>Assistent: Markieren</div> + <div mat-dialog-content> + <h3>Text</h3> + <aspect-rich-text-editor class="input1" [(content)]="text1"></aspect-rich-text-editor> + + <h3>Tooltip</h3> + <mat-checkbox [(ngModel)]="showHelper" [style]="'margin-bottom: 20px;'"> + Anzeigen + </mat-checkbox> + </div> + <div mat-dialog-actions> + <button mat-button + [mat-dialog-close]="{ text1, showHelper }"> + {{'confirm' | translate }} + </button> + <button mat-button mat-dialog-close>{{'cancel' | translate }}</button> + </div> + `, + styles: ` + .mat-mdc-dialog-content {display: flex; flex-direction: column;} + .mat-mdc-dialog-content *:not(h3, mat-divider) {padding-left: 30px;} + h3 {text-decoration: underline;} + .input1 {min-height: 400px;} + ` +}) +export class Text2WizardDialogComponent { + text1: string = '[Frage] Markiere eine Stelle im Text, an der du das erkennst.'; + showHelper: boolean = true; +} 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 66096a09fe22392260bb68f0ef59dbab78f80d3f..2013fe5198f8ae6ff01d85e2f0578ac092e80461 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 @@ -211,14 +211,20 @@ <button mat-stroked-button (click)="applyTemplate('text')"> Stimulus: Text </button> + <button mat-stroked-button (click)="applyTemplate('text2')"> + Markieren + </button> <button mat-stroked-button (click)="applyTemplate('input')"> Antwortfeld(er) </button> <button mat-stroked-button (click)="applyTemplate('radio')"> MC mit Text </button> + <button mat-stroked-button (click)="applyTemplate('radio_images')"> + MC mit Bild + </button> <button mat-stroked-button (click)="applyTemplate('likert')"> - Likert + Richtig/Falsch-Aufgaben </button> </div> </mat-tab> diff --git a/projects/editor/src/app/services/template.service.ts b/projects/editor/src/app/services/template.service.ts index d2a3c22033adf1507dadad471d88ec69a115dc01..0ee8d251945126f9e41e7d385f59ffdeafcc4966 100644 --- a/projects/editor/src/app/services/template.service.ts +++ b/projects/editor/src/app/services/template.service.ts @@ -6,11 +6,14 @@ import { PositionProperties, PropertyGroupGenerators } from 'common/models/eleme import { Section, SectionProperties } from 'common/models/section'; import { UnitService } from 'editor/src/app/services/unit-services/unit.service'; import { IDService } from 'editor/src/app/services/id.service'; -import { TextLabel } from 'common/models/elements/label-interfaces'; +import { TextImageLabel, TextLabel } from 'common/models/elements/label-interfaces'; import { PositionedUIElement, UIElement, UIElementType } from 'common/models/elements/element'; import { TextWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/text.dialog.component'; import { LikertWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/likert.dialog.component'; import { InputWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/input.dialog.component'; +import { RadioImagesWizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/radio2.dialog.component'; +import { Text2WizardDialogComponent } from 'editor/src/app/components/dialogs/wizards/text2.dialog.component'; +import { LikertRowElement, LikertRowProperties } from 'common/models/elements/compound-elements/likert/likert-row'; @Injectable({ providedIn: 'root' @@ -41,6 +44,12 @@ export class TemplateService { if (result) resolve(this.createTextSection(result)); }); break; + case 'text2': + this.dialog.open(Text2WizardDialogComponent, {}) + .afterClosed().subscribe((result: { text1: string, showHelper: boolean }) => { + if (result) resolve(this.createText2Section(result)); + }); + break; case 'input': this.dialog.open(InputWizardDialogComponent, {}) .afterClosed().subscribe((result: { @@ -60,10 +69,16 @@ export class TemplateService { if (result) resolve(this.createRadioSection(result)); }); break; + case 'radio_images': + this.dialog.open(RadioImagesWizardDialogComponent, {}) + .afterClosed().subscribe((result: { label1: string, options: TextLabel[], itemsPerRow: number }) => { + if (result) resolve(this.createRadioImagesSection(result)); + }); + break; case 'likert': this.dialog.open(LikertWizardDialogComponent, {}) - .afterClosed().subscribe(() => { - resolve(this.createLikertSection()); + .afterClosed().subscribe((result: { text1: string, text2: string, options: TextImageLabel[], rows: TextImageLabel[] }) => { + if (result) resolve(this.createLikertSection(result)); }); break; default: @@ -94,6 +109,35 @@ export class TemplateService { } as SectionProperties); } + private createText2Section(config: { text1: string, showHelper: boolean }): Section { + const sectionElements: UIElement[] = [ + this.createElement( + 'text', + { gridRow: 1, gridColumn: 1, gridRowRange: 2, marginBottom: { value: 40, unit: 'px' } }, + { + text: config.text1 + }) + ]; + if (config.showHelper) { + sectionElements.push( + this.createElement( + 'button', + { gridRow: 1, gridColumn: 2 }, + { + imageSrc: '', + tooltipText: 'Drücke kurz auf den Knopf mit dem Stift. Drücke danach auf den Anfang' + + 'eines Wortes. Halte gedrückt und ziehe im Text so weit, wie du markieren möchtest.', + tooltipPosition: 'left' + }) + ); + } + return new Section({ + elements: sectionElements, + ...config.showHelper && { autoColumnSize: false }, + ...config.showHelper && { gridColumnSizes: [{ value: 1, unit: 'fr' }, { value: 45, unit: 'px' }] } + } as SectionProperties); + } + private createInputSection(config: { text: string, answerCount: number, useTextAreas: boolean, numbering: 'latin' | 'decimal' | 'bullets' | 'none', fieldLength: 'very-small' | 'small' | 'medium' | 'large', expectedCharsCount: number }): Section { @@ -118,7 +162,7 @@ export class TemplateService { sectionElements.push( this.createElement( 'text', - { gridRow: i + 2, gridColumn: 1, marginTop: { value: 16, unit: 'px' }}, + { gridRow: i + 2, gridColumn: 1, marginTop: { value: 16, unit: 'px' } }, { text: `${numberingChars[i]}` } ) ); @@ -197,9 +241,45 @@ export class TemplateService { } as SectionProperties); } - private createLikertSection(): Section { + private createRadioImagesSection(config: { label1: string, options: TextLabel[], itemsPerRow: number }): Section { return new Section({ - // elements: [] + elements: [ + this.createElement( + 'text', + { gridRow: 1, gridColumn: 1, marginBottom: { value: 5, unit: 'px' } }, + { text: config.label1 }), + this.createElement( + 'radio-group-images', + { gridRow: 2, gridColumn: 1 }, + { options: config.options, itemsPerRow: config.itemsPerRow }) + ] + } as SectionProperties); + } + + private createLikertSection(config: { text1: string, text2: string, options: TextImageLabel[], rows: TextImageLabel[] }): Section { + return new Section({ + elements: [ + this.createElement( + 'text', + { gridRow: 1, gridColumn: 1, marginBottom: { value: 10, unit: 'px' } }, + { text: config.text1 }), + this.createElement( + 'likert', + { gridRow: 2, gridColumn: 1 }, + { + options: config.options, + rows: config.rows.map(row => new LikertRowElement({ + id: this.idService.getAndRegisterNewID('likert-row'), + rowLabel: { + ...row + }, + columnCount: config.options.length + } as LikertRowProperties)), + label: config.text2, + stickyHeader: true, + firstColumnSizeRatio: 3 + }) + ] } as SectionProperties); }