mat-contenteditable
Version:
Angular contenteditable directive for Angular forms and Material Design
213 lines (212 loc) • 15.8 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,uselessCode} checked by tsc
*/
import * as tslib_1 from "tslib";
import { Directive, Input, HostBinding, ViewContainerRef, Optional, Self } from '@angular/core';
import { MatFormFieldControl, ErrorStateMatcher } from '@angular/material';
import { Subject } from 'rxjs';
import { NgControl, NgForm, FormGroupDirective } from '@angular/forms';
import { _MatInputMixinBase } from './mat-contenteditable.directive';
var MatCkeditorDirective = /** @class */ (function (_super) {
tslib_1.__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> </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, 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 }
]; };
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));
export { MatCkeditorDirective };
if (false) {
/**
* Implemented as part of MatFormFieldControl.
* See https://material.angular.io/guide/creating-a-custom-form-field-control
* @type {?}
*/
MatCkeditorDirective.nextId;
/** @type {?} */
MatCkeditorDirective.prototype.stateChanges;
/** @type {?} */
MatCkeditorDirective.prototype.id;
/** @type {?} */
MatCkeditorDirective.prototype.placeholder;
/** @type {?} */
MatCkeditorDirective.prototype.contentEmpty;
/** @type {?} */
MatCkeditorDirective.prototype.focused;
/** @type {?} */
MatCkeditorDirective.prototype.required;
/** @type {?} */
MatCkeditorDirective.prototype.errorState;
/** @type {?} */
MatCkeditorDirective.prototype.errorStateMatcher;
/** @type {?} */
MatCkeditorDirective.prototype.controlType;
/** @type {?} */
MatCkeditorDirective.prototype.describedBy;
/** @type {?} */
MatCkeditorDirective.prototype.editor;
/** @type {?} */
MatCkeditorDirective.prototype.viewRef;
/** @type {?} */
MatCkeditorDirective.prototype.ngControl;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mat-ckeditor.directive.js","sourceRoot":"ng://mat-contenteditable/","sources":["lib/mat-ckeditor.directive.ts"],"names":[],"mappings":"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","sourcesContent":["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"]}