UNPKG

@knora/action

Version:
286 lines 32.5 kB
import * as tslib_1 from "tslib"; import { Component, ElementRef, EventEmitter, Input, Output, ViewChild } from '@angular/core'; import { FormBuilder, FormControl, FormGroup } from '@angular/forms'; import { MatMenuTrigger } from '@angular/material'; let StringLiteralInputComponent = class StringLiteralInputComponent { constructor(_fb) { this._fb = _fb; this.languages = ['de', 'fr', 'it', 'en']; /** * Optional placeholder for the input field e.g. Label * * @param {string} [placeholder='Label'] */ this.placeholder = 'Label'; /** * Optional form field input type: textarea? set to true for textarea * otherwise it's a simple (short) input field * * @param {boolean} [textarea=false] */ this.textarea = false; /** * Optional form field value of type StringLiteral[] * * @param {StringLiteral[]} value */ this.value = []; /** * Optional disable the input field in case of no right to edit the field/value * * @param {boolean}: [disabled=false] */ this.disabled = false; /** * The readonly attribute specifies whether the control may be modified by the user. * * @param {boolean}: [readonly=false] */ this.readonly = false; /** * Returns (output) an array of StringLiteral on any change on the input field. * * @emits {StringLiteral[]} dataChanged */ this.dataChanged = new EventEmitter(); /** * Returns (output) true when the field was touched. This can be used to validate data, e.g. in case a value is required * * @emits {boolean} touched */ this.touched = new EventEmitter(); /** * Returns true when a user press ENTER. This can be used to submit data in the parent component. * * * @emits {boolean} enter */ this.enter = new EventEmitter(); // set selected language, if it's not defined yet if (!this.language) { if (localStorage.getItem('session') !== null) { // get language from the logged-in user profile data this.language = JSON.parse(localStorage.getItem('session')).user.lang; } else { // get default language from browser this.language = navigator.language.substr(0, 2); } } // does the defined language exists in our supported languages list? if (this.languages.indexOf(this.language) === -1) { // if not, select the first language from our list of supported languages this.language = this.languages[0]; } } ngOnInit() { // if (this.placeholder.length > 0) { // this.placeholder += ' (' + this.language + ')'; // } // reset stringLiterals if they have empty values this.resetValues(); // build the form this.form = this._fb.group({ text: new FormControl({ value: '', disabled: this.disabled }, { // updateOn: 'blur' }) }); // update values on form change this.form.valueChanges.subscribe(data => this.onValueChanged()); // get value from stringLiterals const val = this.getValueFromStringLiteral(this.language); this.updateFormField(val); } /** * @ignore * * emit data to parent on any change on the input field */ onValueChanged() { if (!this.form) { return; } const form = this.form; const control = form.get('text'); this.touched.emit(control && control.dirty); // if (control && control.dirty) { // console.warn('control dirty'); // } this.updateStringLiterals(this.language, this.form.controls['text'].value); this.dataChanged.emit(this.value); } toggleAll() { // TODO: open/show all languages with their values } /** * @ignore * * Set the language after selecting; This updates the array of StringLiterals: adds item with the selected language if it doesn't exist */ setLanguage(lang) { if (this.language === lang) { // console.warn('DO NOTHING! this language was already selected'); } else { // clean stringLIteral value for previous language, if text field is empty this.updateStringLiterals(this.language, this.form.controls['text'].value); this.language = lang; // update form field value / reset in case of no value const val = this.getValueFromStringLiteral(lang); this.updateFormField(val); } } /** * @ignore * * Switch focus to input field after selecting a language */ switchFocus() { // close the menu if (!this.textarea && this.btnToSelectLanguage && this.btnToSelectLanguage.menuOpen) { this.btnToSelectLanguage.closeMenu(); } if (!this.disabled) { this.form.controls['text'].enable(); this.textInput.nativeElement.focus(); } } /** * @ignore * * Set the value in the input field */ updateFormField(value) { if (!value) { value = ''; } this.form.controls['text'].setValue(value); } /** * @ignore * * Update the array of StringLiterals depending on value / empty value add or remove item from array. */ updateStringLiterals(lang, value) { const index = this.value.findIndex(i => i.language === lang); if (index > -1 && this.value[index].value.length > 0) { // value is not empty and exists in list of stringLiterals // console.log('update existing value for ' + lang + ' on position ' + index); this.value[index].value = value; } if ((!value || value.length === 0) && index > -1) { // value is empty: delete stringLiteral item for this language // console.log('delete empty value for ' + lang + ' on position ' + index); this.value.splice(index, 1); } if (index < 0 && value) { // value doesn't exist in stringLiterals: add one // console.log('add new value (' + value + ') for ' + lang); const newValue = { value: value, language: lang }; this.value.push(newValue); } } /** * @ignore * * In case of strange array of StringLiterals, this method will reset to a API-conform array. This means an array without empty values. */ resetValues() { const length = this.value.length; if (length > 0) { let index = length - 1; while (index >= 0) { // remove items with empty value if (!this.value[index].value.length) { this.value.splice(index, 1); } index--; } // does an item for selected lanuage exists if (this.value.findIndex(i => i.language === this.language) === -1) { this.language = this.value[0].language; } } else { this.value = []; } } /** * @ignore * * Get the value from array of StringLiterals for the selected language */ getValueFromStringLiteral(lang) { // console.log('existing value in', this.value); // get index for this language const index = this.value.findIndex(i => i.language === lang); if (this.value[index] && this.value[index].value.length > 0) { return this.value[index].value; } else { return undefined; } } }; StringLiteralInputComponent.ctorParameters = () => [ { type: FormBuilder } ]; tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", String) ], StringLiteralInputComponent.prototype, "placeholder", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", String) ], StringLiteralInputComponent.prototype, "language", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", Boolean) ], StringLiteralInputComponent.prototype, "textarea", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", Array) ], StringLiteralInputComponent.prototype, "value", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", Boolean) ], StringLiteralInputComponent.prototype, "disabled", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", Boolean) ], StringLiteralInputComponent.prototype, "readonly", void 0); tslib_1.__decorate([ Output(), tslib_1.__metadata("design:type", EventEmitter) ], StringLiteralInputComponent.prototype, "dataChanged", void 0); tslib_1.__decorate([ Output(), tslib_1.__metadata("design:type", EventEmitter) ], StringLiteralInputComponent.prototype, "touched", void 0); tslib_1.__decorate([ Output(), tslib_1.__metadata("design:type", EventEmitter) ], StringLiteralInputComponent.prototype, "enter", void 0); tslib_1.__decorate([ ViewChild('textInput', { static: false }), tslib_1.__metadata("design:type", ElementRef) ], StringLiteralInputComponent.prototype, "textInput", void 0); tslib_1.__decorate([ ViewChild('btnToSelectLanguage', { static: false }), tslib_1.__metadata("design:type", MatMenuTrigger) ], StringLiteralInputComponent.prototype, "btnToSelectLanguage", void 0); StringLiteralInputComponent = tslib_1.__decorate([ Component({ selector: 'kui-string-literal-input', template: "<form [formGroup]=\"form\">\n\n <!-- default input element -->\n <mat-form-field *ngIf=\"!textarea\" class=\"string-literal short-text\">\n\n <!-- select: button to select language -->\n <button mat-button type=\"button\" matPrefix class=\"select-lang\" [matMenuTriggerFor]=\"selectLanguage\"\n #btnToSelectLanguage=\"matMenuTrigger\" (click)=\"form.controls['text'].disable()\">\n <span class=\"label\">{{language}}</span>\n <mat-icon class=\"icon\" matSuffix>keyboard_arrow_down</mat-icon>\n </button>\n\n <!-- select: menu with list of languages -->\n <mat-menu #selectLanguage=\"matMenu\">\n <button mat-menu-item type=\"button\" *ngFor=\"let lang of languages\"\n (click)=\"setLanguage(lang);switchFocus()\">\n <span [class.existing-value]=\"getValueFromStringLiteral(lang)\">{{lang}}</span>\n </button>\n <!-- TODO / QUESTION: should we support a show all button, to display values for all languages?\n <mat-divider></mat-divider>\n <button mat-menu-item type=\"button\" (click)=\"toggleAll()\">\n <span>Show values for all languages</span>\n </button>\n -->\n </mat-menu>\n\n <!-- input field-->\n <input matInput [placeholder]=\"placeholder\" [formControl]=\"form.controls['text']\" #textInput\n [readonly]=\"readonly\" (keyup.enter)=\"enter.emit(true)\">\n </mat-form-field>\n\n <!-- input element type is textarea -->\n <div *ngIf=\"textarea\" class=\"string-literal long-text\">\n <!-- button toggle group: buttons to select language -->\n <mat-button-toggle-group matPrefix #group=\"matButtonToggleGroup\" vertical class=\"string-literal-select-lang\">\n <mat-button-toggle *ngFor=\"let lang of languages\" (click)=\"setLanguage(lang);switchFocus()\"\n [checked]=\"lang === language\">\n <span [class.existing-value]=\"getValueFromStringLiteral(lang)\">{{lang}}</span>\n </mat-button-toggle>\n </mat-button-toggle-group>\n <mat-form-field class=\"string-literal-textarea\">\n <!-- textarea -->\n <textarea matInput [placeholder]=\"placeholder\" [formControl]=\"form.controls['text']\" #textInput\n [readonly]=\"readonly\"></textarea>\n </mat-form-field>\n </div>\n\n</form>\n", styles: [".mat-form-field{width:100%!important}.existing-value{font-weight:700}"] }), tslib_1.__metadata("design:paramtypes", [FormBuilder]) ], StringLiteralInputComponent); export { StringLiteralInputComponent }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"string-literal-input.component.js","sourceRoot":"ng://@knora/action/","sources":["lib/string-literal-input/string-literal-input.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, Input, OnInit, Output, ViewChild } from '@angular/core';\nimport { FormBuilder, FormControl, FormGroup } from '@angular/forms';\nimport { MatMenuTrigger } from '@angular/material';\nimport { StringLiteral } from '@knora/api';\n\n@Component({\n    selector: 'kui-string-literal-input',\n    templateUrl: './string-literal-input.component.html',\n    styleUrls: ['./string-literal-input.component.scss']\n})\nexport class StringLiteralInputComponent implements OnInit {\n\n\n    languages: string[] = ['de', 'fr', 'it', 'en'];\n\n    /**\n     * Optional placeholder for the input field e.g. Label\n     *\n     * @param  {string} [placeholder='Label']\n     */\n    @Input() placeholder?: string = 'Label';\n\n    /**\n     * Optional predefined (selected) language: en, de, it, fr, etc.\n     *\n     * @param  {string} language\n     */\n    @Input() language?: string;\n\n    /**\n     * Optional form field input type: textarea? set to true for textarea\n     * otherwise it's a simple (short) input field\n     *\n     * @param  {boolean} [textarea=false]\n     */\n    @Input() textarea?: boolean = false;\n\n    /**\n     * Optional form field value of type StringLiteral[]\n     *\n     * @param {StringLiteral[]} value\n     */\n    @Input() value?: StringLiteral[] = [];\n\n    /**\n     * Optional disable the input field in case of no right to edit the field/value\n     *\n     * @param {boolean}: [disabled=false]\n     */\n    @Input() disabled?: boolean = false;\n\n    /**\n     * The readonly attribute specifies whether the control may be modified by the user.\n     *\n     * @param {boolean}: [readonly=false]\n     */\n    @Input() readonly?: boolean = false;\n\n    /**\n     * Returns (output) an array of StringLiteral on any change on the input field.\n     *\n     * @emits {StringLiteral[]} dataChanged\n     */\n    @Output() dataChanged: EventEmitter<StringLiteral[]> = new EventEmitter<StringLiteral[]>();\n\n    /**\n     * Returns (output) true when the field was touched. This can be used to validate data, e.g. in case a value is required\n     *\n     * @emits {boolean} touched\n     */\n    @Output() touched: EventEmitter<boolean> = new EventEmitter<boolean>();\n\n    /**\n     * Returns true when a user press ENTER. This can be used to submit data in the parent component.\n     *\n     * * @emits {boolean} enter\n     */\n    @Output() enter: EventEmitter<boolean> = new EventEmitter<boolean>();\n\n    @ViewChild('textInput', { static: false }) textInput: ElementRef;\n\n    @ViewChild('btnToSelectLanguage', { static: false }) btnToSelectLanguage: MatMenuTrigger;\n\n    form: FormGroup;\n\n    constructor (\n        private _fb: FormBuilder\n    ) {\n\n        // set selected language, if it's not defined yet\n        if (!this.language) {\n            if (localStorage.getItem('session') !== null) {\n                // get language from the logged-in user profile data\n                this.language = JSON.parse(localStorage.getItem('session')).user.lang;\n            } else {\n                // get default language from browser\n                this.language = navigator.language.substr(0, 2);\n            }\n        }\n\n        // does the defined language exists in our supported languages list?\n        if (this.languages.indexOf(this.language) === -1) {\n            // if not, select the first language from our list of supported languages\n            this.language = this.languages[0];\n        }\n\n    }\n\n    ngOnInit() {\n\n        // if (this.placeholder.length > 0) {\n        //     this.placeholder += ' (' + this.language + ')';\n        // }\n\n        // reset stringLiterals if they have empty values\n        this.resetValues();\n\n        // build the form\n        this.form = this._fb.group({\n            text: new FormControl(\n                {\n                    value: '',\n                    disabled: this.disabled\n                },\n                {\n                    // updateOn: 'blur'\n                }\n            )\n        });\n        // update values on form change\n        this.form.valueChanges.subscribe(data => this.onValueChanged());\n\n        // get value from stringLiterals\n        const val = this.getValueFromStringLiteral(this.language);\n        this.updateFormField(val);\n\n\n    }\n\n    /**\n     * @ignore\n     *\n     * emit data to parent on any change on the input field\n     */\n    onValueChanged() {\n        if (!this.form) {\n            return;\n        }\n\n        const form = this.form;\n        const control = form.get('text');\n        this.touched.emit(control && control.dirty);\n\n        // if (control && control.dirty) {\n        // console.warn('control dirty');\n\n        // }\n\n        this.updateStringLiterals(this.language, this.form.controls['text'].value);\n\n        this.dataChanged.emit(this.value);\n\n    }\n\n    toggleAll() {\n        // TODO: open/show all languages with their values\n    }\n\n    /**\n     * @ignore\n     *\n     * Set the language after selecting; This updates the array of StringLiterals: adds item with the selected language if it doesn't exist\n     */\n    setLanguage(lang: string) {\n\n        if (this.language === lang) {\n            // console.warn('DO NOTHING! this language was already selected');\n        } else {\n            // clean stringLIteral value for previous language, if text field is empty\n            this.updateStringLiterals(this.language, this.form.controls['text'].value);\n\n            this.language = lang;\n            // update form field value / reset in case of no value\n            const val = this.getValueFromStringLiteral(lang);\n            this.updateFormField(val);\n        }\n    }\n\n    /**\n     * @ignore\n     *\n     * Switch focus to input field after selecting a language\n     */\n    switchFocus() {\n        // close the menu\n        if (!this.textarea && this.btnToSelectLanguage && this.btnToSelectLanguage.menuOpen) {\n            this.btnToSelectLanguage.closeMenu();\n        }\n\n        if (!this.disabled) {\n            this.form.controls['text'].enable();\n            this.textInput.nativeElement.focus();\n        }\n    }\n\n    /**\n     * @ignore\n     *\n     * Set the value in the input field\n     */\n    updateFormField(value: string) {\n        if (!value) {\n            value = '';\n        }\n        this.form.controls['text'].setValue(value);\n    }\n\n    /**\n     * @ignore\n     *\n     * Update the array of StringLiterals depending on value / empty value add or remove item from array.\n     */\n    updateStringLiterals(lang: string, value?: string) {\n        const index = this.value.findIndex(i => i.language === lang);\n\n        if (index > -1 && this.value[index].value.length > 0) {\n            // value is not empty and exists in list of stringLiterals\n            // console.log('update existing value for ' + lang + ' on position ' + index);\n            this.value[index].value = value;\n        }\n\n        if ((!value || value.length === 0) && index > -1) {\n            // value is empty: delete stringLiteral item for this language\n            // console.log('delete empty value for ' + lang + ' on position ' + index);\n            this.value.splice(index, 1);\n        }\n\n        if (index < 0 && value) {\n            // value doesn't exist in stringLiterals: add one\n            // console.log('add new value (' + value + ') for ' + lang);\n            const newValue: StringLiteral = {\n                value: value,\n                language: lang\n            };\n            this.value.push(newValue);\n        }\n\n    }\n\n    /**\n     * @ignore\n     *\n     * In case of strange array of StringLiterals, this method will reset to a API-conform array. This means an array without empty values.\n     */\n    resetValues() {\n        const length: number = this.value.length;\n\n        if (length > 0) {\n            let index = length - 1;\n            while (index >= 0) {\n                // remove items with empty value\n                if (!this.value[index].value.length) {\n                    this.value.splice(index, 1);\n                }\n                index--;\n            }\n\n            // does an item for selected lanuage exists\n            if (this.value.findIndex(i => i.language === this.language) === -1) {\n                this.language = this.value[0].language;\n            }\n\n        } else {\n            this.value = [];\n        }\n    }\n\n    /**\n     * @ignore\n     *\n     * Get the value from array of StringLiterals for the selected language\n     */\n    getValueFromStringLiteral(lang: string): string {\n        // console.log('existing value in', this.value);\n        // get index for this language\n        const index = this.value.findIndex(i => i.language === lang);\n\n        if (this.value[index] && this.value[index].value.length > 0) {\n            return this.value[index].value;\n        } else {\n            return undefined;\n        }\n\n    }\n\n}\n"]}