UNPKG

ng-zorro-antd

Version:

An enterprise-class UI components based on Ant Design and Angular

128 lines 17.2 kB
/** * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE */ import { Directive, EventEmitter, forwardRef, Output } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { fromEvent } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { NzDestroyService } from 'ng-zorro-antd/core/services'; import { NZ_MENTION_CONFIG } from './config'; import * as i0 from "@angular/core"; import * as i1 from "ng-zorro-antd/core/services"; import * as i2 from "./mention.service"; export const NZ_MENTION_TRIGGER_ACCESSOR = { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => NzMentionTriggerDirective), multi: true }; export class NzMentionTriggerDirective { constructor(el, ngZone, ref, destroy$, nzMentionService) { this.el = el; this.ngZone = ngZone; this.ref = ref; this.destroy$ = destroy$; this.nzMentionService = nzMentionService; this.onChange = () => { }; this.onTouched = () => { }; // eslint-disable-next-line @angular-eslint/no-output-on-prefix this.onFocusin = new EventEmitter(); // eslint-disable-next-line @angular-eslint/no-output-on-prefix this.onBlur = new EventEmitter(); // eslint-disable-next-line @angular-eslint/no-output-on-prefix this.onInput = new EventEmitter(); // eslint-disable-next-line @angular-eslint/no-output-on-prefix this.onKeydown = new EventEmitter(); // eslint-disable-next-line @angular-eslint/no-output-on-prefix this.onClick = new EventEmitter(); } completeEvents() { this.onFocusin.complete(); this.onBlur.complete(); this.onInput.complete(); this.onKeydown.complete(); this.onClick.complete(); } focus(caretPos = null) { this.el.nativeElement.focus(); this.el.nativeElement.setSelectionRange(caretPos, caretPos); } insertMention(mention) { const value = this.el.nativeElement.value; const insertValue = `${mention.mention}${NZ_MENTION_CONFIG.split}`; const newValue = [ value.slice(0, mention.startPos + 1), insertValue, value.slice(mention.endPos, value.length) ].join(''); this.el.nativeElement.value = newValue; this.focus(mention.startPos + insertValue.length + 1); this.onChange(newValue); this.value = newValue; } writeValue(value) { this.value = value; if (typeof value === 'string') { this.el.nativeElement.value = value; } else { this.el.nativeElement.value = ''; } } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } ngAfterViewInit() { this.nzMentionService.registerTrigger(this); this.setupEventListener('blur', this.onBlur); this.setupEventListener('focusin', this.onFocusin); this.setupEventListener('input', this.onInput, true); this.setupEventListener('click', this.onClick, true); this.setupEventListener('keydown', this.onKeydown, true); } ngOnDestroy() { this.completeEvents(); } setupEventListener(eventName, eventEmitter, shouldPassEvent = false) { this.ngZone.runOutsideAngular(() => { fromEvent(this.el.nativeElement, eventName) .pipe(takeUntil(this.destroy$)) .subscribe(event => { if (eventEmitter.observers.length) { this.ngZone.run(() => { eventEmitter.emit(shouldPassEvent ? event : undefined); this.ref.markForCheck(); }); } }); }); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.1.2", ngImport: i0, type: NzMentionTriggerDirective, deps: [{ token: i0.ElementRef }, { token: i0.NgZone }, { token: i0.ChangeDetectorRef }, { token: i1.NzDestroyService }, { token: i2.NzMentionService }], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.1.2", type: NzMentionTriggerDirective, isStandalone: true, selector: "input[nzMentionTrigger], textarea[nzMentionTrigger]", outputs: { onFocusin: "onFocusin", onBlur: "onBlur", onInput: "onInput", onKeydown: "onKeydown", onClick: "onClick" }, host: { attributes: { "autocomplete": "off" } }, providers: [NzDestroyService, NZ_MENTION_TRIGGER_ACCESSOR], exportAs: ["nzMentionTrigger"], ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.1.2", ngImport: i0, type: NzMentionTriggerDirective, decorators: [{ type: Directive, args: [{ selector: 'input[nzMentionTrigger], textarea[nzMentionTrigger]', exportAs: 'nzMentionTrigger', providers: [NzDestroyService, NZ_MENTION_TRIGGER_ACCESSOR], host: { autocomplete: 'off' }, standalone: true }] }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.NgZone }, { type: i0.ChangeDetectorRef }, { type: i1.NzDestroyService }, { type: i2.NzMentionService }], propDecorators: { onFocusin: [{ type: Output }], onBlur: [{ type: Output }], onInput: [{ type: Output }], onKeydown: [{ type: Output }], onClick: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mention-trigger.js","sourceRoot":"","sources":["../../../components/mention/mention-trigger.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAGL,SAAS,EAET,YAAY,EAEZ,UAAU,EAGV,MAAM,EACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAE3C,OAAO,EAAE,gBAAgB,EAAE,MAAM,6BAA6B,CAAC;AAG/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,UAAU,CAAC;;;;AAI7C,MAAM,CAAC,MAAM,2BAA2B,GAAqB;IAC3D,OAAO,EAAE,iBAAiB;IAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,yBAAyB,CAAC;IACxD,KAAK,EAAE,IAAI;CACZ,CAAC;AAWF,MAAM,OAAO,yBAAyB;IAgBpC,YACS,EAAsD,EACrD,MAAc,EACd,GAAsB,EACtB,QAA0B,EAC1B,gBAAkC;QAJnC,OAAE,GAAF,EAAE,CAAoD;QACrD,WAAM,GAAN,MAAM,CAAQ;QACd,QAAG,GAAH,GAAG,CAAmB;QACtB,aAAQ,GAAR,QAAQ,CAAkB;QAC1B,qBAAgB,GAAhB,gBAAgB,CAAkB;QApB5C,aAAQ,GAAiB,GAAG,EAAE,GAAE,CAAC,CAAC;QAClC,cAAS,GAAkB,GAAG,EAAE,GAAE,CAAC,CAAC;QAEpC,+DAA+D;QAC5C,cAAS,GAAuB,IAAI,YAAY,EAAE,CAAC;QACtE,+DAA+D;QAC5C,WAAM,GAAuB,IAAI,YAAY,EAAE,CAAC;QACnE,+DAA+D;QAC5C,YAAO,GAAgC,IAAI,YAAY,EAAE,CAAC;QAC7E,+DAA+D;QAC5C,cAAS,GAAgC,IAAI,YAAY,EAAE,CAAC;QAC/E,+DAA+D;QAC5C,YAAO,GAA6B,IAAI,YAAY,EAAE,CAAC;IASvE,CAAC;IAEJ,cAAc;QACZ,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;QAC1B,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;QACvB,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;QACxB,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;QAC1B,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,WAA0B,IAAI;QAClC,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;QAC9B,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,iBAAiB,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC9D,CAAC;IAED,aAAa,CAAC,OAAgB;QAC5B,MAAM,KAAK,GAAW,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,CAAC;QAClD,MAAM,WAAW,GAAG,GAAG,OAAO,CAAC,OAAO,GAAG,iBAAiB,CAAC,KAAK,EAAE,CAAC;QACnE,MAAM,QAAQ,GAAG;YACf,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;YACpC,WAAW;YACX,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC;SAC1C,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACX,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,QAAQ,CAAC;QACvC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QACtD,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;QACxB,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC;IACxB,CAAC;IAED,UAAU,CAAC,KAAa;QACtB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,KAAK,CAAC;QACtC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,GAAG,EAAE,CAAC;QACnC,CAAC;IACH,CAAC;IAED,gBAAgB,CAAC,EAA2B;QAC1C,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,EAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAED,eAAe;QACb,IAAI,CAAC,gBAAgB,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;QAE5C,IAAI,CAAC,kBAAkB,CAAC,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;QAC7C,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;QACnD,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QACrD,IAAI,CAAC,kBAAkB,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QACrD,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;IAC3D,CAAC;IAED,WAAW;QACT,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAEO,kBAAkB,CAAI,SAAiB,EAAE,YAA6B,EAAE,eAAe,GAAG,KAAK;QACrG,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE;YACjC,SAAS,CAAI,IAAI,CAAC,EAAE,CAAC,aAAa,EAAE,SAAS,CAAC;iBAC3C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBAC9B,SAAS,CAAC,KAAK,CAAC,EAAE;gBACjB,IAAI,YAAY,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC;oBAClC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;wBACnB,YAAY,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;wBACvD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;oBAC1B,CAAC,CAAC,CAAC;gBACL,CAAC;YACH,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACL,CAAC;8GA/FU,yBAAyB;kGAAzB,yBAAyB,0QANzB,CAAC,gBAAgB,EAAE,2BAA2B,CAAC;;2FAM/C,yBAAyB;kBATrC,SAAS;mBAAC;oBACT,QAAQ,EAAE,qDAAqD;oBAC/D,QAAQ,EAAE,kBAAkB;oBAC5B,SAAS,EAAE,CAAC,gBAAgB,EAAE,2BAA2B,CAAC;oBAC1D,IAAI,EAAE;wBACJ,YAAY,EAAE,KAAK;qBACpB;oBACD,UAAU,EAAE,IAAI;iBACjB;kMAMoB,SAAS;sBAA3B,MAAM;gBAEY,MAAM;sBAAxB,MAAM;gBAEY,OAAO;sBAAzB,MAAM;gBAEY,SAAS;sBAA3B,MAAM;gBAEY,OAAO;sBAAzB,MAAM","sourcesContent":["/**\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://github.com/NG-ZORRO/ng-zorro-antd/blob/master/LICENSE\n */\n\nimport {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  ExistingProvider,\n  forwardRef,\n  NgZone,\n  OnDestroy,\n  Output\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\nimport { fromEvent } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport { NzDestroyService } from 'ng-zorro-antd/core/services';\nimport { OnChangeType, OnTouchedType } from 'ng-zorro-antd/core/types';\n\nimport { NZ_MENTION_CONFIG } from './config';\nimport { Mention } from './mention.component';\nimport { NzMentionService } from './mention.service';\n\nexport const NZ_MENTION_TRIGGER_ACCESSOR: ExistingProvider = {\n  provide: NG_VALUE_ACCESSOR,\n  useExisting: forwardRef(() => NzMentionTriggerDirective),\n  multi: true\n};\n\n@Directive({\n  selector: 'input[nzMentionTrigger], textarea[nzMentionTrigger]',\n  exportAs: 'nzMentionTrigger',\n  providers: [NzDestroyService, NZ_MENTION_TRIGGER_ACCESSOR],\n  host: {\n    autocomplete: 'off'\n  },\n  standalone: true\n})\nexport class NzMentionTriggerDirective implements ControlValueAccessor, OnDestroy, AfterViewInit {\n  onChange: OnChangeType = () => {};\n  onTouched: OnTouchedType = () => {};\n\n  // eslint-disable-next-line @angular-eslint/no-output-on-prefix\n  @Output() readonly onFocusin: EventEmitter<void> = new EventEmitter();\n  // eslint-disable-next-line @angular-eslint/no-output-on-prefix\n  @Output() readonly onBlur: EventEmitter<void> = new EventEmitter();\n  // eslint-disable-next-line @angular-eslint/no-output-on-prefix\n  @Output() readonly onInput: EventEmitter<KeyboardEvent> = new EventEmitter();\n  // eslint-disable-next-line @angular-eslint/no-output-on-prefix\n  @Output() readonly onKeydown: EventEmitter<KeyboardEvent> = new EventEmitter();\n  // eslint-disable-next-line @angular-eslint/no-output-on-prefix\n  @Output() readonly onClick: EventEmitter<MouseEvent> = new EventEmitter();\n  value?: string;\n\n  constructor(\n    public el: ElementRef<HTMLInputElement | HTMLTextAreaElement>,\n    private ngZone: NgZone,\n    private ref: ChangeDetectorRef,\n    private destroy$: NzDestroyService,\n    private nzMentionService: NzMentionService\n  ) {}\n\n  completeEvents(): void {\n    this.onFocusin.complete();\n    this.onBlur.complete();\n    this.onInput.complete();\n    this.onKeydown.complete();\n    this.onClick.complete();\n  }\n\n  focus(caretPos: number | null = null): void {\n    this.el.nativeElement.focus();\n    this.el.nativeElement.setSelectionRange(caretPos, caretPos);\n  }\n\n  insertMention(mention: Mention): void {\n    const value: string = this.el.nativeElement.value;\n    const insertValue = `${mention.mention}${NZ_MENTION_CONFIG.split}`;\n    const newValue = [\n      value.slice(0, mention.startPos + 1),\n      insertValue,\n      value.slice(mention.endPos, value.length)\n    ].join('');\n    this.el.nativeElement.value = newValue;\n    this.focus(mention.startPos + insertValue.length + 1);\n    this.onChange(newValue);\n    this.value = newValue;\n  }\n\n  writeValue(value: string): void {\n    this.value = value;\n    if (typeof value === 'string') {\n      this.el.nativeElement.value = value;\n    } else {\n      this.el.nativeElement.value = '';\n    }\n  }\n\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  ngAfterViewInit(): void {\n    this.nzMentionService.registerTrigger(this);\n\n    this.setupEventListener('blur', this.onBlur);\n    this.setupEventListener('focusin', this.onFocusin);\n    this.setupEventListener('input', this.onInput, true);\n    this.setupEventListener('click', this.onClick, true);\n    this.setupEventListener('keydown', this.onKeydown, true);\n  }\n\n  ngOnDestroy(): void {\n    this.completeEvents();\n  }\n\n  private setupEventListener<T>(eventName: string, eventEmitter: EventEmitter<T>, shouldPassEvent = false): void {\n    this.ngZone.runOutsideAngular(() => {\n      fromEvent<T>(this.el.nativeElement, eventName)\n        .pipe(takeUntil(this.destroy$))\n        .subscribe(event => {\n          if (eventEmitter.observers.length) {\n            this.ngZone.run(() => {\n              eventEmitter.emit(shouldPassEvent ? event : undefined);\n              this.ref.markForCheck();\n            });\n          }\n        });\n    });\n  }\n}\n"]}