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
JavaScript
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"]}