UNPKG

gentics-ui-core

Version:

This is the common core framework for the Gentics CMS and Mesh UI, and other Angular applications.

253 lines 30.6 kB
import { ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, HostListener, Input, Output, Renderer2, ViewChild } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import * as i0 from "@angular/core"; const GTX_INPUT_VALUE_ACCESSOR = { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => InputField), multi: true }; /** * E-mail validator regex from Angular 8 * @todo Implement with validators * @see https://github.com/angular/angular/blob/8.2.9/packages/forms/src/validators.ts#L60 */ const EMAIL_REGEXP = "^(?=.{1,254}$)(?=.{1,64}@)[-!#$%&'*+/0-9=?A-Z^_`a-z{|}~]+(\\.[-!#$%&'*+/0-9=?A-Z^_`a-z{|}~]+)*@[A-Za-z0-9]([A-Za-z0-9-]{0,61}[A-Za-z0-9])?(\\.[A-Za-z0-9]([A-Za-z0-9-]{0,61}[A-Za-z0-9])?)*$"; /** * Telephone number validator regex * @todo Implement with validators * @see https://stackoverflow.com/a/26516985 */ const TEL_REGEXP = "^([()\\- x+]*\d[()\\- x+]*){4,16}$"; /** * URL validator regex * @todo Implement with validators * @see https://stackoverflow.com/a/52017332 */ const URL_REGEXP = "(^|\\s)((https?:\\/\\/)?[\\w-]+(\\.[\\w-]+)+\\.?(:\\d+)?(\\/\\S*)?)"; const ACTIVE_CLASS = 'active'; /** * The InputField wraps the native `<input>` form element but should only be used for * text, number, password, tel, email or url types. Other types (date, range, file) should have dedicated components. * * * Note that the class is named `InputField` since `Input` is used by the Angular framework to denote * component inputs. * * ```html * <gtx-input label="Text Input Label"></gtx-input> * <gtx-input placeholder="Number Input Placeholder" * type="number" min="0" max="100" step="5"></gtx-input> * ``` */ export class InputField { constructor(renderer, changeDetector) { this.renderer = renderer; this.changeDetector = changeDetector; /** * Sets the input field to be auto-focused. Handled by `AutofocusDirective`. */ this.autofocus = false; /** * Sets the disabled state */ this.disabled = false; /** * A label for the input */ this.label = ''; /** * Sets the readonly state of the input */ this.readonly = false; /** * Sets the required state of the input */ this.required = false; /** * Can be "text", "number", "password", "tel", "email" or "url". */ this.type = 'text'; /** * Sets the value of the input. */ this.value = ''; /** * Fires when the input loses focus. */ this.blur = new EventEmitter(); /** * Fires when the input gains focus. */ this.focus = new EventEmitter(); /** * Fires whenever a char is entered into the field. */ this.change = new EventEmitter(); this.onChange = (newValue) => { }; this.onTouched = () => { }; } ngOnInit() { /** * Set default regex patterns for specific field types if not set */ if (!this.pattern) { switch (this.type) { case 'email': this.pattern = EMAIL_REGEXP; break; case 'tel': this.pattern = TEL_REGEXP; break; case 'url': this.pattern = URL_REGEXP; break; default: } } } /** * The Materialize input includes a dynamic label that changes position depending on the state of the input. * When the label has the "active" class, it moves above the input, otherwise it resides inside the input * itself. * * The Materialize "forms.js" script normally takes care of adding/removing the active class on page load, * but this does not work in a SPA setting where new views with inputs can be created without a page load * event to trigger the `Materialize.updateTextFields()` method. Therefore we need to handle it ourselves * when the input component is created. */ ngAfterViewInit() { const input = this.inputElement.nativeElement; const label = this.labelElement.nativeElement; if (input && label) { if (String(this.value).length > 0 || this.placeholder) { this.renderer.addClass(label, ACTIVE_CLASS); } else { this.renderer.removeClass(label, ACTIVE_CLASS); } } } ngOnChanges(changes) { const valueChange = changes['value']; if (valueChange) { this.writeValue(valueChange.currentValue); } } onKeyDown(event) { if (this.type === 'number') { const keyboardEvent = event; if ((keyboardEvent.key === '-' && event.target.value) || keyboardEvent.key === '+') { keyboardEvent.preventDefault(); } else if (keyboardEvent.key === '-' && !event.target.value) { event.target.value = '-'; } } } onBlur(e) { e.stopPropagation(); const target = e.target; this.blur.emit(this.normalizeValue(target.value)); this.onTouched(); } onFocus(e) { const target = e.target; this.focus.emit(this.normalizeValue(target.value)); } onInput(e) { const target = e.target; this.updateValue(target); const value = this.currentValue = this.normalizeValue(target.value); this.onChange(value); this.change.emit(value); } writeValue(valueToWrite) { const value = this.normalizeValue(valueToWrite); if (value !== this.currentValue) { this.renderer.setProperty(this.inputElement.nativeElement, 'value', this.currentValue = value); } } // ValueAccessor members registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(disabled) { this.disabled = disabled; this.changeDetector.markForCheck(); } normalizeValue(val) { if (this.type === 'number') { return val == null ? 0 : Number(val); } else { return val == null ? '' : String(val); } } updateValue(target) { if (this.type === 'number') { if (this.max && Number(target.value) > this.max) { target.value = String(this.max); } else if (this.min && Number(target.value) < this.min) { target.value = String(this.min); } } } } /** @nocollapse */ InputField.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: InputField, deps: [{ token: i0.Renderer2 }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ InputField.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: InputField, selector: "gtx-input", inputs: { autocomplete: "autocomplete", autofocus: "autofocus", disabled: "disabled", id: "id", label: "label", max: "max", min: "min", maxlength: "maxlength", name: "name", pattern: "pattern", placeholder: "placeholder", readonly: "readonly", required: "required", step: "step", type: "type", value: "value" }, outputs: { blur: "blur", focus: "focus", change: "change" }, host: { listeners: { "keydown": "onKeyDown($event)" } }, providers: [GTX_INPUT_VALUE_ACCESSOR], viewQueries: [{ propertyName: "inputElement", first: true, predicate: ["inputElement"], descendants: true, static: true }, { propertyName: "labelElement", first: true, predicate: ["labelElement"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<input\n [attr.autocomplete]=\"autocomplete || null\"\n [attr.id]=\"id\"\n [attr.max]=\"max\"\n [attr.maxlength]=\"maxlength\"\n [attr.min]=\"min\"\n [attr.name]=\"name\"\n [attr.pattern]=\"pattern\"\n [attr.placeholder]=\"placeholder\"\n [attr.step]=\"step\"\n\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [required]=\"required\"\n [type]=\"type\"\n\n (blur)=\"onBlur($event)\"\n (change)=\"onBlur($event)\"\n (focus)=\"onFocus($event)\"\n (input)=\"onInput($event)\"\n\n #inputElement\n><label [attr.for]=\"id\" (click)=\"inputElement.focus()\" #labelElement>{{ label }}</label>\n" }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: InputField, decorators: [{ type: Component, args: [{ selector: 'gtx-input', providers: [GTX_INPUT_VALUE_ACCESSOR], template: "<input\n [attr.autocomplete]=\"autocomplete || null\"\n [attr.id]=\"id\"\n [attr.max]=\"max\"\n [attr.maxlength]=\"maxlength\"\n [attr.min]=\"min\"\n [attr.name]=\"name\"\n [attr.pattern]=\"pattern\"\n [attr.placeholder]=\"placeholder\"\n [attr.step]=\"step\"\n\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [required]=\"required\"\n [type]=\"type\"\n\n (blur)=\"onBlur($event)\"\n (change)=\"onBlur($event)\"\n (focus)=\"onFocus($event)\"\n (input)=\"onInput($event)\"\n\n #inputElement\n><label [attr.for]=\"id\" (click)=\"inputElement.focus()\" #labelElement>{{ label }}</label>\n" }] }], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { autocomplete: [{ type: Input }], autofocus: [{ type: Input }], disabled: [{ type: Input }], id: [{ type: Input }], label: [{ type: Input }], max: [{ type: Input }], min: [{ type: Input }], maxlength: [{ type: Input }], name: [{ type: Input }], pattern: [{ type: Input }], placeholder: [{ type: Input }], readonly: [{ type: Input }], required: [{ type: Input }], step: [{ type: Input }], type: [{ type: Input }], value: [{ type: Input }], blur: [{ type: Output }], focus: [{ type: Output }], change: [{ type: Output }], inputElement: [{ type: ViewChild, args: ['inputElement', { static: true }] }], labelElement: [{ type: ViewChild, args: ['labelElement', { static: true }] }], onKeyDown: [{ type: HostListener, args: ['keydown', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input.component.js","sourceRoot":"","sources":["../../../../../src/components/input/input.component.ts","../../../../../src/components/input/input.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    HostListener,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    Renderer2,\n    SimpleChanges,\n    ViewChild\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\n\nconst GTX_INPUT_VALUE_ACCESSOR = {\n    provide: NG_VALUE_ACCESSOR,\n    useExisting: forwardRef(() => InputField),\n    multi: true\n};\n\n/**\n * E-mail validator regex from Angular 8\n * @todo Implement with validators\n * @see https://github.com/angular/angular/blob/8.2.9/packages/forms/src/validators.ts#L60\n */\nconst EMAIL_REGEXP =\n    \"^(?=.{1,254}$)(?=.{1,64}@)[-!#$%&'*+/0-9=?A-Z^_`a-z{|}~]+(\\\\.[-!#$%&'*+/0-9=?A-Z^_`a-z{|}~]+)*@[A-Za-z0-9]([A-Za-z0-9-]{0,61}[A-Za-z0-9])?(\\\\.[A-Za-z0-9]([A-Za-z0-9-]{0,61}[A-Za-z0-9])?)*$\";\n\n/**\n * Telephone number validator regex\n * @todo Implement with validators\n * @see https://stackoverflow.com/a/26516985\n */\nconst TEL_REGEXP = \"^([()\\\\- x+]*\\d[()\\\\- x+]*){4,16}$\";\n\n/**\n * URL validator regex\n * @todo Implement with validators\n * @see https://stackoverflow.com/a/52017332\n */\nconst URL_REGEXP = \"(^|\\\\s)((https?:\\\\/\\\\/)?[\\\\w-]+(\\\\.[\\\\w-]+)+\\\\.?(:\\\\d+)?(\\\\/\\\\S*)?)\";\n\nconst ACTIVE_CLASS = 'active';\n\n/**\n * The InputField wraps the native `<input>` form element but should only be used for\n * text, number, password, tel, email or url types. Other types (date, range, file) should have dedicated components.\n *\n *\n * Note that the class is named `InputField` since `Input` is used by the Angular framework to denote\n * component inputs.\n *\n * ```html\n * <gtx-input label=\"Text Input Label\"></gtx-input>\n * <gtx-input placeholder=\"Number Input Placeholder\"\n *            type=\"number\" min=\"0\" max=\"100\" step=\"5\"></gtx-input>\n * ```\n */\n@Component({\n    selector: 'gtx-input',\n    templateUrl: './input.tpl.html',\n    providers: [GTX_INPUT_VALUE_ACCESSOR]\n})\nexport class InputField implements AfterViewInit, ControlValueAccessor, OnInit, OnChanges {\n    /**\n     * Sets autocomplete attribute on the input field\n     */\n    @Input() autocomplete: string;\n    /**\n     * Sets the input field to be auto-focused. Handled by `AutofocusDirective`.\n     */\n    @Input() autofocus: boolean = false;\n\n    /**\n     * Sets the disabled state\n     */\n    @Input() disabled: boolean = false;\n\n    /**\n     * Input field id\n     */\n    @Input() id: string;\n\n    /**\n     * A label for the input\n     */\n    @Input() label: string = '';\n\n    /**\n     * Max allowed value (applies when type = \"number\")\n     */\n    @Input() max: number;\n\n    /**\n     * Min allowed value (applies when type = \"number\")\n     */\n    @Input() min: number;\n\n    /**\n     * Max allowed length in characters\n     */\n    @Input() maxlength: number;\n\n    /**\n     * Input field name\n     */\n    @Input() name: string;\n\n    /**\n     * Regex pattern for complex validation\n     */\n    @Input() pattern: string;\n\n    /**\n     * Placeholder text to display when the field is empty\n     */\n    @Input() placeholder: string;\n\n    /**\n     * Sets the readonly state of the input\n     */\n    @Input() readonly: boolean = false;\n\n    /**\n     * Sets the required state of the input\n     */\n    @Input() required: boolean = false;\n\n    /**\n     * Increment step (applies when type = \"number\")\n     */\n    @Input() step: number;\n\n    /**\n     * Can be \"text\", \"number\", \"password\", \"tel\", \"email\" or \"url\".\n     */\n    @Input() type: 'text' | 'number' | 'password' | 'tel' | 'email' | 'url' = 'text';\n\n    /**\n     * Sets the value of the input.\n     */\n    @Input() value: string|number = '';\n\n    /**\n     * Fires when the input loses focus.\n     */\n    @Output() blur = new EventEmitter<string|number>();\n\n    /**\n     * Fires when the input gains focus.\n     */\n    @Output() focus = new EventEmitter<string|number>();\n\n    /**\n     * Fires whenever a char is entered into the field.\n     */\n    @Output() change = new EventEmitter<string|number>();\n\n    @ViewChild('inputElement', { static: true }) private inputElement: ElementRef;\n    @ViewChild('labelElement', { static: true }) private labelElement: ElementRef;\n\n    private currentValue: string | number;\n\n    constructor(private renderer: Renderer2,\n                private changeDetector: ChangeDetectorRef) { }\n\n\n    ngOnInit(): void {\n        /**\n         * Set default regex patterns for specific field types if not set\n         */\n        if (!this.pattern) {\n            switch (this.type) {\n                case 'email':\n                    this.pattern = EMAIL_REGEXP;\n                    break;\n                case 'tel':\n                    this.pattern = TEL_REGEXP;\n                    break;\n                case 'url':\n                    this.pattern = URL_REGEXP;\n                    break;\n                default:\n            }\n        }\n    }\n\n    /**\n     * The Materialize input includes a dynamic label that changes position depending on the state of the input.\n     * When the label has the \"active\" class, it moves above the input, otherwise it resides inside the input\n     * itself.\n     *\n     * The Materialize \"forms.js\" script normally takes care of adding/removing the active class on page load,\n     * but this does not work in a SPA setting where new views with inputs can be created without a page load\n     * event to trigger the `Materialize.updateTextFields()` method. Therefore we need to handle it ourselves\n     * when the input component is created.\n     */\n    ngAfterViewInit(): void {\n        const input: HTMLInputElement = this.inputElement.nativeElement;\n        const label: HTMLLabelElement = this.labelElement.nativeElement;\n\n        if (input && label) {\n            if (String(this.value).length > 0 || this.placeholder) {\n                this.renderer.addClass(label, ACTIVE_CLASS);\n            } else {\n                this.renderer.removeClass(label, ACTIVE_CLASS);\n            }\n        }\n    }\n\n    ngOnChanges(changes: SimpleChanges): void {\n        const valueChange = changes['value'];\n        if (valueChange) {\n            this.writeValue(valueChange.currentValue);\n        }\n    }\n\n    @HostListener('keydown', ['$event'])\n    public onKeyDown(event) {\n        if (this.type === 'number') {\n            const keyboardEvent = event as KeyboardEvent;\n            if ((keyboardEvent.key === '-' && event.target.value) || keyboardEvent.key === '+') {\n                keyboardEvent.preventDefault();\n            } else if (keyboardEvent.key === '-' && !event.target.value) {\n                event.target.value = '-';\n            }\n        }\n    }\n\n    onBlur(e: Event): void {\n        e.stopPropagation();\n        const target = e.target as HTMLInputElement;\n        this.blur.emit(this.normalizeValue(target.value));\n        this.onTouched();\n    }\n\n    onFocus(e: Event): void {\n        const target = e.target as HTMLInputElement;\n        this.focus.emit(this.normalizeValue(target.value));\n    }\n\n    onInput(e: Event): void {\n        const target = e.target as HTMLInputElement;\n        this.updateValue(target);\n        const value = this.currentValue = this.normalizeValue(target.value);\n        this.onChange(value);\n        this.change.emit(value);\n    }\n\n    writeValue(valueToWrite: any): void {\n        const value = this.normalizeValue(valueToWrite);\n        if (value !== this.currentValue) {\n            this.renderer.setProperty(this.inputElement.nativeElement, 'value', this.currentValue = value);\n        }\n    }\n\n    // ValueAccessor members\n    registerOnChange(fn: (newValue: string | number) => void): void {\n        this.onChange = fn;\n    }\n    registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n    setDisabledState(disabled: boolean): void {\n        this.disabled = disabled;\n        this.changeDetector.markForCheck();\n    }\n    private onChange = (newValue: string | number): void => {};\n    private onTouched = (): void => {};\n\n    private normalizeValue(val: any): string|number {\n        if (this.type === 'number') {\n            return val == null ? 0 : Number(val);\n        } else {\n            return val == null ? '' : String(val);\n        }\n    }\n\n    private updateValue(target: HTMLInputElement): void {\n        if (this.type === 'number') {\n            if (this.max && Number(target.value) > this.max) {\n                target.value = String(this.max);\n            } else if (this.min && Number(target.value) < this.min) {\n                target.value = String(this.min);\n            }\n        }\n    }\n}\n","<input\n    [attr.autocomplete]=\"autocomplete || null\"\n    [attr.id]=\"id\"\n    [attr.max]=\"max\"\n    [attr.maxlength]=\"maxlength\"\n    [attr.min]=\"min\"\n    [attr.name]=\"name\"\n    [attr.pattern]=\"pattern\"\n    [attr.placeholder]=\"placeholder\"\n    [attr.step]=\"step\"\n\n    [disabled]=\"disabled\"\n    [readonly]=\"readonly\"\n    [required]=\"required\"\n    [type]=\"type\"\n\n    (blur)=\"onBlur($event)\"\n    (change)=\"onBlur($event)\"\n    (focus)=\"onFocus($event)\"\n    (input)=\"onInput($event)\"\n\n    #inputElement\n><label [attr.for]=\"id\" (click)=\"inputElement.focus()\" #labelElement>{{ label }}</label>\n"]}