ng-zorro-antd
Version:
An enterprise-class UI components based on Ant Design and Angular
128 lines • 17.2 kB
JavaScript
/**
* 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":"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","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"]}