UNPKG

gentics-ui-core

Version:

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

164 lines 19.8 kB
import { ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, Input, Output, Renderer2, ViewChild } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import * as i0 from "@angular/core"; import * as i1 from "ngx-autosize"; const GTX_TEXTAREA_VALUE_ACCESSOR = { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => Textarea), multi: true }; /** * The Textarea wraps the native `<textarea>` form element. Textareas automatically grow to accommodate their content. * * ```html * <gtx-textarea label="Message" [(ngModel)]="message"></gtx-textarea> * ``` */ export class Textarea { constructor(renderer, changeDetector, elementRef) { this.renderer = renderer; this.changeDetector = changeDetector; this.elementRef = elementRef; /** * Sets the textarea to be auto-focused. Handled by `AutofocusDirective`. */ this.autofocus = false; /** * Sets the disabled state. */ this.disabled = false; /** * Sets the readonly state. */ this.readonly = false; /** * Sets the required state. */ this.required = false; /** * Sets the value of the control. */ this.value = ''; /** * Sets the label of the control. */ this.label = ''; /** * Blur event. */ this.blur = new EventEmitter(); /** * Focus event. */ this.focus = new EventEmitter(); /** * Change event. */ this.change = new EventEmitter(); this.valueIsValid = true; } /** * Sets the maximum number of characters permitted. */ set maxlength(val) { if (val != null && !isNaN(val) && val > 0) { this._maxlength = Number(val); } else { this._maxlength = undefined; } } get maxlength() { return this._maxlength; } ngOnChanges(changes) { const valueChange = changes['value']; if (valueChange) { this.writeValue(valueChange.currentValue); } } onBlur(e) { e.stopPropagation(); const value = e.target.value; this.blur.emit(this.normalizeValue(value)); this.onTouched(); } onChangeEvent(e) { e.stopPropagation(); } onFocus(e) { const value = e.target.value; this.focus.emit(value); } onInput(e) { const value = this.currentValue = e.target.value; this.onChange(value); this.change.emit(value); this.onTouched(); setTimeout(() => { const element = this.elementRef.nativeElement; this.valueIsValid = !element.classList.contains('ng-touched') || !element.classList.contains('ng-invalid'); }); } writeValue(valueToWrite) { const value = this.normalizeValue(valueToWrite); if (value !== this.currentValue) { this.renderer.setProperty(this.nativeTextarea.nativeElement, 'value', this.currentValue = value); } } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(disabled) { this.disabled = disabled; this.changeDetector.markForCheck(); } onChange(newValue) { } onTouched() { } normalizeValue(value) { return (value == null ? '' : String(value)).replace(/\r\n?/g, '\n'); } } /** @nocollapse */ Textarea.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Textarea, deps: [{ token: i0.Renderer2 }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ Textarea.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: Textarea, selector: "gtx-textarea", inputs: { autofocus: "autofocus", disabled: "disabled", maxlength: "maxlength", name: "name", pattern: "pattern", placeholder: "placeholder", readonly: "readonly", required: "required", validationErrorTooltip: "validationErrorTooltip", value: "value", label: "label", id: "id" }, outputs: { blur: "blur", focus: "focus", change: "change" }, providers: [GTX_TEXTAREA_VALUE_ACCESSOR], viewQueries: [{ propertyName: "nativeTextarea", first: true, predicate: ["textarea"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<textarea\n class=\"materialize-textarea\"\n autosize\n [minRows]=\"1\"\n [attr.id]=\"id\"\n [attr.maxlength]=\"maxlength\"\n [attr.name]=\"name\"\n [attr.pattern]=\"pattern\"\n [attr.placeholder]=\"placeholder\"\n [attr.title]=\"(!valueIsValid && validationErrorTooltip) ? validationErrorTooltip : ''\"\n\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [required]=\"required\"\n\n (blur)=\"onBlur($event)\"\n (change)=\"onChangeEvent($event)\"\n (focus)=\"onFocus($event)\"\n (input)=\"onInput($event)\"\n\n #textarea\n></textarea>\n<label [attr.for]=\"id\">{{ label }}</label>\n", directives: [{ type: i1.AutosizeDirective, selector: "[autosize]", inputs: ["onlyGrow", "useImportant", "minRows", "autosize", "maxRows"], outputs: ["resized"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Textarea, decorators: [{ type: Component, args: [{ selector: 'gtx-textarea', providers: [GTX_TEXTAREA_VALUE_ACCESSOR], template: "<textarea\n class=\"materialize-textarea\"\n autosize\n [minRows]=\"1\"\n [attr.id]=\"id\"\n [attr.maxlength]=\"maxlength\"\n [attr.name]=\"name\"\n [attr.pattern]=\"pattern\"\n [attr.placeholder]=\"placeholder\"\n [attr.title]=\"(!valueIsValid && validationErrorTooltip) ? validationErrorTooltip : ''\"\n\n [disabled]=\"disabled\"\n [readonly]=\"readonly\"\n [required]=\"required\"\n\n (blur)=\"onBlur($event)\"\n (change)=\"onChangeEvent($event)\"\n (focus)=\"onFocus($event)\"\n (input)=\"onInput($event)\"\n\n #textarea\n></textarea>\n<label [attr.for]=\"id\">{{ label }}</label>\n" }] }], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { autofocus: [{ type: Input }], disabled: [{ type: Input }], maxlength: [{ type: Input }], name: [{ type: Input }], pattern: [{ type: Input }], placeholder: [{ type: Input }], readonly: [{ type: Input }], required: [{ type: Input }], validationErrorTooltip: [{ type: Input }], value: [{ type: Input }], label: [{ type: Input }], id: [{ type: Input }], blur: [{ type: Output }], focus: [{ type: Output }], change: [{ type: Output }], nativeTextarea: [{ type: ViewChild, args: ['textarea', { static: true }] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"textarea.component.js","sourceRoot":"","sources":["../../../../../src/components/textarea/textarea.component.ts","../../../../../src/components/textarea/textarea.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    Input,\n    OnChanges,\n    Output,\n    Renderer2,\n    SimpleChanges,\n    ViewChild\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\n\n\nconst GTX_TEXTAREA_VALUE_ACCESSOR = {\n    provide: NG_VALUE_ACCESSOR,\n    useExisting: forwardRef(() => Textarea),\n    multi: true\n};\n\n/**\n * The Textarea wraps the native `<textarea>` form element. Textareas automatically grow to accommodate their content.\n *\n * ```html\n * <gtx-textarea label=\"Message\" [(ngModel)]=\"message\"></gtx-textarea>\n * ```\n */\n@Component({\n    selector: 'gtx-textarea',\n    templateUrl: './textarea.tpl.html',\n    providers: [GTX_TEXTAREA_VALUE_ACCESSOR]\n})\nexport class Textarea implements ControlValueAccessor, OnChanges {\n    /**\n     * Sets the textarea 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     * Sets the maximum number of characters permitted.\n     */\n    @Input() set maxlength(val: any) {\n        if (val != null && !isNaN(val) && val > 0) {\n            this._maxlength = Number(val);\n        } else {\n            this._maxlength = undefined;\n        }\n    }\n    get maxlength(): any {\n        return this._maxlength;\n    }\n\n    /**\n     * The name of the control.\n     */\n    @Input() name: string;\n\n    /**\n     * Regex pattern for complex validation.\n     * This requires that this control is either part of a form or that\n     * its value is bound with ngModel.\n     */\n    @Input() pattern: string;\n\n    /**\n     * A placeholder text to display when the control is empty.\n     */\n    @Input() placeholder: string;\n\n    /**\n     * Sets the readonly state.\n     */\n    @Input() readonly: boolean = false;\n\n    /**\n     * Sets the required state.\n     */\n    @Input() required: boolean = false;\n\n    /**\n     * Tooltip for validation errors.\n     */\n    @Input() validationErrorTooltip: string;\n\n    /**\n     * Sets the value of the control.\n     */\n    @Input() value: string = '';\n\n    /**\n     * Sets the label of the control.\n     */\n    @Input() label: string = '';\n\n    /**\n     * Sets an id for the control.\n     */\n    @Input() id: string;\n\n    /**\n     * Blur event.\n     */\n    @Output() blur = new EventEmitter<string>();\n\n    /**\n     * Focus event.\n     */\n    @Output() focus = new EventEmitter<string>();\n\n    /**\n     * Change event.\n     */\n    @Output() change = new EventEmitter<string>();\n\n    valueIsValid: boolean = true;\n\n    @ViewChild('textarea', { static: true }) private nativeTextarea: ElementRef;\n    private _maxlength: number;\n    private currentValue: string;\n    private previousHeight: number;\n\n    constructor(\n        private renderer: Renderer2,\n        private changeDetector: ChangeDetectorRef,\n        private elementRef: ElementRef\n    ) { }\n\n    ngOnChanges(changes: SimpleChanges): void {\n        const valueChange = changes['value'];\n        if (valueChange) {\n            this.writeValue(valueChange.currentValue);\n        }\n    }\n\n    onBlur(e: Event): void {\n        e.stopPropagation();\n        const value = (e.target as HTMLTextAreaElement).value;\n        this.blur.emit(this.normalizeValue(value));\n        this.onTouched();\n    }\n\n    onChangeEvent(e: Event): void {\n        e.stopPropagation();\n    }\n\n    onFocus(e: Event): void {\n        const value = (e.target as HTMLTextAreaElement).value;\n        this.focus.emit(value);\n    }\n\n    onInput(e: Event): void {\n        const value = this.currentValue = (e.target as HTMLTextAreaElement).value;\n        this.onChange(value);\n        this.change.emit(value);\n        this.onTouched();\n\n        setTimeout(() => {\n            const element: HTMLTextAreaElement = this.elementRef.nativeElement;\n            this.valueIsValid = !element.classList.contains('ng-touched') || !element.classList.contains('ng-invalid');\n        });\n    }\n\n    writeValue(valueToWrite: any): void {\n        const value = this.normalizeValue(valueToWrite);\n        if (value !== this.currentValue) {\n            this.renderer.setProperty(this.nativeTextarea.nativeElement, 'value', this.currentValue = value);\n        }\n    }\n\n    registerOnChange(fn: (newValue: string) => void): void {\n        this.onChange = fn;\n    }\n\n    registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    setDisabledState(disabled: boolean): void {\n        this.disabled = disabled;\n        this.changeDetector.markForCheck();\n    }\n\n    private onChange(newValue: string): void { }\n    private onTouched(): void { }\n\n    private normalizeValue(value: any): string {\n        return (value == null ? '' : String(value)).replace(/\\r\\n?/g, '\\n');\n    }\n}\n","<textarea\n    class=\"materialize-textarea\"\n    autosize\n    [minRows]=\"1\"\n    [attr.id]=\"id\"\n    [attr.maxlength]=\"maxlength\"\n    [attr.name]=\"name\"\n    [attr.pattern]=\"pattern\"\n    [attr.placeholder]=\"placeholder\"\n    [attr.title]=\"(!valueIsValid && validationErrorTooltip) ? validationErrorTooltip : ''\"\n\n    [disabled]=\"disabled\"\n    [readonly]=\"readonly\"\n    [required]=\"required\"\n\n    (blur)=\"onBlur($event)\"\n    (change)=\"onChangeEvent($event)\"\n    (focus)=\"onFocus($event)\"\n    (input)=\"onInput($event)\"\n\n    #textarea\n></textarea>\n<label [attr.for]=\"id\">{{ label }}</label>\n"]}