@libs-ui/components-inputs-mention 0.1.1-1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -0
- package/defines/caret-coords.define.d.ts +5 -0
- package/defines/template.define.d.ts +4 -0
- package/defines/utils.define.d.ts +18 -0
- package/esm2022/defines/caret-coords.define.mjs +115 -0
- package/esm2022/defines/template.define.mjs +26 -0
- package/esm2022/defines/utils.define.mjs +279 -0
- package/esm2022/index.mjs +3 -0
- package/esm2022/interfaces/mention.interface.mjs +2 -0
- package/esm2022/libs-ui-components-inputs-mention.mjs +5 -0
- package/esm2022/list/list.component.mjs +165 -0
- package/esm2022/mention.directive.mjs +507 -0
- package/fesm2022/libs-ui-components-inputs-mention.mjs +1084 -0
- package/fesm2022/libs-ui-components-inputs-mention.mjs.map +1 -0
- package/index.d.ts +2 -0
- package/interfaces/mention.interface.d.ts +40 -0
- package/list/list.component.d.ts +36 -0
- package/mention.directive.d.ts +49 -0
- package/package.json +30 -0
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, model, signal, viewChild, viewChildren } from '@angular/core';
|
|
2
|
+
import { LibsUiComponentsAvatarComponent } from '@libs-ui/components-avatar';
|
|
3
|
+
import { LibsUiComponentsPopoverComponent } from '@libs-ui/components-popover';
|
|
4
|
+
import { set } from '@libs-ui/utils';
|
|
5
|
+
import { fromEvent, Subject, takeUntil } from 'rxjs';
|
|
6
|
+
import { getCaretCoordinates } from '../defines/caret-coords.define';
|
|
7
|
+
import { UtilsKeyCodeConstant } from '@libs-ui/utils';
|
|
8
|
+
import { getContentEditableCaretCoords, isInputOrTextAreaElement } from '../defines/utils.define';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
export class LibsUiComponentsInputsMentionListComponent {
|
|
11
|
+
// #region PROPERTY
|
|
12
|
+
items = signal([]);
|
|
13
|
+
activeIndex = signal(0);
|
|
14
|
+
hidden = signal(false);
|
|
15
|
+
styleOff = signal(false);
|
|
16
|
+
parentHandlerKeyDown = signal(undefined);
|
|
17
|
+
dropUp = signal(false);
|
|
18
|
+
coords = signal({ top: 0, left: 0, bottom: 0 });
|
|
19
|
+
offset = signal(0);
|
|
20
|
+
nativeParentElement = signal(undefined);
|
|
21
|
+
onDestroy = new Subject();
|
|
22
|
+
// #region INPUT
|
|
23
|
+
labelKey = model('label');
|
|
24
|
+
zIndex = model(1000);
|
|
25
|
+
isIframe = model.required();
|
|
26
|
+
defaultAvatar = model();
|
|
27
|
+
/* VIEW CHILD */
|
|
28
|
+
elementEl = viewChild('element');
|
|
29
|
+
itemsEl = viewChildren('item');
|
|
30
|
+
ngOnInit() {
|
|
31
|
+
if (this.isIframe()) {
|
|
32
|
+
this.initEvent(window.parent, 'wheel');
|
|
33
|
+
this.initEvent(window.parent, 'resize');
|
|
34
|
+
return;
|
|
35
|
+
}
|
|
36
|
+
this.initEvent(window, 'wheel');
|
|
37
|
+
this.initEvent(window, 'resize');
|
|
38
|
+
}
|
|
39
|
+
/* FUNCTIONS */
|
|
40
|
+
initEvent(element, eventName) {
|
|
41
|
+
fromEvent(element, eventName)
|
|
42
|
+
.pipe(takeUntil(this.onDestroy))
|
|
43
|
+
.subscribe((event) => {
|
|
44
|
+
if (eventName === 'resize' || !this.elementEl()?.nativeElement.contains(event.target)) {
|
|
45
|
+
this.handlerListenClose(event);
|
|
46
|
+
}
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
handlerListenClose(event) {
|
|
50
|
+
if (event.target !== this.elementEl()?.nativeElement) {
|
|
51
|
+
this.hidden.set(true);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
// lots of confusion here between relative coordinates and containers
|
|
55
|
+
position(nativeParentElement, iframe, leftDiv) {
|
|
56
|
+
this.nativeParentElement.set(nativeParentElement);
|
|
57
|
+
if (isInputOrTextAreaElement(nativeParentElement)) {
|
|
58
|
+
// parent elements need to have postition:relative for this to work correctly?
|
|
59
|
+
this.coords.set(getCaretCoordinates(nativeParentElement, nativeParentElement.selectionStart));
|
|
60
|
+
this.coords.update((item) => ({
|
|
61
|
+
...item,
|
|
62
|
+
top: nativeParentElement.offsetTop + item.top - nativeParentElement.scrollTop,
|
|
63
|
+
left: nativeParentElement.offsetLeft + item.left - nativeParentElement.scrollLeft + nativeParentElement.getBoundingClientRect().left,
|
|
64
|
+
}));
|
|
65
|
+
// getCretCoordinates() for text/input elements needs an additional offset to position the list correctly
|
|
66
|
+
this.offset.set(this.getBlockCursorDimensions(nativeParentElement).height);
|
|
67
|
+
this.positionElement();
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
if (iframe) {
|
|
71
|
+
const context = { iframe: iframe, parent: window.parent.document.body, windowParent: true };
|
|
72
|
+
// const rect = iframe.getBoundingClientRect();
|
|
73
|
+
const caretRelativeToView = getContentEditableCaretCoords(context);
|
|
74
|
+
const doc = document.documentElement;
|
|
75
|
+
const scrollLeft = (window.scrollX || doc.scrollLeft) - (doc.clientLeft || 0);
|
|
76
|
+
this.coords.update((item) => ({
|
|
77
|
+
...item,
|
|
78
|
+
left: caretRelativeToView.left - scrollLeft - (leftDiv || 0),
|
|
79
|
+
bottom: window.parent.innerHeight - caretRelativeToView.top + 18,
|
|
80
|
+
}));
|
|
81
|
+
if (caretRelativeToView.bottom && caretRelativeToView.bottom !== -1) {
|
|
82
|
+
this.coords.update((item) => ({
|
|
83
|
+
...item,
|
|
84
|
+
bottom: caretRelativeToView.bottom,
|
|
85
|
+
}));
|
|
86
|
+
}
|
|
87
|
+
this.positionElement();
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
const doc = document.documentElement;
|
|
91
|
+
const scrollLeft = (window.scrollX || doc.scrollLeft) - (doc.clientLeft || 0);
|
|
92
|
+
const scrollTop = (window.scrollX || doc.scrollTop) - (doc.clientTop || 0);
|
|
93
|
+
// bounding rectangles are relative to view, offsets are relative to container?
|
|
94
|
+
const caretRelativeToView = getContentEditableCaretCoords({ iframe, parent: null });
|
|
95
|
+
this.coords.update((item) => ({
|
|
96
|
+
...item,
|
|
97
|
+
top: caretRelativeToView.top - scrollTop + 10,
|
|
98
|
+
left: caretRelativeToView.left - scrollLeft - (leftDiv || 0),
|
|
99
|
+
bottom: caretRelativeToView.bottom,
|
|
100
|
+
}));
|
|
101
|
+
this.positionElement();
|
|
102
|
+
}
|
|
103
|
+
get ActiveItem() {
|
|
104
|
+
return this.items()[this.activeIndex()];
|
|
105
|
+
}
|
|
106
|
+
handlerClick(event, index) {
|
|
107
|
+
event.stopPropagation();
|
|
108
|
+
this.activeIndex.set(index);
|
|
109
|
+
set(event, 'keyCode', UtilsKeyCodeConstant.ENTER);
|
|
110
|
+
this.parentHandlerKeyDown()?.(event, this.nativeParentElement());
|
|
111
|
+
}
|
|
112
|
+
activateNextItem() {
|
|
113
|
+
this.activeIndex.set(this.items().length - 1 > this.activeIndex() ? this.activeIndex() + 1 : this.activeIndex());
|
|
114
|
+
this.scrollToActive();
|
|
115
|
+
}
|
|
116
|
+
activatePreviousItem() {
|
|
117
|
+
this.activeIndex.set(this.activeIndex() > 0 ? this.activeIndex() - 1 : this.activeIndex());
|
|
118
|
+
this.scrollToActive();
|
|
119
|
+
}
|
|
120
|
+
positionElement(left = this.coords().left, top = this.coords().top, dropUp = this.dropUp(), bottom = this.coords().bottom || 0) {
|
|
121
|
+
const el = this.elementEl()?.nativeElement;
|
|
122
|
+
top += dropUp ? 0 : this.offset(); // top of list is next line
|
|
123
|
+
el.style.position = 'absolute';
|
|
124
|
+
el.style.left = left + 'px';
|
|
125
|
+
el.style.top = bottom !== -1 ? 'auto' : top + 'px';
|
|
126
|
+
el.style.bottom = bottom === -1 ? 'auto' : bottom + 'px';
|
|
127
|
+
this.nativeParentElement()?.normalize();
|
|
128
|
+
}
|
|
129
|
+
getBlockCursorDimensions(nativeParentElement) {
|
|
130
|
+
const parentStyles = window.getComputedStyle(nativeParentElement);
|
|
131
|
+
return {
|
|
132
|
+
height: parseFloat(parentStyles.lineHeight),
|
|
133
|
+
width: parseFloat(parentStyles.fontSize),
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
handleActiveItem(event, index) {
|
|
137
|
+
event.stopPropagation();
|
|
138
|
+
this.activeIndex.set(index);
|
|
139
|
+
}
|
|
140
|
+
scrollToActive() {
|
|
141
|
+
if (!this.itemsEl() || this.itemsEl().length <= this.activeIndex()) {
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
this.itemsEl()[this.activeIndex()].nativeElement.scrollIntoView();
|
|
145
|
+
}
|
|
146
|
+
scrollContent() {
|
|
147
|
+
setTimeout(() => {
|
|
148
|
+
const element = this.elementEl();
|
|
149
|
+
if (element && element.nativeElement) {
|
|
150
|
+
element.nativeElement.scrollTop = 0;
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
ngOnDestroy() {
|
|
155
|
+
this.onDestroy.next();
|
|
156
|
+
this.onDestroy.complete();
|
|
157
|
+
}
|
|
158
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LibsUiComponentsInputsMentionListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
159
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: LibsUiComponentsInputsMentionListComponent, isStandalone: true, selector: "libs_ui-components-inputs-mention-list", inputs: { labelKey: { classPropertyName: "labelKey", publicName: "labelKey", isSignal: true, isRequired: false, transformFunction: null }, zIndex: { classPropertyName: "zIndex", publicName: "zIndex", isSignal: true, isRequired: false, transformFunction: null }, isIframe: { classPropertyName: "isIframe", publicName: "isIframe", isSignal: true, isRequired: true, transformFunction: null }, defaultAvatar: { classPropertyName: "defaultAvatar", publicName: "defaultAvatar", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { labelKey: "labelKeyChange", zIndex: "zIndexChange", isIframe: "isIframeChange", defaultAvatar: "defaultAvatarChange" }, viewQueries: [{ propertyName: "elementEl", first: true, predicate: ["element"], descendants: true, isSignal: true }, { propertyName: "itemsEl", predicate: ["item"], descendants: true, isSignal: true }], ngImport: i0, template: "<div\n #element\n class=\"libs-ui-mention-list\"\n [style.zIndex]=\"zIndex()\"\n [class.hidden]=\"hidden()\">\n @for (item of items(); track item) {\n <div\n #item\n class=\"libs-ui-mention-list-item\"\n [class.libs-ui-mention-list-item-active]=\"$index === activeIndex()\"\n (mousedown)=\"handlerClick($event, $index)\"\n (mouseenter)=\"handleActiveItem($event, $index)\">\n <div class=\"px-[12px] py-[6px] flex items-center\">\n <libs_ui-components-avatar\n [size]=\"24\"\n [linkAvatar]=\"item.avatar\"\n [linkAvatarError]=\"defaultAvatar()\"\n [getLastTextAfterSpace]=\"true\"\n [textAvatar]=\"item.name\"\n [idGenColor]=\"item.id\" />\n <libs_ui-components-popover\n [classInclude]=\"'libs-ui-font-h5r'\"\n [type]=\"'text'\"\n [ignoreShowPopover]=\"true\"\n [config]=\"{ width: 250 }\">\n <!-- // ta.m \u1EA9n show tooltip v\u00EC ch\u01B0a x\u1EED l\u00FD \u0111c v\u1ECB tr\u00ED -->\n {{ item.name + ' (' + item.username + ')' }}\n </libs_ui-components-popover>\n </div>\n </div>\n }\n</div>\n", styles: [".libs-ui-mention-list{position:absolute;top:-999px;left:-9999px;z-index:1202;float:left;width:315px;padding:4px 0;background-color:#fff;border:solid 1px #e6e8ed;border-radius:5px;max-height:190px;overflow:auto;box-shadow:0 2px 10px 1px #3333331a}.libs-ui-mention-list .libs-ui-mention-list-item{background:#fff;cursor:pointer}.libs-ui-mention-list .libs-ui-mention-list-item:hover,.libs-ui-mention-list .libs-ui-mention-list-item-active{background-color:var(--libs-ui-color-light-3, #f4f8ff)}\n"], dependencies: [{ kind: "component", type: LibsUiComponentsAvatarComponent, selector: "libs_ui-components-avatar", inputs: ["typeShape", "classInclude", "size", "linkAvatar", "linkAvatarError", "classImageInclude", "zIndexPreviewImage", "clickPreviewImage", "idGenColor", "getLastTextAfterSpace", "textAvatar", "textAvatarClassInclude", "containertextAvatarClassInclude"], outputs: ["outAvatarError", "outEventPreviewImage"] }, { kind: "component", type: LibsUiComponentsPopoverComponent, selector: "libs_ui-components-popover,[LibsUiComponentsPopoverDirective]", inputs: ["debugId", "flagMouse", "type", "mode", "config", "ignoreShowPopover", "elementRefCustom", "initEventInElementRefCustom", "classInclude", "ignoreHiddenPopoverContentWhenMouseLeave", "ignoreStopPropagationEvent", "ignoreCursorPointerModeLikeClick", "isAddContentToParentDocument", "ignoreClickOutside"], outputs: ["outEvent", "outChangStageFlagMouse", "outEventPopoverContent", "outFunctionsControl"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
160
|
+
}
|
|
161
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LibsUiComponentsInputsMentionListComponent, decorators: [{
|
|
162
|
+
type: Component,
|
|
163
|
+
args: [{ selector: 'libs_ui-components-inputs-mention-list', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, imports: [LibsUiComponentsAvatarComponent, LibsUiComponentsPopoverComponent], template: "<div\n #element\n class=\"libs-ui-mention-list\"\n [style.zIndex]=\"zIndex()\"\n [class.hidden]=\"hidden()\">\n @for (item of items(); track item) {\n <div\n #item\n class=\"libs-ui-mention-list-item\"\n [class.libs-ui-mention-list-item-active]=\"$index === activeIndex()\"\n (mousedown)=\"handlerClick($event, $index)\"\n (mouseenter)=\"handleActiveItem($event, $index)\">\n <div class=\"px-[12px] py-[6px] flex items-center\">\n <libs_ui-components-avatar\n [size]=\"24\"\n [linkAvatar]=\"item.avatar\"\n [linkAvatarError]=\"defaultAvatar()\"\n [getLastTextAfterSpace]=\"true\"\n [textAvatar]=\"item.name\"\n [idGenColor]=\"item.id\" />\n <libs_ui-components-popover\n [classInclude]=\"'libs-ui-font-h5r'\"\n [type]=\"'text'\"\n [ignoreShowPopover]=\"true\"\n [config]=\"{ width: 250 }\">\n <!-- // ta.m \u1EA9n show tooltip v\u00EC ch\u01B0a x\u1EED l\u00FD \u0111c v\u1ECB tr\u00ED -->\n {{ item.name + ' (' + item.username + ')' }}\n </libs_ui-components-popover>\n </div>\n </div>\n }\n</div>\n", styles: [".libs-ui-mention-list{position:absolute;top:-999px;left:-9999px;z-index:1202;float:left;width:315px;padding:4px 0;background-color:#fff;border:solid 1px #e6e8ed;border-radius:5px;max-height:190px;overflow:auto;box-shadow:0 2px 10px 1px #3333331a}.libs-ui-mention-list .libs-ui-mention-list-item{background:#fff;cursor:pointer}.libs-ui-mention-list .libs-ui-mention-list-item:hover,.libs-ui-mention-list .libs-ui-mention-list-item-active{background-color:var(--libs-ui-color-light-3, #f4f8ff)}\n"] }]
|
|
164
|
+
}] });
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"list.component.js","sourceRoot":"","sources":["../../../../../../../libs-ui/components/inputs/mention/src/list/list.component.ts","../../../../../../../libs-ui/components/inputs/mention/src/list/list.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, model, OnDestroy, OnInit, signal, viewChild, viewChildren } from '@angular/core';\nimport { LibsUiComponentsAvatarComponent } from '@libs-ui/components-avatar';\nimport { LibsUiComponentsPopoverComponent } from '@libs-ui/components-popover';\nimport { set } from '@libs-ui/utils';\nimport { fromEvent, Subject, takeUntil } from 'rxjs';\nimport { getCaretCoordinates } from '../defines/caret-coords.define';\nimport { UtilsKeyCodeConstant } from '@libs-ui/utils';\nimport { getContentEditableCaretCoords, isInputOrTextAreaElement } from '../defines/utils.define';\n\n@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'libs_ui-components-inputs-mention-list',\n  templateUrl: './list.component.html',\n  styleUrls: ['./list.component.scss'],\n  standalone: true,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [LibsUiComponentsAvatarComponent, LibsUiComponentsPopoverComponent],\n})\nexport class LibsUiComponentsInputsMentionListComponent implements OnInit, OnDestroy {\n  // #region PROPERTY\n  public items = signal<Array<Record<string, string | undefined>>>([]);\n  public activeIndex = signal<number>(0);\n  public hidden = signal<boolean>(false);\n  public styleOff = signal<boolean>(false);\n  public parentHandlerKeyDown = signal<((e: Event, element: HTMLInputElement | undefined) => unknown) | undefined>(undefined);\n\n  protected dropUp = signal<boolean>(false);\n\n  private coords = signal<{ top: number; left: number; bottom?: number }>({ top: 0, left: 0, bottom: 0 });\n  private offset = signal<number>(0);\n  private nativeParentElement = signal<HTMLInputElement | undefined>(undefined);\n  private onDestroy = new Subject<void>();\n\n  // #region INPUT\n  labelKey = model<string>('label');\n  zIndex = model<number>(1000);\n  isIframe = model.required<boolean>();\n  defaultAvatar = model<string>();\n\n  /* VIEW CHILD */\n  readonly elementEl = viewChild<ElementRef>('element');\n  readonly itemsEl = viewChildren<ElementRef>('item');\n\n  ngOnInit() {\n    if (this.isIframe()) {\n      this.initEvent(window.parent, 'wheel');\n      this.initEvent(window.parent, 'resize');\n      return;\n    }\n    this.initEvent(window, 'wheel');\n    this.initEvent(window, 'resize');\n  }\n\n  /* FUNCTIONS */\n  private initEvent(element: Window, eventName: string) {\n    fromEvent<Event>(element, eventName)\n      .pipe(takeUntil(this.onDestroy))\n      .subscribe((event: Event) => {\n        if (eventName === 'resize' || !this.elementEl()?.nativeElement.contains(event.target)) {\n          this.handlerListenClose(event);\n        }\n      });\n  }\n\n  private handlerListenClose(event: Event) {\n    if (event.target !== this.elementEl()?.nativeElement) {\n      this.hidden.set(true);\n    }\n  }\n\n  // lots of confusion here between relative coordinates and containers\n  position(nativeParentElement: HTMLInputElement, iframe?: HTMLIFrameElement, leftDiv?: number): void {\n    this.nativeParentElement.set(nativeParentElement);\n    if (isInputOrTextAreaElement(nativeParentElement)) {\n      // parent elements need to have postition:relative for this to work correctly?\n      this.coords.set(getCaretCoordinates(nativeParentElement, nativeParentElement.selectionStart));\n      this.coords.update((item) => ({\n        ...item,\n        top: nativeParentElement.offsetTop + item.top - nativeParentElement.scrollTop,\n        left: nativeParentElement.offsetLeft + item.left - nativeParentElement.scrollLeft + nativeParentElement.getBoundingClientRect().left,\n      }));\n\n      // getCretCoordinates() for text/input elements needs an additional offset to position the list correctly\n      this.offset.set(this.getBlockCursorDimensions(nativeParentElement).height);\n      this.positionElement();\n      return;\n    }\n    if (iframe) {\n      const context: { iframe: HTMLIFrameElement; parent: Element | null; windowParent?: boolean } = { iframe: iframe, parent: window.parent.document.body, windowParent: true };\n      // const rect = iframe.getBoundingClientRect();\n      const caretRelativeToView = getContentEditableCaretCoords(context);\n      const doc = document.documentElement;\n      const scrollLeft = (window.scrollX || doc.scrollLeft) - (doc.clientLeft || 0);\n      this.coords.update((item) => ({\n        ...item,\n        left: caretRelativeToView.left - scrollLeft - (leftDiv || 0),\n        bottom: window.parent.innerHeight - caretRelativeToView.top + 18,\n      }));\n      if (caretRelativeToView.bottom && caretRelativeToView.bottom !== -1) {\n        this.coords.update((item) => ({\n          ...item,\n          bottom: caretRelativeToView.bottom,\n        }));\n      }\n      this.positionElement();\n\n      return;\n    }\n    const doc = document.documentElement;\n    const scrollLeft = (window.scrollX || doc.scrollLeft) - (doc.clientLeft || 0);\n    const scrollTop = (window.scrollX || doc.scrollTop) - (doc.clientTop || 0);\n    // bounding rectangles are relative to view, offsets are relative to container?\n    const caretRelativeToView = getContentEditableCaretCoords({ iframe, parent: null });\n\n    this.coords.update((item) => ({\n      ...item,\n      top: caretRelativeToView.top - scrollTop + 10,\n      left: caretRelativeToView.left - scrollLeft - (leftDiv || 0),\n      bottom: caretRelativeToView.bottom,\n    }));\n    this.positionElement();\n  }\n\n  get ActiveItem() {\n    return this.items()[this.activeIndex()];\n  }\n\n  protected handlerClick(event: Event, index: number) {\n    event.stopPropagation();\n    this.activeIndex.set(index);\n    set(event as KeyboardEvent, 'keyCode', UtilsKeyCodeConstant.ENTER);\n    this.parentHandlerKeyDown()?.(event, this.nativeParentElement());\n  }\n\n  activateNextItem() {\n    this.activeIndex.set(this.items().length - 1 > this.activeIndex() ? this.activeIndex() + 1 : this.activeIndex());\n    this.scrollToActive();\n  }\n\n  activatePreviousItem() {\n    this.activeIndex.set(this.activeIndex() > 0 ? this.activeIndex() - 1 : this.activeIndex());\n    this.scrollToActive();\n  }\n\n  private positionElement(left: number = this.coords().left, top: number = this.coords().top, dropUp: boolean = this.dropUp(), bottom: number = this.coords().bottom || 0) {\n    const el = this.elementEl()?.nativeElement;\n\n    top += dropUp ? 0 : this.offset(); // top of list is next line\n    el.style.position = 'absolute';\n    el.style.left = left + 'px';\n    el.style.top = bottom !== -1 ? 'auto' : top + 'px';\n    el.style.bottom = bottom === -1 ? 'auto' : bottom + 'px';\n    this.nativeParentElement()?.normalize();\n  }\n\n  private getBlockCursorDimensions(nativeParentElement: HTMLInputElement) {\n    const parentStyles = window.getComputedStyle(nativeParentElement);\n\n    return {\n      height: parseFloat(parentStyles.lineHeight),\n      width: parseFloat(parentStyles.fontSize),\n    };\n  }\n\n  protected handleActiveItem(event: Event, index: number) {\n    event.stopPropagation();\n    this.activeIndex.set(index);\n  }\n\n  private scrollToActive() {\n    if (!this.itemsEl() || this.itemsEl().length <= this.activeIndex()) {\n      return;\n    }\n    this.itemsEl()[this.activeIndex()].nativeElement.scrollIntoView();\n  }\n\n  scrollContent() {\n    setTimeout(() => {\n      const element = this.elementEl();\n      if (element && element.nativeElement) {\n        element.nativeElement.scrollTop = 0;\n      }\n    });\n  }\n\n  ngOnDestroy(): void {\n    this.onDestroy.next();\n    this.onDestroy.complete();\n  }\n}\n","<div\n  #element\n  class=\"libs-ui-mention-list\"\n  [style.zIndex]=\"zIndex()\"\n  [class.hidden]=\"hidden()\">\n  @for (item of items(); track item) {\n    <div\n      #item\n      class=\"libs-ui-mention-list-item\"\n      [class.libs-ui-mention-list-item-active]=\"$index === activeIndex()\"\n      (mousedown)=\"handlerClick($event, $index)\"\n      (mouseenter)=\"handleActiveItem($event, $index)\">\n      <div class=\"px-[12px] py-[6px] flex items-center\">\n        <libs_ui-components-avatar\n          [size]=\"24\"\n          [linkAvatar]=\"item.avatar\"\n          [linkAvatarError]=\"defaultAvatar()\"\n          [getLastTextAfterSpace]=\"true\"\n          [textAvatar]=\"item.name\"\n          [idGenColor]=\"item.id\" />\n        <libs_ui-components-popover\n          [classInclude]=\"'libs-ui-font-h5r'\"\n          [type]=\"'text'\"\n          [ignoreShowPopover]=\"true\"\n          [config]=\"{ width: 250 }\">\n          <!-- // ta.m ẩn show tooltip vì chưa xử lý đc vị trí -->\n          {{ item.name + ' (' + item.username + ')' }}\n        </libs_ui-components-popover>\n      </div>\n    </div>\n  }\n</div>\n"]}
|