UNPKG

@steinv/filippine

Version:

Filippine - acrostic word puzzle with Angular

117 lines 18 kB
import { Component, Input, ChangeDetectorRef, ChangeDetectionStrategy, Output, EventEmitter } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; export class FilippineComponent { constructor(_changeDetectorRef) { this._changeDetectorRef = _changeDetectorRef; this.answer = new EventEmitter(); /** * Emits when the form has been completed * returns false when answered wrongly, true when correctly answered */ this.completed = new EventEmitter(); this.filippineForm = new FormGroup({}); this.columns = 1; this.tiles = []; } ngOnInit() { /** * Calculate how big the grid should be */ const left = this.configuration.questions.reduce((acc, q) => acc = acc > q.answerPosition ? acc : q.answerPosition, 0); const right = this.configuration.questions.reduce((acc, q) => acc = acc > q.answerLength - q.answerPosition ? acc : q.answerLength - q.answerPosition, 0); this.columns = left + right; /** * map the configuration to the tiles on the grid * Creates a filippineForm that can validate each question */ let inputFieldCounter = 0; this.configuration.questions.map((m, i) => { const blankCellsLeft = left - m.answerPosition; const blankCellsRight = this.columns - (m.answerLength + blankCellsLeft); const inputCells = m.answerLength; this.tiles.push(new Array()); if (blankCellsLeft > 0) { this.tiles[i].push({ colspan: blankCellsLeft, question: false, highlight: false }); } const formQuestion = new FormGroup({}, this.rightAnswer(m)); for (let index = 0; index < inputCells; index++) { this.tiles[i].push({ index: inputFieldCounter, highlight: (index === m.answerPosition), colspan: 1, question: true, name: index, coordinates: i.toString() + ',' + (index + blankCellsLeft).toString() // coorindates row, column }); inputFieldCounter++; formQuestion.addControl(index.toString(), new FormControl('')); // m.answer.charAt(index) } this.filippineForm.addControl(i.toString(), formQuestion); if (blankCellsRight > 0) { this.tiles[i].push({ colspan: blankCellsRight, question: false, highlight: false }); } this.filippineForm.valueChanges.subscribe(() => { for (const group in this.filippineForm.controls) { const controlGroup = this.filippineForm.get(group); for (const field in controlGroup.controls) { const inputField = controlGroup.get(field); if (!inputField || !inputField.value) { return; } } } this.completed.emit(this.filippineForm.valid); }); }); } rightAnswer(question) { return (c) => { let formGroup = c; let invalid = false; let reply = ''; for (const field in formGroup.controls) { const control = formGroup.get(field); // only validate answer when entire group is filled if (!control || !control.value) { return null; } // concat control value upon reply string reply += control.value; // validate response if (question.answer && question.answer.length >= +field && control.value.toUpperCase() !== question.answer.charAt(+field).toUpperCase()) { invalid = true; } } // emit answer that client entered if (reply) { this.answer.emit({ question: question, answer: reply, }); } return invalid ? { invalid: true } : null; }; } ngAfterViewInit() { this._changeDetectorRef.detectChanges(); } } FilippineComponent.decorators = [ { type: Component, args: [{ selector: 'filippine', template: "\n<form [formGroup]=\"filippineForm\">\n <grid-list \n *ngFor=\"let row of tiles; let i = index\" \n [cols]=\"columns\" \n gutterSize=\"0\"\n [formGroupName]=\"i\"\n >\n <grid-tile \n *ngFor=\"let tile of row\"\n [colspan]=\"tile.colspan\"\n [ngClass]=\"{'input': tile.question, 'spacer': !tile.question, '': true, 'highlight': tile.highlight}\">\n <input \n *ngIf=\"tile.question === true\" \n AutoTab \n [id]=\"tile.coordinates\" \n [name]=\"tile.index\" \n type=\"text\" \n maxlength=\"1\" \n [formControlName]=\"tile.name\" \n />\n </grid-tile>\n </grid-list>\n</form>", changeDetection: ChangeDetectionStrategy.OnPush, styles: ["grid-tile{border:1px solid #000;box-shadow:0 0 0 1px #000;display:flex;position:relative}grid-tile input{background:transparent;border:none;height:0;margin:0;outline:none;padding:50% 0;text-align:center;text-transform:uppercase;width:100%}.spacer{background-color:#000}.input{background-color:#fff;border:1px solid #000;box-shadow:0 0 0 1px #000}.highlight{background-color:#ff0}grid-list.ng-invalid .input{border-color:red;box-shadow:0 0 0 1px red;z-index:1}"] },] } ]; FilippineComponent.ctorParameters = () => [ { type: ChangeDetectorRef } ]; FilippineComponent.propDecorators = { configuration: [{ type: Input }], answer: [{ type: Output }], completed: [{ type: Output }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"filippine.component.js","sourceRoot":"../../../projects/filippine/src/","sources":["lib/filippine/filippine.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Answer } from './../configuration';\nimport { Configuration, Question } from '../configuration';\nimport { Component, Input, OnInit, ChangeDetectorRef, ChangeDetectionStrategy, AfterViewInit, Output, EventEmitter } from '@angular/core';\nimport { AbstractControl, FormControl, FormGroup, ValidatorFn } from '@angular/forms';\n\n@Component({\n  selector: 'filippine',\n  templateUrl: './filippine.component.html',\n  styleUrls: ['./filippine.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FilippineComponent implements OnInit, AfterViewInit {\n\n  @Input()\n  public configuration!: Configuration;\n\n  @Output()\n  public answer = new EventEmitter<Answer>();\n\n  /**\n   * Emits when the form has been completed\n   * returns false when answered wrongly, true when correctly answered\n   */\n  @Output()\n  public completed = new EventEmitter<boolean>();\n\n  public filippineForm = new FormGroup({});\n  public columns: number = 1;\n  public tiles: Array<Array<any>> = [];\n\n  public constructor(\n    private readonly _changeDetectorRef: ChangeDetectorRef,\n  ) { }\n\n  public ngOnInit(): void {\n    /**\n     * Calculate how big the grid should be\n     */\n    const left = this.configuration.questions.reduce((acc: any, q: any) => acc = acc > q.answerPosition ? acc : q.answerPosition, 0);\n    const right = this.configuration.questions.reduce((acc: any, q: any) => acc = acc > q.answerLength - q.answerPosition ? acc : q.answerLength - q.answerPosition, 0);\n    this.columns = left + right;\n\n    /**\n     * map the configuration to the tiles on the grid\n     * Creates a filippineForm that can validate each question\n     */\n    let inputFieldCounter = 0;\n    this.configuration.questions.map(\n      (m: Question, i: number) => {\n        const blankCellsLeft = left - m.answerPosition;\n        const blankCellsRight = this.columns - (m.answerLength + blankCellsLeft);\n        const inputCells = m.answerLength;\n\n        this.tiles.push(new Array());\n        if (blankCellsLeft > 0) {\n          this.tiles[i].push({ colspan: blankCellsLeft, question: false, highlight: false });\n        }\n\n        const formQuestion = new FormGroup({}, this.rightAnswer(m));\n        for (let index = 0; index < inputCells; index++) {\n          this.tiles[i].push({\n            index: inputFieldCounter,\n            highlight: (index === m.answerPosition),        // highlighted box or not\n            colspan: 1,                                     // colspan for question is always 1\n            question: true,                                 // question or spacer\n            name: index,                                    // formControlName based on index\n            coordinates: i.toString() + ',' + (index + blankCellsLeft).toString() // coorindates row, column\n          });\n          inputFieldCounter++;\n          formQuestion.addControl(index.toString(), new FormControl(''));// m.answer.charAt(index)\n        }\n        this.filippineForm.addControl(i.toString(), formQuestion);\n\n        if (blankCellsRight > 0) {\n          this.tiles[i].push({ colspan: blankCellsRight, question: false, highlight: false });\n        }\n\n        this.filippineForm.valueChanges.subscribe(() => {\n          for (const group in this.filippineForm.controls) {\n            const controlGroup = this.filippineForm.get(group) as FormGroup;\n            for (const field in controlGroup.controls) {\n              const inputField = controlGroup.get(field);\n              if (!inputField || !inputField.value) {\n                return;\n              }\n            }\n          }\n\n          this.completed.emit(this.filippineForm.valid);\n        });\n      }\n    );\n  }\n\n  public rightAnswer(question: Question): ValidatorFn {\n    return (c: AbstractControl): { [key: string]: any } | null => {\n      let formGroup = c as FormGroup;\n\n      let invalid = false;\n      let reply = '';\n      for (const field in formGroup.controls) {\n        const control = formGroup.get(field);\n\n        // only validate answer when entire group is filled\n        if (!control || !control.value) {\n          return null;\n        }\n\n        // concat control value upon reply string\n        reply += control.value;\n\n        // validate response\n        if (question.answer &&\n          question.answer.length >= +field &&\n          control.value.toUpperCase() !== question.answer.charAt(+field).toUpperCase()\n        ) {\n          invalid = true;\n        }\n      }\n\n\n      // emit answer that client entered\n      if (reply) {\n        this.answer.emit({\n          question: question,\n          answer: reply,\n        });\n      }\n      return invalid ? { invalid: true } : null;\n    }\n  }\n\n  public ngAfterViewInit(): void {\n    this._changeDetectorRef.detectChanges();\n  }\n}\n\n"]}