UNPKG

mat-contenteditable

Version:

Angular contenteditable directive for Angular forms and Material Design

996 lines (986 loc) 71.6 kB
import { __extends } from 'tslib'; import { Directive, ElementRef, Renderer2, HostListener, Input, HostBinding, Optional, Self, ViewContainerRef, NgModule } from '@angular/core'; import { MatFormFieldControl } from '@angular/material/form-field'; import { FormGroupDirective, NgControl, NgForm } from '@angular/forms'; import { ErrorStateMatcher, mixinErrorState } from '@angular/material/core'; import { coerceBooleanProperty } from '@angular/cdk/coercion'; import { Subject } from 'rxjs'; import { MatFormFieldControl as MatFormFieldControl$1, ErrorStateMatcher as ErrorStateMatcher$1 } from '@angular/material'; /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ /** * \@docs-private */ var /** * \@docs-private */ MatInputBase = /** @class */ (function () { function MatInputBase(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl) { this._defaultErrorStateMatcher = _defaultErrorStateMatcher; this._parentForm = _parentForm; this._parentFormGroup = _parentFormGroup; this.ngControl = ngControl; } return MatInputBase; }()); /** @type {?} */ var _MatInputMixinBase = mixinErrorState(MatInputBase); var MatContenteditableDirective = /** @class */ (function (_super) { __extends(MatContenteditableDirective, _super); function MatContenteditableDirective(elementRef, renderer, ngControl, _parentForm, _parentFormGroup, _defaultErrorStateMatcher) { var _this = _super.call(this, _defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl) || this; _this.elementRef = elementRef; _this.renderer = renderer; _this.ngControl = ngControl; _this.stateChanges = new Subject(); _this.id = "mat-input-" + MatContenteditableDirective.nextId++; _this.focused = false; _this.contentEmpty = ['<br>', '<div><br></div>']; _this._required = false; _this._disabled = false; _this.controlType = 'mat-input'; _this.describedBy = ''; _this.propValueAccessor = 'innerHTML'; // Setting the value accessor directly (instead of using // the providers) to avoid running into a circular import. if (_this.ngControl != null) { _this.ngControl.valueAccessor = _this; } return _this; } Object.defineProperty(MatContenteditableDirective.prototype, "value", { get: /** * @return {?} */ function () { return this.elementRef.nativeElement[this.propValueAccessor]; }, set: /** * @param {?} value * @return {?} */ function (value) { if (value !== this.value) { this.elementRef.nativeElement[this.propValueAccessor] = value; this.stateChanges.next(); } }, enumerable: true, configurable: true }); Object.defineProperty(MatContenteditableDirective.prototype, "placeholder", { get: /** * @return {?} */ function () { return this._placeholder; }, set: /** * @param {?} plh * @return {?} */ function (plh) { this._placeholder = plh; this.stateChanges.next(); }, enumerable: true, configurable: true }); Object.defineProperty(MatContenteditableDirective.prototype, "empty", { get: /** * @return {?} */ function () { return !this.value || this.contentEmpty.includes(this.value); }, enumerable: true, configurable: true }); Object.defineProperty(MatContenteditableDirective.prototype, "shouldLabelFloat", { get: /** * @return {?} */ function () { return this.focused || !this.empty; }, enumerable: true, configurable: true }); Object.defineProperty(MatContenteditableDirective.prototype, "required", { get: /** * @return {?} */ function () { return this._required; }, set: /** * @param {?} req * @return {?} */ function (req) { this._required = coerceBooleanProperty(req); this.stateChanges.next(); }, enumerable: true, configurable: true }); Object.defineProperty(MatContenteditableDirective.prototype, "disabled", { get: /** * @return {?} */ function () { return this._disabled; }, set: /** * @param {?} dis * @return {?} */ function (dis) { this._disabled = coerceBooleanProperty(dis); this.stateChanges.next(); }, enumerable: true, configurable: true }); /** * @return {?} */ MatContenteditableDirective.prototype.ngDoCheck = /** * @return {?} */ function () { if (this.ngControl) { // We need to re-evaluate this on every change detection cycle, because there are some // error triggers that we can't subscribe to (e.g. parent form submissions). This means // that whatever logic is in here has to be super lean or we risk destroying the performance. this.updateErrorState(); } }; /** * @return {?} */ MatContenteditableDirective.prototype.callOnChange = /** * @return {?} */ function () { if (typeof this.onChange === 'function') { this.onChange(this.elementRef.nativeElement[this.propValueAccessor]); } }; /** * @return {?} */ MatContenteditableDirective.prototype.callOnFocused = /** * @return {?} */ function () { if (this.focused !== true) { this.focused = true; this.stateChanges.next(); } }; /** * @return {?} */ MatContenteditableDirective.prototype.callOnTouched = /** * @return {?} */ function () { if (typeof this.onTouched === 'function') { this.onTouched(); } if (this.focused !== false) { this.focused = false; this.stateChanges.next(); } }; /** * @param {?} ids * @return {?} */ MatContenteditableDirective.prototype.setDescribedByIds = /** * @param {?} ids * @return {?} */ function (ids) { this.describedBy = ids.join(' '); }; /** * @return {?} */ MatContenteditableDirective.prototype.onContainerClick = /** * @return {?} */ function () { this.elementRef.nativeElement.focus(); }; /** * Writes a new value to the element. * This method will be called by the forms API to write * to the view when programmatic (model -> view) changes are requested. * * See: [ControlValueAccessor](https://angular.io/api/forms/ControlValueAccessor#members) */ /** * Writes a new value to the element. * This method will be called by the forms API to write * to the view when programmatic (model -> view) changes are requested. * * See: [ControlValueAccessor](https://angular.io/api/forms/ControlValueAccessor#members) * @param {?} value * @return {?} */ MatContenteditableDirective.prototype.writeValue = /** * Writes a new value to the element. * This method will be called by the forms API to write * to the view when programmatic (model -> view) changes are requested. * * See: [ControlValueAccessor](https://angular.io/api/forms/ControlValueAccessor#members) * @param {?} value * @return {?} */ function (value) { /** @type {?} */ var normalizedValue = value == null ? '' : value; this.renderer.setProperty(this.elementRef.nativeElement, this.propValueAccessor, normalizedValue); }; /** * Registers a callback function that should be called when * the control's value changes in the UI. * * This is called by the forms API on initialization so it can update * the form model when values propagate from the view (view -> model). */ /** * Registers a callback function that should be called when * the control's value changes in the UI. * * This is called by the forms API on initialization so it can update * the form model when values propagate from the view (view -> model). * @param {?} fn * @return {?} */ MatContenteditableDirective.prototype.registerOnChange = /** * Registers a callback function that should be called when * the control's value changes in the UI. * * This is called by the forms API on initialization so it can update * the form model when values propagate from the view (view -> model). * @param {?} fn * @return {?} */ function (fn) { this.onChange = fn; }; /** * Registers a callback function that should be called when the control receives a blur event. * This is called by the forms API on initialization so it can update the form model on blur. */ /** * Registers a callback function that should be called when the control receives a blur event. * This is called by the forms API on initialization so it can update the form model on blur. * @param {?} fn * @return {?} */ MatContenteditableDirective.prototype.registerOnTouched = /** * Registers a callback function that should be called when the control receives a blur event. * This is called by the forms API on initialization so it can update the form model on blur. * @param {?} fn * @return {?} */ function (fn) { this.onTouched = fn; }; /** * This function is called by the forms API when the control status changes to or from "DISABLED". * Depending on the value, it should enable or disable the appropriate DOM element. */ /** * This function is called by the forms API when the control status changes to or from "DISABLED". * Depending on the value, it should enable or disable the appropriate DOM element. * @param {?} isDisabled * @return {?} */ MatContenteditableDirective.prototype.setDisabledState = /** * This function is called by the forms API when the control status changes to or from "DISABLED". * Depending on the value, it should enable or disable the appropriate DOM element. * @param {?} isDisabled * @return {?} */ function (isDisabled) { if (isDisabled) { this.renderer.setAttribute(this.elementRef.nativeElement, 'disabled', 'true'); this.removeDisabledState = this.renderer.listen(this.elementRef.nativeElement, 'keydown', this.listenerDisabledState); } else { if (this.removeDisabledState) { this.renderer.removeAttribute(this.elementRef.nativeElement, 'disabled'); this.removeDisabledState(); } } }; /** * @param {?} e * @return {?} */ MatContenteditableDirective.prototype.listenerDisabledState = /** * @param {?} e * @return {?} */ function (e) { e.preventDefault(); }; /** * Implemented as part of MatFormFieldControl. * See https://material.angular.io/guide/creating-a-custom-form-field-control */ MatContenteditableDirective.nextId = 0; MatContenteditableDirective.decorators = [ { type: Directive, args: [{ selector: '[contenteditable]', providers: [ { provide: MatFormFieldControl, useExisting: MatContenteditableDirective }, ] },] } ]; /** @nocollapse */ MatContenteditableDirective.ctorParameters = function () { return [ { type: ElementRef }, { type: Renderer2 }, { type: NgControl, decorators: [{ type: Optional }, { type: Self }] }, { type: NgForm, decorators: [{ type: Optional }] }, { type: FormGroupDirective, decorators: [{ type: Optional }] }, { type: ErrorStateMatcher } ]; }; MatContenteditableDirective.propDecorators = { value: [{ type: Input }], id: [{ type: HostBinding }], placeholder: [{ type: Input }], contentEmpty: [{ type: Input }], required: [{ type: Input }], disabled: [{ type: Input }], errorState: [{ type: HostBinding, args: ['attr.aria-invalid',] }], errorStateMatcher: [{ type: Input }], describedBy: [{ type: HostBinding, args: ['attr.aria-describedby',] }], propValueAccessor: [{ type: Input }], callOnChange: [{ type: HostListener, args: ['input',] }], callOnFocused: [{ type: HostListener, args: ['focus',] }], callOnTouched: [{ type: HostListener, args: ['blur',] }] }; return MatContenteditableDirective; }(_MatInputMixinBase)); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ var MatCkeditorDirective = /** @class */ (function (_super) { __extends(MatCkeditorDirective, _super); function MatCkeditorDirective( // @Host() @Self() @Optional() public editor: CKEditorComponent, viewRef, ngControl, _parentForm, _parentFormGroup, _defaultErrorStateMatcher) { var _this = _super.call(this, _defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl) || this; _this.viewRef = viewRef; _this.ngControl = ngControl; _this.stateChanges = new Subject(); _this.id = "mat-input-" + MatCkeditorDirective.nextId++; // Need support from Ckeditor _this.placeholder = ''; _this.contentEmpty = ['<br>', '<p>&nbsp;</p>']; _this.focused = false; _this.required = false; _this.controlType = 'mat-input'; _this.describedBy = ''; return _this; } Object.defineProperty(MatCkeditorDirective.prototype, "value", { get: /** * @return {?} */ function () { return !!this.editor.editorInstance && this.editor.editorInstance.getData(); }, set: /** * @param {?} value * @return {?} */ function (value) { if (value !== this.value) { this.editor.data = value; this.stateChanges.next(); } }, enumerable: true, configurable: true }); Object.defineProperty(MatCkeditorDirective.prototype, "empty", { get: /** * @return {?} */ function () { return !this.value || this.contentEmpty.includes(this.value); }, enumerable: true, configurable: true }); Object.defineProperty(MatCkeditorDirective.prototype, "shouldLabelFloat", { get: /** * @return {?} */ function () { return this.focused || !this.empty; }, enumerable: true, configurable: true }); Object.defineProperty(MatCkeditorDirective.prototype, "disabled", { get: /** * @return {?} */ function () { return this.editor.disabled; }, set: /** * @param {?} isDisabled * @return {?} */ function (isDisabled) { this.editor.setDisabledState(isDisabled); this.stateChanges.next(); }, enumerable: true, configurable: true }); /** * @return {?} */ MatCkeditorDirective.prototype.ngOnInit = /** * @return {?} */ function () { var _this = this; // Can't use injection to get component reference // https://github.com/angular/angular/issues/8277 this.editor = this.viewRef['_data'].componentView.component; this.editor.blur.subscribe(function () { _this.focused = false; _this.stateChanges.next(); }); this.editor.focus.subscribe(function () { _this.focused = true; _this.stateChanges.next(); }); }; /** * @return {?} */ MatCkeditorDirective.prototype.ngDoCheck = /** * @return {?} */ function () { if (this.ngControl) { // We need to re-evaluate this on every change detection cycle, because there are some // error triggers that we can't subscribe to (e.g. parent form submissions). This means // that whatever logic is in here has to be super lean or we risk destroying the performance. this.updateErrorState(); } }; /** * @param {?} ids * @return {?} */ MatCkeditorDirective.prototype.setDescribedByIds = /** * @param {?} ids * @return {?} */ function (ids) { this.describedBy = ids.join(' '); }; /** * @return {?} */ MatCkeditorDirective.prototype.onContainerClick = /** * @return {?} */ function () { if (this.editor.editorInstance) { this.editor.editorInstance.editing.view.focus(); this.stateChanges.next(); } }; /** * Implemented as part of MatFormFieldControl. * See https://material.angular.io/guide/creating-a-custom-form-field-control */ MatCkeditorDirective.nextId = 0; MatCkeditorDirective.decorators = [ { type: Directive, args: [{ selector: '[matCkeditor]', providers: [ { provide: MatFormFieldControl$1, useExisting: MatCkeditorDirective }, ] },] } ]; /** @nocollapse */ MatCkeditorDirective.ctorParameters = function () { return [ { type: ViewContainerRef }, { type: NgControl, decorators: [{ type: Optional }, { type: Self }] }, { type: NgForm, decorators: [{ type: Optional }] }, { type: FormGroupDirective, decorators: [{ type: Optional }] }, { type: ErrorStateMatcher$1 } ]; }; MatCkeditorDirective.propDecorators = { value: [{ type: Input }], id: [{ type: HostBinding }], placeholder: [{ type: Input }], contentEmpty: [{ type: Input }], required: [{ type: Input }], disabled: [{ type: Input }], errorState: [{ type: HostBinding, args: ['attr.aria-invalid',] }], errorStateMatcher: [{ type: Input }], describedBy: [{ type: HostBinding, args: ['attr.aria-describedby',] }] }; return MatCkeditorDirective; }(_MatInputMixinBase)); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ var MatCkeditorBalloonDirective = /** @class */ (function (_super) { __extends(MatCkeditorBalloonDirective, _super); function MatCkeditorBalloonDirective() { return _super !== null && _super.apply(this, arguments) || this; } Object.defineProperty(MatCkeditorBalloonDirective.prototype, "toolbar", { set: /** * @param {?} show * @return {?} */ function (show) { if (this.editor && show !== this.toolbarOpen) { /** @type {?} */ var balloon = this.editor.editorInstance.plugins.get('BalloonToolbar'); if (show) { this.showToolbar(balloon); } else { balloon.hide(); this.toolbarOpen = false; } } }, enumerable: true, configurable: true }); /** * @return {?} */ MatCkeditorBalloonDirective.prototype.ngOnInit = /** * @return {?} */ function () { var _this = this; _super.prototype.ngOnInit.call(this); this.editor.ready.subscribe(function (editor) { /** @type {?} */ var balloon = editor.plugins.get('BalloonToolbar'); balloon.stopListening(editor.model.document.selection, 'change:range'); balloon.stopListening(balloon, '_selectionChangeDebounced'); }); this.editor.focus.subscribe(function () { if (_this.toolbarOpen) { /** @type {?} */ var balloon = _this.editor.editorInstance.plugins.get('BalloonToolbar'); _this.showToolbar(balloon); } }); }; /** * @param {?} balloon * @return {?} */ MatCkeditorBalloonDirective.prototype.showToolbar = /** * @param {?} balloon * @return {?} */ function (balloon) { if (!balloon._balloon.hasView(balloon.toolbarView)) { balloon.listenTo(this.editor.editorInstance.ui, 'update', function () { balloon._balloon.updatePosition(balloon._getBalloonPositionData()); }); balloon._balloon.add({ view: balloon.toolbarView, position: balloon._getBalloonPositionData(), balloonClassName: 'ck-toolbar-container' }); this.toolbarOpen = true; } }; MatCkeditorBalloonDirective.decorators = [ { type: Directive, args: [{ selector: '[matCkeditorBalloon]', providers: [ { provide: MatFormFieldControl, useExisting: MatCkeditorBalloonDirective }, ] },] } ]; MatCkeditorBalloonDirective.propDecorators = { toolbar: [{ type: Input }] }; return MatCkeditorBalloonDirective; }(MatCkeditorDirective)); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ var FormFieldSizerDirective = /** @class */ (function () { function FormFieldSizerDirective(renderer, elementRef) { this.renderer = renderer; this.elementRef = elementRef; } /** * @return {?} */ FormFieldSizerDirective.prototype.ngAfterContentInit = /** * @return {?} */ function () { this.updateSize(); }; /** * @return {?} */ FormFieldSizerDirective.prototype.updateSize = /** * @return {?} */ function () { var _this = this; /** @type {?} */ var infix = this.getElement('mat-form-field-infix'); this.renderer.removeStyle(infix, 'min-height'); setTimeout(function () { /** @type {?} */ var wrapper = _this.getElement('mat-form-field-wrapper'); /** @type {?} */ var offset = _this.elementRef.nativeElement.offsetHeight - wrapper.offsetHeight - parseFloat(getComputedStyle(wrapper).marginTop) - parseFloat(getComputedStyle(wrapper).marginBottom) + parseFloat(getComputedStyle(infix).height); _this.renderer.setStyle(infix, 'min-height', offset + "px"); }); }; /** * @param {?} name * @return {?} */ FormFieldSizerDirective.prototype.getElement = /** * @param {?} name * @return {?} */ function (name) { return this.elementRef.nativeElement.getElementsByClassName(name).item(0); }; FormFieldSizerDirective.decorators = [ { type: Directive, args: [{ selector: '[formFieldSizer]' },] } ]; /** @nocollapse */ FormFieldSizerDirective.ctorParameters = function () { return [ { type: Renderer2 }, { type: ElementRef } ]; }; return FormFieldSizerDirective; }()); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ var MatContenteditableModule = /** @class */ (function () { function MatContenteditableModule() { } MatContenteditableModule.decorators = [ { type: NgModule, args: [{ imports: [], declarations: [MatContenteditableDirective, MatCkeditorDirective, MatCkeditorBalloonDirective, FormFieldSizerDirective], exports: [MatContenteditableDirective, MatCkeditorDirective, MatCkeditorBalloonDirective, FormFieldSizerDirective], },] } ]; return MatContenteditableModule; }()); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ /** * @param {?} html * @return {?} */ function getText(html) { if (html) { /** @type {?} */ var element = document.createElement('span'); element.innerHTML = html; return element.textContent.replace(/\s/g, ''); } return ''; } /** * \@description * Provides a set of built-in validators that can be used by form controls. * * A validator is a function that processes a `FormControl` or collection of * controls and returns an error map or null. A null map means that validation has passed. * * @see [Form Validation](/guide/form-validation) * * \@publicApi */ var /** * \@description * Provides a set of built-in validators that can be used by form controls. * * A validator is a function that processes a `FormControl` or collection of * controls and returns an error map or null. A null map means that validation has passed. * * @see [Form Validation](/guide/form-validation) * * \@publicApi */ HtmlValidators = /** @class */ (function () { function HtmlValidators() { } /** * @description * Validator that requires the control have a non-empty value. * * @usageNotes * * ### Validate that the field is non-empty * * ```typescript * const control = new FormControl('', Validators.required); * * console.log(control.errors); // {required: true} * ``` * * @returns An error map with the `required` property * if the validation check fails, otherwise `null`. * */ /** * \@description * Validator that requires the control have a non-empty value. * * \@usageNotes * * ### Validate that the field is non-empty * * ```typescript * const control = new FormControl('', Validators.required); * * console.log(control.errors); // {required: true} * ``` * * @param {?} control * @return {?} An error map with the `required` property * if the validation check fails, otherwise `null`. * */ HtmlValidators.required = /** * \@description * Validator that requires the control have a non-empty value. * * \@usageNotes * * ### Validate that the field is non-empty * * ```typescript * const control = new FormControl('', Validators.required); * * console.log(control.errors); // {required: true} * ``` * * @param {?} control * @return {?} An error map with the `required` property * if the validation check fails, otherwise `null`. * */ function (control) { /** @type {?} */ var text = getText(control.value); return text ? null : { 'required': true }; }; /** * @description * Validator that requires the length of the control's value to be greater than or equal * to the provided minimum length. This validator is also provided by default if you use the * the HTML5 `minlength` attribute. * * @usageNotes * * ### Validate that the field has a minimum of 3 characters * * ```typescript * const control = new FormControl('ng', Validators.minLength(3)); * * console.log(control.errors); // {minlength: {requiredLength: 3, actualLength: 2}} * ``` * * ```html * <input minlength="5"> * ``` * * @returns A validator function that returns an error map with the * `minlength` if the validation check fails, otherwise `null`. */ /** * \@description * Validator that requires the length of the control's value to be greater than or equal * to the provided minimum length. This validator is also provided by default if you use the * the HTML5 `minlength` attribute. * * \@usageNotes * * ### Validate that the field has a minimum of 3 characters * * ```typescript * const control = new FormControl('ng', Validators.minLength(3)); * * console.log(control.errors); // {minlength: {requiredLength: 3, actualLength: 2}} * ``` * * ```html * <input minlength="5"> * ``` * * @param {?} minLength * @return {?} A validator function that returns an error map with the * `minlength` if the validation check fails, otherwise `null`. */ HtmlValidators.minLength = /** * \@description * Validator that requires the length of the control's value to be greater than or equal * to the provided minimum length. This validator is also provided by default if you use the * the HTML5 `minlength` attribute. * * \@usageNotes * * ### Validate that the field has a minimum of 3 characters * * ```typescript * const control = new FormControl('ng', Validators.minLength(3)); * * console.log(control.errors); // {minlength: {requiredLength: 3, actualLength: 2}} * ``` * * ```html * <input minlength="5"> * ``` * * @param {?} minLength * @return {?} A validator function that returns an error map with the * `minlength` if the validation check fails, otherwise `null`. */ function (minLength) { /** @type {?} */ var fn = function (control) { /** @type {?} */ var text = getText(control.value); if (text) { return text.length < minLength ? { 'minlength': { 'requiredLength': minLength, 'actualLength': text.length } } : null; } return null; // don't validate empty values to allow optional controls }; return fn; }; /** * @description * Validator that requires the length of the control's value to be less than or equal * to the provided maximum length. This validator is also provided by default if you use the * the HTML5 `maxlength` attribute. * * @usageNotes * * ### Validate that the field has maximum of 5 characters * * ```typescript * const control = new FormControl('Angular', Validators.maxLength(5)); * * console.log(control.errors); // {maxlength: {requiredLength: 5, actualLength: 7}} * ``` * * ```html * <input maxlength="5"> * ``` * * @returns A validator function that returns an error map with the * `maxlength` property if the validation check fails, otherwise `null`. */ /** * \@description * Validator that requires the length of the control's value to be less than or equal * to the provided maximum length. This validator is also provided by default if you use the * the HTML5 `maxlength` attribute. * * \@usageNotes * * ### Validate that the field has maximum of 5 characters * * ```typescript * const control = new FormControl('Angular', Validators.maxLength(5)); * * console.log(control.errors); // {maxlength: {requiredLength: 5, actualLength: 7}} * ``` * * ```html * <input maxlength="5"> * ``` * * @param {?} maxLength * @return {?} A validator function that returns an error map with the * `maxlength` property if the validation check fails, otherwise `null`. */ HtmlValidators.maxLength = /** * \@description * Validator that requires the length of the control's value to be less than or equal * to the provided maximum length. This validator is also provided by default if you use the * the HTML5 `maxlength` attribute. * * \@usageNotes * * ### Validate that the field has maximum of 5 characters * * ```typescript * const control = new FormControl('Angular', Validators.maxLength(5)); * * console.log(control.errors); // {maxlength: {requiredLength: 5, actualLength: 7}} * ``` * * ```html * <input maxlength="5"> * ``` * * @param {?} maxLength * @return {?} A validator function that returns an error map with the * `maxlength` property if the validation check fails, otherwise `null`. */ function (maxLength) { /** @type {?} */ var fn = function (control) { /** @type {?} */ var text = getText(control.value); return text.length > maxLength ? { 'maxlength': { 'requiredLength': maxLength, 'actualLength': text.length } } : null; }; return fn; }; return HtmlValidators; }()); /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ /** * @fileoverview added by tsickle * @suppress {checkTypes,extraRequire,uselessCode} checked by tsc */ export { _MatInputMixinBase, MatContenteditableDirective, MatCkeditorDirective, MatCkeditorBalloonDirective, FormFieldSizerDirective, MatContenteditableModule, HtmlValidators }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"mat-contenteditable.js.map","sources":["ng://mat-contenteditable/lib/mat-contenteditable.directive.ts","ng://mat-contenteditable/lib/mat-ckeditor.directive.ts","ng://mat-contenteditable/lib/mat-ckeditor-balloon.directive.ts","ng://mat-contenteditable/lib/form-field-sizer.directive.ts","ng://mat-contenteditable/lib/mat-contenteditable.module.ts","ng://mat-contenteditable/lib/validators.ts"],"sourcesContent":["import {\n  Directive,\n  ElementRef,\n  Renderer2,\n  HostListener,\n  Input,\n  HostBinding,\n  Optional,\n  Self,\n  DoCheck,\n} from '@angular/core';\nimport { MatFormFieldControl } from '@angular/material/form-field';\nimport { ControlValueAccessor, FormGroupDirective, NgControl, NgForm } from '@angular/forms';\nimport { ErrorStateMatcher, mixinErrorState, CanUpdateErrorStateCtor, CanUpdateErrorState } from '@angular/material/core';\nimport { coerceBooleanProperty } from '@angular/cdk/coercion';\nimport { Subject } from 'rxjs';\n\n// Boilerplate for applying mixins to MatInput.\n/** @docs-private */\nclass MatInputBase {\n  constructor(public _defaultErrorStateMatcher: ErrorStateMatcher,\n              public _parentForm: NgForm,\n              public _parentFormGroup: FormGroupDirective,\n              /** @docs-private */\n              public ngControl: NgControl) {}\n}\nexport const _MatInputMixinBase: CanUpdateErrorStateCtor & typeof MatInputBase =\n  mixinErrorState(MatInputBase);\n\n\n@Directive({\n  selector: '[contenteditable]',\n  providers: [\n    { provide: MatFormFieldControl, useExisting: MatContenteditableDirective },\n  ]\n})\nexport class MatContenteditableDirective extends _MatInputMixinBase\n  implements ControlValueAccessor, MatFormFieldControl<string>, DoCheck, CanUpdateErrorState {\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * See https://material.angular.io/guide/creating-a-custom-form-field-control\n   */\n  static nextId = 0;\n\n  @Input()\n  get value(): string { return this.elementRef.nativeElement[this.propValueAccessor]; }\n  set value(value: string) {\n    if (value !== this.value) {\n      this.elementRef.nativeElement[this.propValueAccessor] = value;\n      this.stateChanges.next();\n    }\n  }\n\n  readonly stateChanges: Subject<void> = new Subject<void>();\n\n  @HostBinding() id = `mat-input-${MatContenteditableDirective.nextId++}`;\n\n  @Input()\n  get placeholder() {\n    return this._placeholder;\n  }\n  set placeholder(plh) {\n    this._placeholder = plh;\n    this.stateChanges.next();\n  }\n  private _placeholder: string;\n\n  focused = false;\n\n  @Input() contentEmpty: Array<string> = ['<br>', '<div><br></div>'];\n  get empty(): boolean {\n    return !this.value || this.contentEmpty.includes(this.value);\n  }\n\n  get shouldLabelFloat(): boolean { return this.focused || !this.empty; }\n\n  @Input()\n  get required() {\n    return this._required;\n  }\n  set required(req) {\n    this._required = coerceBooleanProperty(req);\n    this.stateChanges.next();\n  }\n  private _required = false;\n\n  @Input()\n  get disabled() {\n    return this._disabled;\n  }\n  set disabled(dis) {\n    this._disabled = coerceBooleanProperty(dis);\n    this.stateChanges.next();\n  }\n  private _disabled = false;\n\n  @HostBinding('attr.aria-invalid') errorState: boolean;\n  @Input() errorStateMatcher: ErrorStateMatcher;\n\n  controlType = 'mat-input';\n\n  @HostBinding('attr.aria-describedby') describedBy = '';\n\n\n  // Part of ControlValueAccessor\n  private onChange: (value: string) => void;\n  private onTouched: () => void;\n  private removeDisabledState: () => void;\n\n  @Input() propValueAccessor = 'innerHTML';\n\n  constructor(\n    private elementRef: ElementRef,\n    private renderer: Renderer2,\n    @Optional() @Self() public ngControl: NgControl,\n    @Optional() _parentForm: NgForm,\n    @Optional() _parentFormGroup: FormGroupDirective,\n    _defaultErrorStateMatcher: ErrorStateMatcher,\n  ) {\n    super(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl);\n    // Setting the value accessor directly (instead of using\n    // the providers) to avoid running into a circular import.\n    if (this.ngControl != null) { this.ngControl.valueAccessor = this; }\n  }\n\n  ngDoCheck() {\n    if (this.ngControl) {\n      // We need to re-evaluate this on every change detection cycle, because there are some\n      // error triggers that we can't subscribe to (e.g. parent form submissions). This means\n      // that whatever logic is in here has to be super lean or we risk destroying the performance.\n      this.updateErrorState();\n    }\n  }\n\n  @HostListener('input')\n  callOnChange() {\n    if (typeof this.onChange === 'function') {\n      this.onChange(this.elementRef.nativeElement[this.propValueAccessor]);\n    }\n  }\n\n  @HostListener('focus')\n  callOnFocused() {\n    if (this.focused !== true) {\n      this.focused = true;\n      this.stateChanges.next();\n    }\n  }\n\n  @HostListener('blur')\n  callOnTouched() {\n    if (typeof this.onTouched === 'function') {\n      this.onTouched();\n    }\n    if (this.focused !== false) {\n      this.focused = false;\n      this.stateChanges.next();\n    }\n  }\n\n  setDescribedByIds(ids: string[]) {\n    this.describedBy = ids.join(' ');\n  }\n\n  onContainerClick() { this.elementRef.nativeElement.focus(); }\n\n  /**\n   * Writes a new value to the element.\n   * This method will be called by the forms API to write\n   * to the view when programmatic (model -> view) changes are requested.\n   *\n   * See: [ControlValueAccessor](https://angular.io/api/forms/ControlValueAccessor#members)\n   */\n  writeValue(value: any): void {\n    const normalizedValue = value == null ? '' : value;\n    this.renderer.setProperty(this.elementRef.nativeElement, this.propValueAccessor, normalizedValue);\n  }\n\n  /**\n   * Registers a callback function that should be called when\n   * the control's value changes in the UI.\n   *\n   * This is called by the forms API on initialization so it can update\n   * the form model when values propagate from the view (view -> model).\n   */\n  registerOnChange(fn: () => void): void {\n    this.onChange = fn;\n  }\n\n  /**\n   * Registers a callback function that should be called when the control receives a blur event.\n   * This is called by the forms API on initialization so it can update the form model on blur.\n   */\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  /**\n   * This function is called by the forms API when the control status changes to or from \"DISABLED\".\n   * Depending on the value, it should enable or disable the appropriate DOM element.\n   */\n  setDisabledState(isDisabled: boolean): void {\n    if (isDisabled) {\n      this.renderer.setAttribute(this.elementRef.nativeElement, 'disabled', 'true');\n      this.removeDisabledState = this.renderer.listen(this.elementRef.nativeElement, 'keydown', this.listenerDisabledState);\n    } else {\n      if (this.removeDisabledState) {\n        this.renderer.removeAttribute(this.elementRef.nativeElement, 'disabled');\n        this.removeDisabledState();\n      }\n    }\n  }\n\n  private listenerDisabledState(e: KeyboardEvent) {\n    e.preventDefault();\n  }\n}\n","import { Directive, Input, HostBinding, ViewContainerRef, OnInit, Optional, Self, DoCheck } from '@angular/core';\nimport { MatFormFieldControl, ErrorStateMatcher, CanUpdateErrorState } from '@angular/material';\n// import { CKEditorComponent } from '@ckeditor/ckeditor5-angular//ckeditor.component';\nimport { Subject } from 'rxjs';\nimport { NgControl, NgForm, FormGroupDirective } from '@angular/forms';\nimport { _MatInputMixinBase } from './mat-contenteditable.directive';\n\n\n@Directive({\n  selector: '[matCkeditor]',\n  providers: [\n    { provide: MatFormFieldControl, useExisting: MatCkeditorDirective },\n  ]\n})\nexport class MatCkeditorDirective  extends _MatInputMixinBase\n  implements MatFormFieldControl<string>, DoCheck, CanUpdateErrorState , OnInit {\n\n  /**\n   * Implemented as part of MatFormFieldControl.\n   * See https://material.angular.io/guide/creating-a-custom-form-field-control\n   */\n  static nextId = 0;\n\n  @Input()\n  get value(): string {\n    return !!this.editor.editorInstance && this.editor.editorInstance.getData();\n  }\n  set value(value: string) {\n    if (value !== this.value) {\n      this.editor.data = value;\n      this.stateChanges.next();\n    }\n  }\n\n  readonly stateChanges: Subject<void> = new Subject<void>();\n\n  @HostBinding() id = `mat-input-${MatCkeditorDirective.nextId++}`;\n\n  // Need support from Ckeditor\n  @Input() placeholder = '';\n\n  @Input() contentEmpty: string[] = ['<br>', '<p>&nbsp;</p>'];\n  get empty(): boolean {\n    return !this.value || this.contentEmpty.includes(this.value);\n  }\n\n  get shouldLabelFloat(): boolean { return this.focused || !this.empty; }\n\n  focused = false;\n\n  @Input() required = false;\n\n  @Input()\n  set disabled(isDisabled: boolean) {\n    this.editor.setDisabledState(isDisabled);\n    this.stateChanges.next();\n  }\n  get disabled() {\n    return this.editor.disabled;\n  }\n\n  @HostBinding('attr.aria-invalid') errorState: boolean;\n  @Input() errorStateMatcher: ErrorStateMatcher;\n\n  controlType = 'mat-input';\n\n  @HostBinding('attr.aria-describedby') describedBy = '';\n\n  protected editor;\n\n  constructor(\n    // @Host() @Self() @Optional() public editor: CKEditorComponent,\n    protected readonly viewRef: ViewContainerRef,\n    @Optional() @Self() public ngControl: NgControl,\n    @Optional() _parentForm: NgForm,\n    @Optional() _parentFormGroup: FormGroupDirective,\n    _defaultErrorStateMatcher: ErrorStateMatcher,\n  ) {\n    super(_defaultErrorStateMatcher, _parentForm, _parentFormGroup, ngControl);\n  }\n\n  ngOnInit() {\n    // Can't use injection to get component reference\n    // https://github.com/angular/angular/issues/8277\n    this.editor = this.viewRef['_data'].componentView.component;\n    this.editor.blur.subscribe(() => {\n      this.focused = false;\n      this.stateChanges.next();\n    });\n    this.editor.focus.subscribe(() => {\n      this.focused = true;\n      this.stateChanges.next();\n    });\n  }\n\n  ngDoCheck() {\n    if (this.ngControl) {\n      // We need to re-evaluate this on every change detection cycle, because there are some\n      // error triggers that we can't subscribe to (e.g. parent form submissions). This means\n      // that whatever logic is in here has to be super lean or we risk destroying the performance.\n      this.updateErrorState();\n    }\n  }\n\n  setDescribedByIds(ids: string[]) {\n    this.describedBy = ids.join(' ');\n  }\n\n  onContainerClick() {\n    if (this.editor.editorInstance) {\n      this.editor.editorInstance.editing.view.focus();\n      this.stateChanges.next();\n    }\n  }\n\n}\n","import { Directive, Input } from '@angular/core';\nimport { MatFormFieldControl } from '@angular/material/form-field';\n\nimport { MatCkeditorDirective } from './mat-ckeditor.directive';\n\n@Directive({\n  selector: '[matCkeditorBalloon]',\n  providers: [\n    { provide: MatFormFieldControl, useExisting: MatCkeditorBalloonDirective },\n  ]\n})\nexport class MatCkeditorBalloonDirective extends MatCkeditorDirective {\n\n  @Input()\n  set toolbar(show: boolean) {\n    if (this.editor && show !== this.toolbarOpen) {\n      const balloon = this.editor.editorInstance.plugins.get('BalloonToolbar');\n      if (show) {\n        this.showToolbar(balloon);\n      } else {\n        balloon.hide();\n        this.toolbarOpen = false;\n      }\n    }\n  }\n  private toolbarOpen: boolean;\n\n  ngOnInit() {\n    super.ngOnInit();\n    this.editor.ready.subscribe(editor => {\n      const balloon = editor.plugins.get('BalloonToolbar');\n      balloon.stopListening(editor.model.document.selection, 'change:range');\n      balloon.stopListening(balloon, '_selectionChangeDebounced');\n    });\n    this.editor.focus.subscribe(() => {\n      if (this.toolbarOpen) {\n        const balloon = this.editor.editorInstance.plugins.get('BalloonToolbar');\n        this.showToolbar(balloon);\n      }\n    });\n  }\n\n  private showToolbar(balloon) {\n    if (!balloon._balloon.hasView(balloon.toolbarView)) {\n      balloon.listenTo(this.editor.editorInstance.ui, 'update', () => {\n        balloon._balloon.updatePosition(balloon._g