@dotglitch/ngx-common 1.0.2
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 +24 -0
- package/components/dynamic-html/dynamic-html.component.d.ts +15 -0
- package/components/dynamic-html/dynamic-html.module.d.ts +10 -0
- package/components/dynamic-html/dynamic-html.service.d.ts +18 -0
- package/components/dynamic-html/types.d.ts +12 -0
- package/components/lazy-loader/lazy-loader.component.d.ts +146 -0
- package/components/lazy-loader/lazy-loader.module.d.ts +10 -0
- package/components/lazy-loader/lazy-loader.service.d.ts +71 -0
- package/components/lazy-loader/types.d.ts +142 -0
- package/components/menu/menu.component.d.ts +52 -0
- package/components/tooltip/tooltip.component.d.ts +35 -0
- package/directives/menu.directive.d.ts +27 -0
- package/directives/tooltip.directive.d.ts +26 -0
- package/directives/utils.d.ts +8 -0
- package/esm2020/components/dynamic-html/dynamic-html.component.mjs +43 -0
- package/esm2020/components/dynamic-html/dynamic-html.module.mjs +27 -0
- package/esm2020/components/dynamic-html/dynamic-html.service.mjs +66 -0
- package/esm2020/components/dynamic-html/types.mjs +7 -0
- package/esm2020/components/lazy-loader/lazy-loader.component.mjs +360 -0
- package/esm2020/components/lazy-loader/lazy-loader.module.mjs +29 -0
- package/esm2020/components/lazy-loader/lazy-loader.service.mjs +215 -0
- package/esm2020/components/lazy-loader/types.mjs +26 -0
- package/esm2020/components/menu/menu.component.mjs +316 -0
- package/esm2020/components/tooltip/tooltip.component.mjs +135 -0
- package/esm2020/directives/menu.directive.mjs +112 -0
- package/esm2020/directives/tooltip.directive.mjs +92 -0
- package/esm2020/directives/utils.mjs +120 -0
- package/esm2020/dotglitch-ngx-common.mjs +5 -0
- package/esm2020/pipes/html-bypass.pipe.mjs +27 -0
- package/esm2020/pipes/resource-bypass.pipe.mjs +27 -0
- package/esm2020/pipes/script-bypass.pipe.mjs +27 -0
- package/esm2020/pipes/style-bypass.pipe.mjs +27 -0
- package/esm2020/pipes/url-bypass.pipe.mjs +27 -0
- package/esm2020/public-api.mjs +39 -0
- package/esm2020/services/dependency.service.mjs +55 -0
- package/esm2020/services/dialog.service.mjs +66 -0
- package/esm2020/services/fetch.service.mjs +71 -0
- package/esm2020/services/keyboard.service.mjs +128 -0
- package/esm2020/types/menu.mjs +2 -0
- package/esm2020/types/popup.mjs +2 -0
- package/esm2020/utils/index.mjs +39 -0
- package/fesm2015/dotglitch-ngx-common.mjs +1997 -0
- package/fesm2015/dotglitch-ngx-common.mjs.map +1 -0
- package/fesm2020/dotglitch-ngx-common.mjs +1977 -0
- package/fesm2020/dotglitch-ngx-common.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/package.json +60 -0
- package/pipes/html-bypass.pipe.d.ts +16 -0
- package/pipes/resource-bypass.pipe.d.ts +16 -0
- package/pipes/script-bypass.pipe.d.ts +16 -0
- package/pipes/style-bypass.pipe.d.ts +16 -0
- package/pipes/url-bypass.pipe.d.ts +16 -0
- package/public-api.d.ts +34 -0
- package/services/dependency.service.d.ts +19 -0
- package/services/dialog.service.d.ts +41 -0
- package/services/fetch.service.d.ts +28 -0
- package/services/keyboard.service.d.ts +79 -0
- package/types/menu.d.ts +90 -0
- package/types/popup.d.ts +27 -0
- package/utils/index.d.ts +19 -0
|
@@ -0,0 +1,316 @@
|
|
|
1
|
+
import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common';
|
|
2
|
+
import { Component, EventEmitter, HostListener, Inject, Input, Optional, Output, TemplateRef } from '@angular/core';
|
|
3
|
+
import { createApplication } from '@angular/platform-browser';
|
|
4
|
+
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
5
|
+
import { MatIconModule } from '@angular/material/icon';
|
|
6
|
+
import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
|
|
7
|
+
import { ComponentPortal, PortalModule } from '@angular/cdk/portal';
|
|
8
|
+
import { firstValueFrom } from 'rxjs';
|
|
9
|
+
import * as i0 from "@angular/core";
|
|
10
|
+
import * as i1 from "@angular/material/dialog";
|
|
11
|
+
import * as i2 from "@angular/cdk/portal";
|
|
12
|
+
import * as i3 from "@angular/platform-browser";
|
|
13
|
+
import * as i4 from "@angular/material/icon";
|
|
14
|
+
import * as i5 from "@angular/material/progress-spinner";
|
|
15
|
+
export const calcMenuItemBounds = async (menuItems, dataObj) => {
|
|
16
|
+
const data = {
|
|
17
|
+
data: dataObj,
|
|
18
|
+
items: menuItems,
|
|
19
|
+
config: {},
|
|
20
|
+
id: null
|
|
21
|
+
};
|
|
22
|
+
return calcComponentBounds(MenuComponent, data);
|
|
23
|
+
};
|
|
24
|
+
const calcComponentBounds = async (component, data) => {
|
|
25
|
+
// Forcibly bootstrap the ctx menu outside of the client application's zone.
|
|
26
|
+
const app = await createApplication({
|
|
27
|
+
providers: [
|
|
28
|
+
{ provide: MAT_DIALOG_DATA, useValue: data }
|
|
29
|
+
]
|
|
30
|
+
});
|
|
31
|
+
const del = document.createElement("div");
|
|
32
|
+
del.style.position = "absolute";
|
|
33
|
+
del.style.left = '-1000vw';
|
|
34
|
+
document.body.append(del);
|
|
35
|
+
const base = app.bootstrap(component, del);
|
|
36
|
+
const { instance } = base;
|
|
37
|
+
await firstValueFrom(app.isStable);
|
|
38
|
+
const el = instance.viewContainer?.element?.nativeElement;
|
|
39
|
+
const rect = el.getBoundingClientRect();
|
|
40
|
+
app.destroy();
|
|
41
|
+
del.remove();
|
|
42
|
+
return rect;
|
|
43
|
+
};
|
|
44
|
+
class TemplateWrapper {
|
|
45
|
+
constructor(dialogRef, _data, viewContainer) {
|
|
46
|
+
this.dialogRef = dialogRef;
|
|
47
|
+
this._data = _data;
|
|
48
|
+
this.viewContainer = viewContainer;
|
|
49
|
+
this.data = _data.data;
|
|
50
|
+
this.template = _data.template;
|
|
51
|
+
// TODO: This is probably invalid
|
|
52
|
+
this.templateType = this.template instanceof TemplateRef ? "template" : "component";
|
|
53
|
+
if (this.templateType == "component") {
|
|
54
|
+
this.componentPortal = new ComponentPortal(this.template);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
TemplateWrapper.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: TemplateWrapper, deps: [{ token: i1.MatDialogRef, optional: true }, { token: MAT_DIALOG_DATA }, { token: i0.ViewContainerRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
59
|
+
TemplateWrapper.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: TemplateWrapper, isStandalone: true, selector: "ngx-menu-template-wrapper", ngImport: i0, template: `
|
|
60
|
+
<ng-container *ngIf="templateType == 'template'; else portalOutlet">
|
|
61
|
+
<ng-container
|
|
62
|
+
[ngTemplateOutlet]="template"
|
|
63
|
+
[ngTemplateOutletContext]="{ '$implicit': data, dialog: dialogRef }"
|
|
64
|
+
/>
|
|
65
|
+
</ng-container>
|
|
66
|
+
<ng-template #portalOutlet [cdkPortalOutlet]="componentPortal" ></ng-template>
|
|
67
|
+
`, isInline: true, dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: PortalModule }, { kind: "directive", type: i2.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }, { kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
68
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: TemplateWrapper, decorators: [{
|
|
69
|
+
type: Component,
|
|
70
|
+
args: [{
|
|
71
|
+
selector: 'ngx-menu-template-wrapper',
|
|
72
|
+
template: `
|
|
73
|
+
<ng-container *ngIf="templateType == 'template'; else portalOutlet">
|
|
74
|
+
<ng-container
|
|
75
|
+
[ngTemplateOutlet]="template"
|
|
76
|
+
[ngTemplateOutletContext]="{ '$implicit': data, dialog: dialogRef }"
|
|
77
|
+
/>
|
|
78
|
+
</ng-container>
|
|
79
|
+
<ng-template #portalOutlet [cdkPortalOutlet]="componentPortal" ></ng-template>
|
|
80
|
+
`,
|
|
81
|
+
imports: [NgTemplateOutlet, PortalModule, NgIf],
|
|
82
|
+
standalone: true
|
|
83
|
+
}]
|
|
84
|
+
}], ctorParameters: function () { return [{ type: i1.MatDialogRef, decorators: [{
|
|
85
|
+
type: Optional
|
|
86
|
+
}] }, { type: undefined, decorators: [{
|
|
87
|
+
type: Inject,
|
|
88
|
+
args: [MAT_DIALOG_DATA]
|
|
89
|
+
}] }, { type: i0.ViewContainerRef }]; } });
|
|
90
|
+
export class MenuComponent {
|
|
91
|
+
constructor(viewContainer, sanitizer, _data, dialog, // optional only for the purpose of estimating dimensions
|
|
92
|
+
dialogRef, changeDetector) {
|
|
93
|
+
this.viewContainer = viewContainer;
|
|
94
|
+
this.sanitizer = sanitizer;
|
|
95
|
+
this._data = _data;
|
|
96
|
+
this.dialog = dialog;
|
|
97
|
+
this.dialogRef = dialogRef;
|
|
98
|
+
this.changeDetector = changeDetector;
|
|
99
|
+
this.closeSignal = new EventEmitter();
|
|
100
|
+
// Check if there are any slashes or dots -- that will clearly exclude it from being a mat icon
|
|
101
|
+
this.matIconRx = /[\/\.]/i;
|
|
102
|
+
this.showIconColumn = true;
|
|
103
|
+
this.showShortcutColumn = true;
|
|
104
|
+
// Defaults are set before @Input() hooks evaluate
|
|
105
|
+
this.data = this._data?.data;
|
|
106
|
+
this.parentCords = this._data?.parentCords;
|
|
107
|
+
this.items = this._data?.items;
|
|
108
|
+
this.config = this._data?.config;
|
|
109
|
+
this.id = this._data?.id;
|
|
110
|
+
}
|
|
111
|
+
ngOnInit() {
|
|
112
|
+
this.items?.forEach(i => {
|
|
113
|
+
if (typeof i == "string")
|
|
114
|
+
return;
|
|
115
|
+
// Set defaults
|
|
116
|
+
i['_disabled'] = false;
|
|
117
|
+
i['_visible'] = true;
|
|
118
|
+
if (i.label)
|
|
119
|
+
try {
|
|
120
|
+
i['_formattedLabel'] = this.formatLabel(i.label);
|
|
121
|
+
}
|
|
122
|
+
catch (e) {
|
|
123
|
+
console.warn(e);
|
|
124
|
+
}
|
|
125
|
+
if (typeof i.isDisabled == "function")
|
|
126
|
+
try {
|
|
127
|
+
i['_disabled'] = i.isDisabled(this.data || {});
|
|
128
|
+
}
|
|
129
|
+
catch (e) {
|
|
130
|
+
console.warn(e);
|
|
131
|
+
}
|
|
132
|
+
if (typeof i.isVisible == "function")
|
|
133
|
+
try {
|
|
134
|
+
i['_visible'] = i.isVisible(this.data || {});
|
|
135
|
+
}
|
|
136
|
+
catch (e) {
|
|
137
|
+
console.warn(e);
|
|
138
|
+
}
|
|
139
|
+
if (typeof i.linkTemplate == "function")
|
|
140
|
+
try {
|
|
141
|
+
i['_link'] = i.linkTemplate(this.data || {});
|
|
142
|
+
}
|
|
143
|
+
catch (e) {
|
|
144
|
+
console.warn(e);
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
// Show the icon column if there are any items with an icon
|
|
148
|
+
this.showIconColumn = !!this.items.find(i => typeof i == "object" &&
|
|
149
|
+
typeof i['icon'] == "string" &&
|
|
150
|
+
i['icon'].length > 2);
|
|
151
|
+
this.showShortcutColumn = !!this.items.find(i => typeof i == "object" &&
|
|
152
|
+
typeof i['shortcut'] == "string" &&
|
|
153
|
+
i['shortcut'].length > 2);
|
|
154
|
+
// setTimeout(() => {
|
|
155
|
+
// this.closeOnLeave = true
|
|
156
|
+
// }, 300);
|
|
157
|
+
}
|
|
158
|
+
ngAfterViewInit() {
|
|
159
|
+
if (this.parentCords) {
|
|
160
|
+
this.selfCords = this.viewContainer?.element?.nativeElement?.getBoundingClientRect();
|
|
161
|
+
this.changeDetector.detectChanges();
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
/**
|
|
165
|
+
*
|
|
166
|
+
* @param item
|
|
167
|
+
* @param evt
|
|
168
|
+
* @returns
|
|
169
|
+
*/
|
|
170
|
+
async onMenuItemClick(item, row, hideBackdrop = false) {
|
|
171
|
+
if (typeof item == 'string')
|
|
172
|
+
return null;
|
|
173
|
+
if (item.separator)
|
|
174
|
+
return null;
|
|
175
|
+
// If cache is enabled, only load if we don't have any children.
|
|
176
|
+
const forceLoad = (item.cacheResolvedChildren ? !item.children : true);
|
|
177
|
+
if (item.childrenResolver && forceLoad) {
|
|
178
|
+
item['_isResolving'] = true;
|
|
179
|
+
item.children = await item.childrenResolver(this.data);
|
|
180
|
+
item['_isResolving'] = false;
|
|
181
|
+
}
|
|
182
|
+
if (!item.childTemplate && !item.children) {
|
|
183
|
+
if (item.action) {
|
|
184
|
+
item.action(this.data);
|
|
185
|
+
this.close();
|
|
186
|
+
}
|
|
187
|
+
// If no action, this is simply a text item.
|
|
188
|
+
return null;
|
|
189
|
+
}
|
|
190
|
+
// Need X pos, Y pos, width and height
|
|
191
|
+
const bounds = row.getBoundingClientRect();
|
|
192
|
+
const cords = {
|
|
193
|
+
top: null,
|
|
194
|
+
left: null,
|
|
195
|
+
bottom: null,
|
|
196
|
+
right: null
|
|
197
|
+
};
|
|
198
|
+
// Set position coordinates
|
|
199
|
+
const { width, height } = await (item.childTemplate
|
|
200
|
+
? calcComponentBounds(TemplateWrapper, { template: item.childTemplate })
|
|
201
|
+
: calcMenuItemBounds(item.children, this.data));
|
|
202
|
+
if (bounds.y + height > window.innerHeight)
|
|
203
|
+
cords.bottom = "0px";
|
|
204
|
+
if (bounds.x + bounds.width + width > window.innerWidth)
|
|
205
|
+
cords.left = ((bounds.x - width)) + "px";
|
|
206
|
+
if (!cords.bottom)
|
|
207
|
+
cords.top = bounds.y + "px";
|
|
208
|
+
if (!cords.left)
|
|
209
|
+
cords.left = bounds.x + bounds.width + "px";
|
|
210
|
+
const component = item.children ? MenuComponent : TemplateWrapper;
|
|
211
|
+
const dialogRef = this.dialog.open(component, {
|
|
212
|
+
position: cords,
|
|
213
|
+
panelClass: ["ngx-ctx-menu", "ngx-app-menu"].concat(this.config?.customClass || []),
|
|
214
|
+
backdropClass: "ngx-ctx-menu-backdrop",
|
|
215
|
+
hasBackdrop: !hideBackdrop,
|
|
216
|
+
data: {
|
|
217
|
+
data: this.data,
|
|
218
|
+
parentCords: this.viewContainer?.element?.nativeElement?.getBoundingClientRect(),
|
|
219
|
+
items: item.children,
|
|
220
|
+
template: item.childTemplate,
|
|
221
|
+
config: this.config
|
|
222
|
+
}
|
|
223
|
+
});
|
|
224
|
+
let _s = dialogRef
|
|
225
|
+
.afterClosed()
|
|
226
|
+
.subscribe((result) => {
|
|
227
|
+
if (result != -1) {
|
|
228
|
+
if (result && typeof item.action == 'function')
|
|
229
|
+
item.action(result);
|
|
230
|
+
this.close();
|
|
231
|
+
}
|
|
232
|
+
else {
|
|
233
|
+
item['_selfclose'] = Date.now();
|
|
234
|
+
}
|
|
235
|
+
_s.unsubscribe();
|
|
236
|
+
});
|
|
237
|
+
return dialogRef;
|
|
238
|
+
}
|
|
239
|
+
/**
|
|
240
|
+
*
|
|
241
|
+
* @param label
|
|
242
|
+
* @returns
|
|
243
|
+
*/
|
|
244
|
+
formatLabel(label) {
|
|
245
|
+
return label.replace(/_([a-z0-9])_/i, (match, group) => `<u>${group}</u>`);
|
|
246
|
+
}
|
|
247
|
+
/**
|
|
248
|
+
* Close the context menu under these circumstances
|
|
249
|
+
*/
|
|
250
|
+
// @HostListener("window:resize", ['event'])
|
|
251
|
+
// @HostListener("window:blur", ['event'])
|
|
252
|
+
close() {
|
|
253
|
+
this.closeSignal.emit();
|
|
254
|
+
this.dialogRef?.close();
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* Check if the dialog is clipping offscreen
|
|
258
|
+
* if so, move it back into view.
|
|
259
|
+
*/
|
|
260
|
+
onResize() {
|
|
261
|
+
const el = this.viewContainer?.element?.nativeElement;
|
|
262
|
+
if (!el)
|
|
263
|
+
return;
|
|
264
|
+
const { width, height, x, y } = el.getBoundingClientRect();
|
|
265
|
+
const target = document.querySelector(".ngx-ctx-menu,.ngx-app-menu");
|
|
266
|
+
if (!target)
|
|
267
|
+
return;
|
|
268
|
+
// Move back into view if we're clipping outside of the bottom
|
|
269
|
+
if (y + height > window.innerHeight) {
|
|
270
|
+
const newTop = (window.innerHeight - (height + (this.config.edgePadding || 12))) + "px";
|
|
271
|
+
target.style['margin-top'] = newTop;
|
|
272
|
+
}
|
|
273
|
+
// Move back into view if we're clipping off the right
|
|
274
|
+
if (x + width > window.innerWidth) {
|
|
275
|
+
const newLeft = (window.innerWidth - (width + (this.config.edgePadding || 12))) + "px";
|
|
276
|
+
target.style['margin-left'] = newLeft;
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
MenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MenuComponent, deps: [{ token: i0.ViewContainerRef }, { token: i3.DomSanitizer }, { token: MAT_DIALOG_DATA, optional: true }, { token: i1.MatDialog, optional: true }, { token: i1.MatDialogRef, optional: true }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
281
|
+
MenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: MenuComponent, isStandalone: true, selector: "ngx-menu", inputs: { data: "data", parentCords: "parentCords", items: "items", config: "config", id: "id" }, outputs: { closeSignal: "closeSignal" }, host: { listeners: { "window:resize": "onResize()" } }, ngImport: i0, template: "<table>\n <tbody>\n <ng-container *ngFor=\"let item of items\">\n <ng-container>\n\n <!-- A row with a click action -->\n <tr #row\n *ngIf=\"item != 'separator' && item.separator != true && item['_visible']\"\n [class.disabled]=\"item['_disabled']\"\n (click)=\"!item['_disabled'] && onMenuItemClick(item, row)\"\n [class.hover]=\"item['children'] && row['hover']\"\n (pointerenter)=\"row['hover'] = true;\"\n (pointerleave)=\"row['hover'] = false\"\n >\n <!-- (item['children']?.length > 0 || item['childTemplate']) && onHover(item, row); closeOnLeave=true -->\n <td class=\"icon\" *ngIf=\"showIconColumn\">\n <img *ngIf=\"matIconRx.test(item.icon); else matIcon\" [src]=\"item.icon\" />\n <ng-template #matIcon>\n <mat-icon [fontIcon]=\"item.icon\"></mat-icon>\n </ng-template>\n </td>\n\n <!-- 'Normal' action based item -->\n <ng-container>\n <td class=\"label\"\n [style.padding-left]=\"showIconColumn ? 0 : '16px'\"\n >\n <a\n [attr.target]=\"item.linkTarget\"\n [attr.href]=\"(item['_link'] || item.link) ? sanitizer.bypassSecurityTrustUrl(item['_link'] || item.link) : undefined\"\n >\n <ng-container\n *ngIf=\"$any(item.labelTemplate)?.prototype; else labelTemplate\"\n [ngTemplateOutlet]=\"$any(item).labelTemplate\"\n [ngTemplateOutletContext]=\"{ '$implicit': data, 'dialog': dialogRef }\"\n />\n\n <ng-template #labelTemplate>\n <ng-container *ngIf=\"!$any(item)?.labelTemplate\">\n <div [innerHTML]=\"item['_formattedLabel']\"></div>\n </ng-container>\n <ng-container *ngIf=\"$any(item)?.labelTemplate\">\n {{$any(item)?.labelTemplate(data || {})}}\n </ng-container>\n </ng-template>\n </a>\n </td>\n </ng-container>\n\n <td class=\"shortcut\" *ngIf=\"showShortcutColumn\">\n {{item.shortcutLabel}}\n </td>\n <td style=\"min-width: 16px\">\n <mat-icon *ngIf=\"\n (item.children && item.children.length > 0) ||\n item.childTemplate ||\n (item.childrenResolver && !item['_isResolving'])\n \"\n sytle=\"transform: translateY(2px)\"\n >\n chevron_right\n </mat-icon>\n\n <mat-progress-spinner *ngIf=\"item['_isResolving']\" mode=\"indeterminate\" [diameter]=\"20\" style=\"margin-right: 4px\"/>\n </td>\n </tr>\n\n <tr *ngIf=\"item != 'separator' && item.separator == true\" class=\"disabled separator\">\n <td class=\"center\" [attr.colspan]=\"2 + (showIconColumn ? 1 : 0) + (showShortcutColumn ? 1 : 0)\">\n <span class=\"hr\">\n {{item['label'] || ''}}\n </span>\n </td>\n </tr>\n <tr *ngIf=\"item == 'separator'\" class=\"disabled separator\">\n <td [attr.colspan]=\"2 + (showIconColumn ? 1 : 0) + (showShortcutColumn ? 1 : 0)\">\n <hr/>\n </td>\n </tr>\n </ng-container>\n </ng-container>\n </tbody>\n</table>\n\n<!-- <div *ngIf=\"true\" class=\"backdrop\"></div> -->\n<!-- <div\n *ngIf=\"parentCords && this.selfCords\"\n class=\"backdrop parent\"\n [style.top]=\"(parentCords.y - selfCords.y + 6) + 'px'\"\n [style.left]=\"(parentCords.x - selfCords.x + 12) + 'px'\"\n [style.width]=\"(parentCords.width) + 'px'\"\n [style.height]=\"(parentCords.height) + 'px'\"\n>\n</div> -->\n<!-- <ng-container *ngIf=\"parentCords && selfCords\">\n <div #top\n class=\"backdrop-outer\"\n [style.bottom]=\"(parentCords.y - selfCords.y)*-1 + parentCords.height + 'px'\"\n style=\"background: #f003;\"\n (pointerenter)=\"onLeave()\"\n ></div>\n <div #right\n class=\"backdrop-outer\"\n [style.left]=\"((parentCords.x - selfCords.x) + parentCords.width) + 'px'\"\n style=\"background: #0f03;\"\n (pointerenter)=\"onLeave()\"\n >\n <div>px: {{parentCords.x}}</div>\n <div>py: {{parentCords.y}}</div>\n <div>pw: {{parentCords.width}}</div>\n <div>ph: {{parentCords.height}}</div>\n <div>sx: {{selfCords.x}}</div>\n <div>sy: {{selfCords.y}}</div>\n <div>sw: {{selfCords.width}}</div>\n <div>sh: {{selfCords.height}}</div>\n </div>\n <div #bottom\n class=\"backdrop-outer\"\n [style.top]=\"((parentCords.y + parentCords.height - selfCords.y)) + 'px'\"\n style=\"background: #00f3;\"\n (pointerenter)=\"onLeave()\"\n ></div>\n <div #left\n class=\"backdrop-outer\"\n [style.right]=\"((parentCords.x - selfCords.x)*-1 + parentCords.width + 32) + 'px'\"\n style=\"background: #fff3;\"\n (pointerenter)=\"onLeave()\"\n ></div>\n</ng-container> -->\n", styles: ["::ng-deep .cdk-overlay-container .ngx-ctx-menu{--mdc-dialog-container-color: var(--ngx-ctx-menu-background-color, #2f2f2f)}::ng-deep .cdk-overlay-container .ngx-ctx-menu .mdc-dialog__container{transform-origin:top left}::ng-deep .cdk-overlay-container .ngx-ctx-menu .mdc-dialog--open .mdc-dialog__container{transform:none}::ng-deep .cdk-overlay-pane.ngx-ctx-menu .mat-mdc-dialog-surface{overflow:visible}:host{-webkit-user-select:none;user-select:none;z-index:1;position:relative;display:block;overflow:hidden auto}table{border-spacing:0;border-radius:5px;padding:4px 0}tr{color:var(--ngx-ctx-menu-text-color, #ccc);font-size:14px;cursor:pointer;transition:background-color 75ms ease,color 75ms ease}tr:not(.disabled):hover{background-color:var(--ngx-ctx-menu-hover-background-color, #94ebeb);color:var(--ngx-ctx-menu-hover-text-color, #000)}tr:not(.disabled):hover a{color:var(--ngx-ctx-menu-hover-text-color, #000)}tr:not(.separator){height:36px}tr.disabled .label{color:var(--ngx-ctx-menu-disabled-text-color, #919191)}tr .center{text-align:center}tr a{outline:0;display:flex;align-items:center;gap:10px;justify-content:space-between;height:100%;width:100%}tr .label{min-width:100px}.hr{height:1px;text-align:center;position:relative}.hr:before,.hr:after{content:\"\";background:var(--ngx-ctx-menu-separator-color, #2a2a2a);display:block;position:absolute;top:0;bottom:0;height:1px;margin:auto;width:300px}.hr:before{right:calc(100% + 4px)}.hr:after{left:calc(100% + 4px)}hr{background:var(--ngx-ctx-menu-separator-color, #2a2a2a);border:0;height:1px;margin:0}.icon{width:24px;height:24px;padding-left:10px}.icon mat-icon{transform:translateY(2px)}.shortcut{color:var(--ngx-ctx-menu-shortcut-text-color, #848484);text-align:end;padding-right:10px;padding-left:12px}.label{height:var(--ngx-ctx-menu-item-height, 30px)}td{vertical-align:middle}\n"], dependencies: [{ kind: "directive", type: NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i4.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatProgressSpinnerModule }, { kind: "component", type: i5.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }] });
|
|
282
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MenuComponent, decorators: [{
|
|
283
|
+
type: Component,
|
|
284
|
+
args: [{ selector: 'ngx-menu', imports: [
|
|
285
|
+
NgIf,
|
|
286
|
+
NgForOf,
|
|
287
|
+
NgTemplateOutlet,
|
|
288
|
+
MatIconModule,
|
|
289
|
+
MatProgressSpinnerModule
|
|
290
|
+
], standalone: true, template: "<table>\n <tbody>\n <ng-container *ngFor=\"let item of items\">\n <ng-container>\n\n <!-- A row with a click action -->\n <tr #row\n *ngIf=\"item != 'separator' && item.separator != true && item['_visible']\"\n [class.disabled]=\"item['_disabled']\"\n (click)=\"!item['_disabled'] && onMenuItemClick(item, row)\"\n [class.hover]=\"item['children'] && row['hover']\"\n (pointerenter)=\"row['hover'] = true;\"\n (pointerleave)=\"row['hover'] = false\"\n >\n <!-- (item['children']?.length > 0 || item['childTemplate']) && onHover(item, row); closeOnLeave=true -->\n <td class=\"icon\" *ngIf=\"showIconColumn\">\n <img *ngIf=\"matIconRx.test(item.icon); else matIcon\" [src]=\"item.icon\" />\n <ng-template #matIcon>\n <mat-icon [fontIcon]=\"item.icon\"></mat-icon>\n </ng-template>\n </td>\n\n <!-- 'Normal' action based item -->\n <ng-container>\n <td class=\"label\"\n [style.padding-left]=\"showIconColumn ? 0 : '16px'\"\n >\n <a\n [attr.target]=\"item.linkTarget\"\n [attr.href]=\"(item['_link'] || item.link) ? sanitizer.bypassSecurityTrustUrl(item['_link'] || item.link) : undefined\"\n >\n <ng-container\n *ngIf=\"$any(item.labelTemplate)?.prototype; else labelTemplate\"\n [ngTemplateOutlet]=\"$any(item).labelTemplate\"\n [ngTemplateOutletContext]=\"{ '$implicit': data, 'dialog': dialogRef }\"\n />\n\n <ng-template #labelTemplate>\n <ng-container *ngIf=\"!$any(item)?.labelTemplate\">\n <div [innerHTML]=\"item['_formattedLabel']\"></div>\n </ng-container>\n <ng-container *ngIf=\"$any(item)?.labelTemplate\">\n {{$any(item)?.labelTemplate(data || {})}}\n </ng-container>\n </ng-template>\n </a>\n </td>\n </ng-container>\n\n <td class=\"shortcut\" *ngIf=\"showShortcutColumn\">\n {{item.shortcutLabel}}\n </td>\n <td style=\"min-width: 16px\">\n <mat-icon *ngIf=\"\n (item.children && item.children.length > 0) ||\n item.childTemplate ||\n (item.childrenResolver && !item['_isResolving'])\n \"\n sytle=\"transform: translateY(2px)\"\n >\n chevron_right\n </mat-icon>\n\n <mat-progress-spinner *ngIf=\"item['_isResolving']\" mode=\"indeterminate\" [diameter]=\"20\" style=\"margin-right: 4px\"/>\n </td>\n </tr>\n\n <tr *ngIf=\"item != 'separator' && item.separator == true\" class=\"disabled separator\">\n <td class=\"center\" [attr.colspan]=\"2 + (showIconColumn ? 1 : 0) + (showShortcutColumn ? 1 : 0)\">\n <span class=\"hr\">\n {{item['label'] || ''}}\n </span>\n </td>\n </tr>\n <tr *ngIf=\"item == 'separator'\" class=\"disabled separator\">\n <td [attr.colspan]=\"2 + (showIconColumn ? 1 : 0) + (showShortcutColumn ? 1 : 0)\">\n <hr/>\n </td>\n </tr>\n </ng-container>\n </ng-container>\n </tbody>\n</table>\n\n<!-- <div *ngIf=\"true\" class=\"backdrop\"></div> -->\n<!-- <div\n *ngIf=\"parentCords && this.selfCords\"\n class=\"backdrop parent\"\n [style.top]=\"(parentCords.y - selfCords.y + 6) + 'px'\"\n [style.left]=\"(parentCords.x - selfCords.x + 12) + 'px'\"\n [style.width]=\"(parentCords.width) + 'px'\"\n [style.height]=\"(parentCords.height) + 'px'\"\n>\n</div> -->\n<!-- <ng-container *ngIf=\"parentCords && selfCords\">\n <div #top\n class=\"backdrop-outer\"\n [style.bottom]=\"(parentCords.y - selfCords.y)*-1 + parentCords.height + 'px'\"\n style=\"background: #f003;\"\n (pointerenter)=\"onLeave()\"\n ></div>\n <div #right\n class=\"backdrop-outer\"\n [style.left]=\"((parentCords.x - selfCords.x) + parentCords.width) + 'px'\"\n style=\"background: #0f03;\"\n (pointerenter)=\"onLeave()\"\n >\n <div>px: {{parentCords.x}}</div>\n <div>py: {{parentCords.y}}</div>\n <div>pw: {{parentCords.width}}</div>\n <div>ph: {{parentCords.height}}</div>\n <div>sx: {{selfCords.x}}</div>\n <div>sy: {{selfCords.y}}</div>\n <div>sw: {{selfCords.width}}</div>\n <div>sh: {{selfCords.height}}</div>\n </div>\n <div #bottom\n class=\"backdrop-outer\"\n [style.top]=\"((parentCords.y + parentCords.height - selfCords.y)) + 'px'\"\n style=\"background: #00f3;\"\n (pointerenter)=\"onLeave()\"\n ></div>\n <div #left\n class=\"backdrop-outer\"\n [style.right]=\"((parentCords.x - selfCords.x)*-1 + parentCords.width + 32) + 'px'\"\n style=\"background: #fff3;\"\n (pointerenter)=\"onLeave()\"\n ></div>\n</ng-container> -->\n", styles: ["::ng-deep .cdk-overlay-container .ngx-ctx-menu{--mdc-dialog-container-color: var(--ngx-ctx-menu-background-color, #2f2f2f)}::ng-deep .cdk-overlay-container .ngx-ctx-menu .mdc-dialog__container{transform-origin:top left}::ng-deep .cdk-overlay-container .ngx-ctx-menu .mdc-dialog--open .mdc-dialog__container{transform:none}::ng-deep .cdk-overlay-pane.ngx-ctx-menu .mat-mdc-dialog-surface{overflow:visible}:host{-webkit-user-select:none;user-select:none;z-index:1;position:relative;display:block;overflow:hidden auto}table{border-spacing:0;border-radius:5px;padding:4px 0}tr{color:var(--ngx-ctx-menu-text-color, #ccc);font-size:14px;cursor:pointer;transition:background-color 75ms ease,color 75ms ease}tr:not(.disabled):hover{background-color:var(--ngx-ctx-menu-hover-background-color, #94ebeb);color:var(--ngx-ctx-menu-hover-text-color, #000)}tr:not(.disabled):hover a{color:var(--ngx-ctx-menu-hover-text-color, #000)}tr:not(.separator){height:36px}tr.disabled .label{color:var(--ngx-ctx-menu-disabled-text-color, #919191)}tr .center{text-align:center}tr a{outline:0;display:flex;align-items:center;gap:10px;justify-content:space-between;height:100%;width:100%}tr .label{min-width:100px}.hr{height:1px;text-align:center;position:relative}.hr:before,.hr:after{content:\"\";background:var(--ngx-ctx-menu-separator-color, #2a2a2a);display:block;position:absolute;top:0;bottom:0;height:1px;margin:auto;width:300px}.hr:before{right:calc(100% + 4px)}.hr:after{left:calc(100% + 4px)}hr{background:var(--ngx-ctx-menu-separator-color, #2a2a2a);border:0;height:1px;margin:0}.icon{width:24px;height:24px;padding-left:10px}.icon mat-icon{transform:translateY(2px)}.shortcut{color:var(--ngx-ctx-menu-shortcut-text-color, #848484);text-align:end;padding-right:10px;padding-left:12px}.label{height:var(--ngx-ctx-menu-item-height, 30px)}td{vertical-align:middle}\n"] }]
|
|
291
|
+
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i3.DomSanitizer }, { type: undefined, decorators: [{
|
|
292
|
+
type: Optional
|
|
293
|
+
}, {
|
|
294
|
+
type: Inject,
|
|
295
|
+
args: [MAT_DIALOG_DATA]
|
|
296
|
+
}] }, { type: i1.MatDialog, decorators: [{
|
|
297
|
+
type: Optional
|
|
298
|
+
}] }, { type: i1.MatDialogRef, decorators: [{
|
|
299
|
+
type: Optional
|
|
300
|
+
}] }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { data: [{
|
|
301
|
+
type: Input
|
|
302
|
+
}], parentCords: [{
|
|
303
|
+
type: Input
|
|
304
|
+
}], items: [{
|
|
305
|
+
type: Input
|
|
306
|
+
}], config: [{
|
|
307
|
+
type: Input
|
|
308
|
+
}], id: [{
|
|
309
|
+
type: Input
|
|
310
|
+
}], closeSignal: [{
|
|
311
|
+
type: Output
|
|
312
|
+
}], onResize: [{
|
|
313
|
+
type: HostListener,
|
|
314
|
+
args: ["window:resize"]
|
|
315
|
+
}] } });
|
|
316
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"menu.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/menu/menu.component.ts","../../../../../packages/common/src/components/menu/menu.component.html"],"names":[],"mappings":"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","sourcesContent":["import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common';\nimport { ChangeDetectorRef, Component, EventEmitter, HostListener, Inject, Input, OnInit, Optional, Output, TemplateRef, Type, ViewContainerRef } from '@angular/core';\nimport { DomSanitizer, createApplication } from '@angular/platform-browser';\nimport { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatProgressSpinnerModule } from '@angular/material/progress-spinner';\n\nimport { ComponentPortal, PortalModule } from '@angular/cdk/portal';\nimport { firstValueFrom } from 'rxjs';\nimport { MenuItem, MenuOptions } from '../../types/menu';\n\nexport const calcMenuItemBounds = async (menuItems: MenuItem[], dataObj: any) => {\n    const data = {\n        data: dataObj,\n        items: menuItems,\n        config: {},\n        id: null\n    }\n\n    return calcComponentBounds(MenuComponent, data);\n}\n\nconst calcComponentBounds = async (component: Type<any>, data: any) => {\n    // Forcibly bootstrap the ctx menu outside of the client application's zone.\n    const app = await createApplication({\n        providers: [\n            { provide: MAT_DIALOG_DATA, useValue: data }\n        ]\n    });\n\n    const del = document.createElement(\"div\");\n    del.style.position = \"absolute\";\n    del.style.left = '-1000vw';\n    document.body.append(del);\n\n    const base = app.bootstrap(component, del);\n    const { instance } = base;\n\n    await firstValueFrom(app.isStable);\n\n    const el: HTMLElement = instance.viewContainer?.element?.nativeElement;\n\n    const rect = el.getBoundingClientRect();\n    app.destroy();\n    del.remove();\n    return rect;\n}\n\n@Component({\n    selector: 'ngx-menu-template-wrapper',\n\n    template: `\n    <ng-container *ngIf=\"templateType == 'template'; else portalOutlet\">\n        <ng-container\n            [ngTemplateOutlet]=\"template\"\n            [ngTemplateOutletContext]=\"{ '$implicit': data, dialog: dialogRef }\"\n        />\n    </ng-container>\n    <ng-template #portalOutlet [cdkPortalOutlet]=\"componentPortal\" ></ng-template>\n`,\n    imports: [ NgTemplateOutlet, PortalModule, NgIf ],\n    standalone: true\n})\nclass TemplateWrapper {\n\n    data: Object;\n    template: TemplateRef<any>;\n\n    templateType: \"template\" | \"component\"\n    componentPortal: ComponentPortal<any>;\n\n    constructor(\n        @Optional() public dialogRef: MatDialogRef<any>,\n        @Inject(MAT_DIALOG_DATA) private _data: any,\n        public viewContainer: ViewContainerRef\n    ) {\n        this.data = _data.data;\n        this.template = _data.template;\n\n        // TODO: This is probably invalid\n        this.templateType = this.template instanceof TemplateRef ? \"template\" : \"component\";\n\n        if (this.templateType == \"component\") {\n            this.componentPortal = new ComponentPortal(this.template as any);\n        }\n    }\n}\n\n@Component({\n    selector: 'ngx-menu',\n    templateUrl: './menu.component.html',\n    styleUrls: ['./menu.component.scss'],\n    imports: [\n        NgIf,\n        NgForOf,\n        NgTemplateOutlet,\n        MatIconModule,\n        MatProgressSpinnerModule\n    ],\n    standalone: true\n})\nexport class MenuComponent implements OnInit {\n    @Input() public data: any;\n    @Input() public parentCords: DOMRect;\n    @Input() public items: MenuItem[];\n    @Input() public config: MenuOptions;\n    @Input() public id: string;\n\n    @Output() closeSignal = new EventEmitter();\n\n    selfCords: DOMRect;\n    // Check if there are any slashes or dots -- that will clearly exclude it from being a mat icon\n    public readonly matIconRx = /[\\/\\.]/i;\n    showIconColumn = true;\n    showShortcutColumn = true;\n\n    constructor(\n        public viewContainer: ViewContainerRef,\n        public sanitizer: DomSanitizer,\n        @Optional() @Inject(MAT_DIALOG_DATA) private _data: any,\n        @Optional() public dialog: MatDialog, // optional only for the purpose of estimating dimensions\n        @Optional() public dialogRef: MatDialogRef<any>,\n        private changeDetector: ChangeDetectorRef\n    ) {\n        // Defaults are set before @Input() hooks evaluate\n        this.data  = this._data?.data;\n        this.parentCords  = this._data?.parentCords;\n        this.items = this._data?.items;\n        this.config = this._data?.config;\n        this.id = this._data?.id;\n    }\n\n    ngOnInit() {\n        this.items?.forEach(i => {\n            if (typeof i == \"string\") return;\n\n            // Set defaults\n            i['_disabled'] = false;\n            i['_visible'] = true;\n\n            if (i.label)\n                try { i['_formattedLabel'] = this.formatLabel(i.label); } catch (e) { console.warn(e) }\n\n            if (typeof i.isDisabled == \"function\")\n                try { i['_disabled'] = i.isDisabled(this.data || {}); } catch(e) { console.warn(e) }\n\n            if (typeof i.isVisible == \"function\")\n                try { i['_visible'] = i.isVisible(this.data || {}); } catch (e) { console.warn(e) }\n\n            if (typeof i.linkTemplate == \"function\")\n                try { i['_link'] = i.linkTemplate(this.data || {}); } catch (e) { console.warn(e) }\n        });\n\n        // Show the icon column if there are any items with an icon\n        this.showIconColumn = !!this.items.find(i =>\n                typeof i == \"object\" &&\n                typeof i['icon'] == \"string\" &&\n                i['icon'].length > 2\n            );\n\n        this.showShortcutColumn = !!this.items.find(i =>\n                typeof i == \"object\" &&\n                typeof i['shortcut'] == \"string\" &&\n                i['shortcut'].length > 2\n            );\n\n        // setTimeout(() => {\n        //     this.closeOnLeave = true\n        // }, 300);\n    }\n\n    ngAfterViewInit() {\n        if (this.parentCords) {\n            this.selfCords = this.viewContainer?.element?.nativeElement?.getBoundingClientRect();\n            this.changeDetector.detectChanges();\n        }\n    }\n\n    /**\n     *\n     * @param item\n     * @param evt\n     * @returns\n     */\n    async onMenuItemClick(item: MenuItem, row: HTMLTableRowElement, hideBackdrop = false) {\n        if (typeof item == 'string') return null;\n        if (item.separator) return null;\n\n        // If cache is enabled, only load if we don't have any children.\n        const forceLoad = (item.cacheResolvedChildren ? !item.children : true);\n\n        if (item.childrenResolver && forceLoad) {\n            item['_isResolving'] = true;\n            item.children = await item.childrenResolver(this.data);\n            item['_isResolving'] = false;\n        }\n\n        if (!item.childTemplate && !item.children) {\n            if (item.action) {\n                item.action(this.data);\n                this.close();\n            }\n            // If no action, this is simply a text item.\n\n            return null;\n        }\n\n        // Need X pos, Y pos, width and height\n        const bounds = row.getBoundingClientRect();\n\n        const cords = {\n            top: null,\n            left: null,\n            bottom: null,\n            right: null\n        };\n\n        // Set position coordinates\n        const { width, height } = await (item.childTemplate\n            ? calcComponentBounds(TemplateWrapper, { template: item.childTemplate })\n            : calcMenuItemBounds(item.children, this.data));\n\n        if (bounds.y + height > window.innerHeight)\n            cords.bottom = \"0px\";\n        if (bounds.x + bounds.width + width > window.innerWidth)\n            cords.left = ((bounds.x - width)) + \"px\";\n\n        if (!cords.bottom) cords.top = bounds.y + \"px\";\n        if (!cords.left) cords.left = bounds.x + bounds.width + \"px\";\n\n        const component = item.children ? MenuComponent : TemplateWrapper as any;\n\n        const dialogRef = this.dialog.open(component, {\n            position: cords,\n            panelClass: [\"ngx-ctx-menu\", \"ngx-app-menu\"].concat(this.config?.customClass || []),\n            backdropClass: \"ngx-ctx-menu-backdrop\",\n            hasBackdrop: !hideBackdrop,\n            data: {\n                data: this.data,\n                parentCords: this.viewContainer?.element?.nativeElement?.getBoundingClientRect(),\n                items: item.children,\n                template: item.childTemplate,\n                config: this.config\n            }\n        });\n\n        let _s = dialogRef\n            .afterClosed()\n                .subscribe((result) => {\n                    if (result != -1) {\n                        if (result && typeof item.action == 'function')\n                            item.action(result);\n\n                        this.close();\n                    }\n                    else {\n                        item['_selfclose'] = Date.now();\n                    }\n\n                    _s.unsubscribe();\n                });\n\n        return dialogRef;\n    }\n\n    /**\n     *\n     * @param label\n     * @returns\n     */\n    formatLabel(label: string): string {\n        return label.replace(/_([a-z0-9])_/i, (match, group) => `<u>${group}</u>`);\n    }\n\n    /**\n     * Close the context menu under these circumstances\n     */\n    // @HostListener(\"window:resize\", ['event'])\n    // @HostListener(\"window:blur\", ['event'])\n    close() {\n        this.closeSignal.emit();\n\n        this.dialogRef?.close();\n    }\n\n    /**\n     * Check if the dialog is clipping offscreen\n     * if so, move it back into view.\n     */\n    @HostListener(\"window:resize\")\n    private onResize() {\n        const el = this.viewContainer?.element?.nativeElement as HTMLElement;\n        if (!el) return;\n\n        const { width, height, x, y } = el.getBoundingClientRect();\n\n        const target = document.querySelector(\".ngx-ctx-menu,.ngx-app-menu\") as HTMLElement;\n        if (!target) return;\n\n        // Move back into view if we're clipping outside of the bottom\n        if (y + height > window.innerHeight) {\n            const newTop = (window.innerHeight - (height + (this.config.edgePadding || 12))) + \"px\";\n            target.style['margin-top'] = newTop;\n        }\n\n        // Move back into view if we're clipping off the right\n        if (x + width > window.innerWidth) {\n            const newLeft = (window.innerWidth - (width + (this.config.edgePadding || 12))) + \"px\"\n            target.style['margin-left'] = newLeft;\n        }\n    }\n\n    // private hoveredItems: MatDialogRef<any>[];\n    // async onHover(item, row: HTMLTableRowElement) {\n    //     this.hoveredItems.forEach(i => i.close(-1));\n\n    //     if (typeof item['_selfclose'] == 'number' && (Date.now() - item['_selfclose']) < 3*1000) return;\n\n    //     const hi = await this.onMenuItemClick(item, row, true);\n    //     this.hoveredItems.push(hi);\n\n    //     hi.afterClosed().toPromise().then(e => {\n    //         this.viewContainer.element.nativeElement.focus();\n    //         this.hoveredItems.splice(this.hoveredItems.findIndex(i => i == hi), 1);\n    //     })\n    // }\n\n    // closeOnLeave = false;\n    // async onLeave() {\n    //     if (this.closeOnLeave) {\n    //         if (this.hoveredItems.length > 0) {\n    //             this.hoveredItems.forEach(i => i.close(-1));\n    //             this.hoveredItems.splice(0);\n    //         }\n    //         else {\n    //             this.closeSignal.next(-1);\n\n    //             this.dialogRef.close(-1);\n    //         }\n    //     }\n    // }\n}\n","<table>\n    <tbody>\n        <ng-container *ngFor=\"let item of items\">\n            <ng-container>\n\n                <!-- A row with a click action -->\n                <tr #row\n                    *ngIf=\"item != 'separator' && item.separator != true && item['_visible']\"\n                    [class.disabled]=\"item['_disabled']\"\n                    (click)=\"!item['_disabled'] && onMenuItemClick(item, row)\"\n                    [class.hover]=\"item['children'] && row['hover']\"\n                    (pointerenter)=\"row['hover'] = true;\"\n                    (pointerleave)=\"row['hover'] = false\"\n                >\n                <!--  (item['children']?.length > 0 || item['childTemplate']) && onHover(item, row); closeOnLeave=true -->\n                    <td class=\"icon\" *ngIf=\"showIconColumn\">\n                        <img *ngIf=\"matIconRx.test(item.icon); else matIcon\" [src]=\"item.icon\" />\n                        <ng-template #matIcon>\n                            <mat-icon [fontIcon]=\"item.icon\"></mat-icon>\n                        </ng-template>\n                    </td>\n\n                    <!-- 'Normal' action based item -->\n                    <ng-container>\n                        <td class=\"label\"\n                            [style.padding-left]=\"showIconColumn ? 0 : '16px'\"\n                        >\n                            <a\n                                [attr.target]=\"item.linkTarget\"\n                                [attr.href]=\"(item['_link'] || item.link) ? sanitizer.bypassSecurityTrustUrl(item['_link'] || item.link) : undefined\"\n                            >\n                                <ng-container\n                                    *ngIf=\"$any(item.labelTemplate)?.prototype; else labelTemplate\"\n                                    [ngTemplateOutlet]=\"$any(item).labelTemplate\"\n                                    [ngTemplateOutletContext]=\"{ '$implicit': data, 'dialog': dialogRef }\"\n                                />\n\n                                <ng-template #labelTemplate>\n                                    <ng-container *ngIf=\"!$any(item)?.labelTemplate\">\n                                        <div [innerHTML]=\"item['_formattedLabel']\"></div>\n                                    </ng-container>\n                                    <ng-container *ngIf=\"$any(item)?.labelTemplate\">\n                                        {{$any(item)?.labelTemplate(data || {})}}\n                                    </ng-container>\n                                </ng-template>\n                            </a>\n                        </td>\n                    </ng-container>\n\n                    <td class=\"shortcut\" *ngIf=\"showShortcutColumn\">\n                        {{item.shortcutLabel}}\n                    </td>\n                    <td style=\"min-width: 16px\">\n                        <mat-icon *ngIf=\"\n                            (item.children && item.children.length > 0) ||\n                            item.childTemplate ||\n                            (item.childrenResolver && !item['_isResolving'])\n                        \"\n                            sytle=\"transform: translateY(2px)\"\n                        >\n                            chevron_right\n                        </mat-icon>\n\n                        <mat-progress-spinner *ngIf=\"item['_isResolving']\" mode=\"indeterminate\" [diameter]=\"20\" style=\"margin-right: 4px\"/>\n                    </td>\n                </tr>\n\n                <tr *ngIf=\"item != 'separator' && item.separator == true\" class=\"disabled separator\">\n                    <td class=\"center\" [attr.colspan]=\"2 + (showIconColumn ? 1 : 0) + (showShortcutColumn ? 1 : 0)\">\n                        <span class=\"hr\">\n                            {{item['label'] || ''}}\n                        </span>\n                    </td>\n                </tr>\n                <tr *ngIf=\"item == 'separator'\" class=\"disabled separator\">\n                    <td [attr.colspan]=\"2 + (showIconColumn ? 1 : 0) + (showShortcutColumn ? 1 : 0)\">\n                        <hr/>\n                    </td>\n                </tr>\n            </ng-container>\n        </ng-container>\n    </tbody>\n</table>\n\n<!-- <div *ngIf=\"true\" class=\"backdrop\"></div> -->\n<!-- <div\n    *ngIf=\"parentCords && this.selfCords\"\n    class=\"backdrop parent\"\n    [style.top]=\"(parentCords.y - selfCords.y + 6) + 'px'\"\n    [style.left]=\"(parentCords.x - selfCords.x + 12) + 'px'\"\n    [style.width]=\"(parentCords.width) + 'px'\"\n    [style.height]=\"(parentCords.height) + 'px'\"\n>\n</div> -->\n<!-- <ng-container *ngIf=\"parentCords && selfCords\">\n    <div #top\n        class=\"backdrop-outer\"\n        [style.bottom]=\"(parentCords.y - selfCords.y)*-1 + parentCords.height + 'px'\"\n        style=\"background: #f003;\"\n        (pointerenter)=\"onLeave()\"\n    ></div>\n    <div #right\n        class=\"backdrop-outer\"\n        [style.left]=\"((parentCords.x - selfCords.x) + parentCords.width) + 'px'\"\n        style=\"background: #0f03;\"\n        (pointerenter)=\"onLeave()\"\n    >\n    <div>px: {{parentCords.x}}</div>\n    <div>py: {{parentCords.y}}</div>\n    <div>pw: {{parentCords.width}}</div>\n    <div>ph: {{parentCords.height}}</div>\n    <div>sx: {{selfCords.x}}</div>\n    <div>sy: {{selfCords.y}}</div>\n    <div>sw: {{selfCords.width}}</div>\n    <div>sh: {{selfCords.height}}</div>\n    </div>\n    <div #bottom\n        class=\"backdrop-outer\"\n        [style.top]=\"((parentCords.y + parentCords.height - selfCords.y)) + 'px'\"\n        style=\"background: #00f3;\"\n        (pointerenter)=\"onLeave()\"\n    ></div>\n    <div #left\n        class=\"backdrop-outer\"\n        [style.right]=\"((parentCords.x - selfCords.x)*-1 + parentCords.width + 32) + 'px'\"\n        style=\"background: #fff3;\"\n        (pointerenter)=\"onLeave()\"\n    ></div>\n</ng-container> -->\n"]}
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, HostListener, Inject, Input, TemplateRef } from '@angular/core';
|
|
3
|
+
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
4
|
+
import { Optional } from '@angular/core';
|
|
5
|
+
import { createApplication } from '@angular/platform-browser';
|
|
6
|
+
import { firstValueFrom } from 'rxjs';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
import * as i1 from "@angular/material/dialog";
|
|
9
|
+
import * as i2 from "@angular/common";
|
|
10
|
+
export const calcTooltipBounds = async (template, data) => {
|
|
11
|
+
const args = {
|
|
12
|
+
data: data || {},
|
|
13
|
+
template,
|
|
14
|
+
config: {},
|
|
15
|
+
selfCords: { left: "0px", top: "0px" },
|
|
16
|
+
ownerCords: { x: 0, y: 0, width: 0, height: 0 },
|
|
17
|
+
id: null
|
|
18
|
+
};
|
|
19
|
+
// Forcibly bootstrap the ctx menu outside of the client application's zone.
|
|
20
|
+
const app = await createApplication({
|
|
21
|
+
providers: [
|
|
22
|
+
{ provide: MAT_DIALOG_DATA, useValue: args }
|
|
23
|
+
]
|
|
24
|
+
});
|
|
25
|
+
const del = document.createElement("div");
|
|
26
|
+
del.style.position = "absolute";
|
|
27
|
+
del.style.left = '-1000vw';
|
|
28
|
+
document.body.append(del);
|
|
29
|
+
const base = app.bootstrap(TooltipComponent, del);
|
|
30
|
+
const { instance } = base;
|
|
31
|
+
await firstValueFrom(app.isStable);
|
|
32
|
+
const el = instance.viewContainer?.element?.nativeElement;
|
|
33
|
+
const rect = el.getBoundingClientRect();
|
|
34
|
+
app.destroy();
|
|
35
|
+
del.remove();
|
|
36
|
+
return rect;
|
|
37
|
+
};
|
|
38
|
+
export class TooltipComponent {
|
|
39
|
+
constructor(viewContainer, _data, dialog, // optional only for the purpose of estimating dimensions
|
|
40
|
+
dialogRef) {
|
|
41
|
+
this.viewContainer = viewContainer;
|
|
42
|
+
this._data = _data;
|
|
43
|
+
this.dialog = dialog;
|
|
44
|
+
this.dialogRef = dialogRef;
|
|
45
|
+
this.hasBootstrapped = false;
|
|
46
|
+
this.pointerIsOnVoid = false;
|
|
47
|
+
this.coverRectCords = {
|
|
48
|
+
top: 0,
|
|
49
|
+
left: 0,
|
|
50
|
+
height: 0,
|
|
51
|
+
width: 0
|
|
52
|
+
};
|
|
53
|
+
// Defaults are set before @Input() hooks evaluate
|
|
54
|
+
this.data = this.data || this._data?.data || {};
|
|
55
|
+
this.config = this.config || this._data?.config;
|
|
56
|
+
this.template = this.template || this._data?.template;
|
|
57
|
+
this.ownerCords = this.ownerCords || this._data?.ownerCords;
|
|
58
|
+
this.selfCords = this.selfCords || this._data?.selfCords;
|
|
59
|
+
}
|
|
60
|
+
ngOnInit() {
|
|
61
|
+
const selfY = parseInt(this.selfCords.top.replace('px', ''));
|
|
62
|
+
const selfX = parseInt(this.selfCords.left.replace('px', ''));
|
|
63
|
+
this.coverRectCords = {
|
|
64
|
+
top: this.ownerCords.y - selfY - 16,
|
|
65
|
+
left: this.ownerCords.x - selfX - 16,
|
|
66
|
+
height: this.ownerCords.height + 32,
|
|
67
|
+
width: this.ownerCords.width + 32
|
|
68
|
+
};
|
|
69
|
+
if (this.template instanceof TemplateRef)
|
|
70
|
+
this.isTemplate = true;
|
|
71
|
+
else if (typeof this.template == "function")
|
|
72
|
+
this.isTemplate = false;
|
|
73
|
+
else
|
|
74
|
+
throw new Error("Unrecognized template object provided.");
|
|
75
|
+
// TODO: resolve the event hook with the .void element
|
|
76
|
+
setTimeout(() => {
|
|
77
|
+
this.hasBootstrapped = true;
|
|
78
|
+
if (this.pointerIsOnVoid)
|
|
79
|
+
this.dialogRef.close();
|
|
80
|
+
}, 10);
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Close the tooltip if these actions occur
|
|
84
|
+
*/
|
|
85
|
+
onClose() {
|
|
86
|
+
this.dialogRef?.close();
|
|
87
|
+
}
|
|
88
|
+
onPointerLeave() {
|
|
89
|
+
this.dialogRef?.close();
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
TooltipComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: TooltipComponent, deps: [{ token: i0.ViewContainerRef }, { token: MAT_DIALOG_DATA, optional: true }, { token: i1.MatDialog, optional: true }, { token: i1.MatDialogRef, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
93
|
+
TooltipComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: TooltipComponent, isStandalone: true, selector: "ngx-tooltip", inputs: { data: "data", config: "config", ownerCords: "ownerCords", selfCords: "selfCords", template: "template" }, host: { listeners: { "window:resize": "onClose()", "window:blur": "onClose()", "pointerleave": "onPointerLeave()" } }, ngImport: i0, template: "<!-- Mouse event blocker for pointer leave -->\n<div\n *ngIf=\"coverRectCords\"\n class=\"owner-mask\"\n [style.top]=\"coverRectCords.top + 'px'\"\n [style.left]=\"coverRectCords.left + 'px'\"\n [style.height]=\"coverRectCords.height + 'px'\"\n [style.width]=\"coverRectCords.width + 'px'\"\n style=\"z-index: -1;\"\n></div>\n\n<div class=\"void\"\n (pointerenter)=\"pointerIsOnVoid = true; hasBootstrapped && dialogRef.close()\"\n (pointerleave)=\"pointerIsOnVoid = false\"\n (pointerdown)=\"hasBootstrapped && dialogRef.close()\"\n></div>\n\n<div class=\"container\">\n <ng-container\n *ngIf=\"isTemplate == false\"\n [ngComponentOutlet]=\"$any(template)\"\n >\n </ng-container>\n\n <ng-container\n *ngIf=\"isTemplate == true\"\n >\n <ng-container\n [ngTemplateOutlet]=\"$any(template)\"\n [ngTemplateOutletContext]=\"{ '$implicit': data }\"\n ></ng-container>\n </ng-container>\n</div>\n", styles: ["::ng-deep .cdk-overlay-container .ngx-tooltip{--mdc-dialog-container-color: var(--ngx-tooltip-background-color, #2f2f2f)}::ng-deep .cdk-overlay-container .ngx-tooltip .mdc-dialog__container{transform-origin:top left}::ng-deep .cdk-overlay-container .ngx-tooltip .mdc-dialog--open .mdc-dialog__container{transform:none}::ng-deep .cdk-overlay-container .ngx-tooltip .mdc-dialog__surface{overflow:visible;background-color:#0000}::ng-deep .cdk-overlay-container .context-menu-backdrop.cdk-overlay-backdrop-showing{opacity:0}::ng-deep .cdk-overlay-pane.ngx-tooltip .mat-dialog-container{padding:0}:host{min-width:2px;min-height:2px;display:block}.void,.owner-mask{position:absolute}.void{top:-100vh;right:-100vw;bottom:-100vh;left:-100vw;z-index:-2}.container{width:100%;height:100%;background:var(--ngx-tooltip-background-color, #333);border-radius:6px;overflow:hidden}\n"], dependencies: [{ kind: "ngmodule", type:
|
|
94
|
+
// NgIf,
|
|
95
|
+
// NgTemplateOutlet,
|
|
96
|
+
// NgComponentOutlet,
|
|
97
|
+
CommonModule }, { kind: "directive", type: i2.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }] });
|
|
98
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: TooltipComponent, decorators: [{
|
|
99
|
+
type: Component,
|
|
100
|
+
args: [{ selector: 'ngx-tooltip', imports: [
|
|
101
|
+
// NgIf,
|
|
102
|
+
// NgTemplateOutlet,
|
|
103
|
+
// NgComponentOutlet,
|
|
104
|
+
CommonModule,
|
|
105
|
+
], standalone: true, template: "<!-- Mouse event blocker for pointer leave -->\n<div\n *ngIf=\"coverRectCords\"\n class=\"owner-mask\"\n [style.top]=\"coverRectCords.top + 'px'\"\n [style.left]=\"coverRectCords.left + 'px'\"\n [style.height]=\"coverRectCords.height + 'px'\"\n [style.width]=\"coverRectCords.width + 'px'\"\n style=\"z-index: -1;\"\n></div>\n\n<div class=\"void\"\n (pointerenter)=\"pointerIsOnVoid = true; hasBootstrapped && dialogRef.close()\"\n (pointerleave)=\"pointerIsOnVoid = false\"\n (pointerdown)=\"hasBootstrapped && dialogRef.close()\"\n></div>\n\n<div class=\"container\">\n <ng-container\n *ngIf=\"isTemplate == false\"\n [ngComponentOutlet]=\"$any(template)\"\n >\n </ng-container>\n\n <ng-container\n *ngIf=\"isTemplate == true\"\n >\n <ng-container\n [ngTemplateOutlet]=\"$any(template)\"\n [ngTemplateOutletContext]=\"{ '$implicit': data }\"\n ></ng-container>\n </ng-container>\n</div>\n", styles: ["::ng-deep .cdk-overlay-container .ngx-tooltip{--mdc-dialog-container-color: var(--ngx-tooltip-background-color, #2f2f2f)}::ng-deep .cdk-overlay-container .ngx-tooltip .mdc-dialog__container{transform-origin:top left}::ng-deep .cdk-overlay-container .ngx-tooltip .mdc-dialog--open .mdc-dialog__container{transform:none}::ng-deep .cdk-overlay-container .ngx-tooltip .mdc-dialog__surface{overflow:visible;background-color:#0000}::ng-deep .cdk-overlay-container .context-menu-backdrop.cdk-overlay-backdrop-showing{opacity:0}::ng-deep .cdk-overlay-pane.ngx-tooltip .mat-dialog-container{padding:0}:host{min-width:2px;min-height:2px;display:block}.void,.owner-mask{position:absolute}.void{top:-100vh;right:-100vw;bottom:-100vh;left:-100vw;z-index:-2}.container{width:100%;height:100%;background:var(--ngx-tooltip-background-color, #333);border-radius:6px;overflow:hidden}\n"] }]
|
|
106
|
+
}], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: undefined, decorators: [{
|
|
107
|
+
type: Optional
|
|
108
|
+
}, {
|
|
109
|
+
type: Inject,
|
|
110
|
+
args: [MAT_DIALOG_DATA]
|
|
111
|
+
}] }, { type: i1.MatDialog, decorators: [{
|
|
112
|
+
type: Optional
|
|
113
|
+
}] }, { type: i1.MatDialogRef, decorators: [{
|
|
114
|
+
type: Optional
|
|
115
|
+
}] }]; }, propDecorators: { data: [{
|
|
116
|
+
type: Input
|
|
117
|
+
}], config: [{
|
|
118
|
+
type: Input
|
|
119
|
+
}], ownerCords: [{
|
|
120
|
+
type: Input
|
|
121
|
+
}], selfCords: [{
|
|
122
|
+
type: Input
|
|
123
|
+
}], template: [{
|
|
124
|
+
type: Input
|
|
125
|
+
}], onClose: [{
|
|
126
|
+
type: HostListener,
|
|
127
|
+
args: ["window:resize"]
|
|
128
|
+
}, {
|
|
129
|
+
type: HostListener,
|
|
130
|
+
args: ["window:blur"]
|
|
131
|
+
}], onPointerLeave: [{
|
|
132
|
+
type: HostListener,
|
|
133
|
+
args: ["pointerleave"]
|
|
134
|
+
}] } });
|
|
135
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/tooltip/tooltip.component.ts","../../../../../packages/common/src/components/tooltip/tooltip.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, HostListener, Inject, Input, TemplateRef, Type, ViewContainerRef } from '@angular/core';\nimport { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { Optional } from '@angular/core';\nimport { createApplication } from '@angular/platform-browser';\nimport { firstValueFrom } from 'rxjs';\nimport { PopupOptions } from '../../types/popup';\n\nexport const calcTooltipBounds = async (template: TemplateRef<any> | Type<any>, data: any) => {\n    const args = {\n        data: data || {},\n        template,\n        config: {},\n        selfCords: { left: \"0px\", top: \"0px\" },\n        ownerCords: { x: 0, y: 0, width: 0, height: 0 },\n        id: null\n    }\n    // Forcibly bootstrap the ctx menu outside of the client application's zone.\n    const app = await createApplication({\n        providers: [\n            { provide: MAT_DIALOG_DATA, useValue: args }\n        ]\n    });\n\n    const del = document.createElement(\"div\");\n    del.style.position = \"absolute\";\n    del.style.left = '-1000vw';\n    document.body.append(del);\n\n    const base = app.bootstrap(TooltipComponent, del);\n    const { instance } = base;\n\n    await firstValueFrom(app.isStable);\n\n    const el: HTMLElement = instance.viewContainer?.element?.nativeElement;\n\n    const rect = el.getBoundingClientRect();\n    app.destroy();\n    del.remove();\n\n    return rect;\n}\n\n\n@Component({\n    selector: 'ngx-tooltip',\n    templateUrl: './tooltip.component.html',\n    styleUrls: ['./tooltip.component.scss'],\n    imports: [\n        // NgIf,\n        // NgTemplateOutlet,\n        // NgComponentOutlet,\n        CommonModule,\n    ],\n    standalone: true\n})\nexport class TooltipComponent {\n    @Input() data: any;\n    @Input() config: PopupOptions;\n    @Input() ownerCords: DOMRect;\n    @Input() selfCords;\n    @Input() template: TemplateRef<any> | Type<any>;\n\n    isTemplate: boolean;\n    hasBootstrapped = false;\n    pointerIsOnVoid = false;\n\n    coverRectCords = {\n        top: 0,\n        left: 0,\n        height: 0,\n        width: 0\n    }\n\n    constructor(\n        public viewContainer: ViewContainerRef,\n        @Optional() @Inject(MAT_DIALOG_DATA) private _data: any,\n        @Optional() public dialog: MatDialog, // optional only for the purpose of estimating dimensions\n        @Optional() public dialogRef: MatDialogRef<any>,\n    ) {\n        // Defaults are set before @Input() hooks evaluate\n        this.data = this.data || this._data?.data || {};\n        this.config = this.config || this._data?.config;\n        this.template = this.template || this._data?.template;\n        this.ownerCords = this.ownerCords || this._data?.ownerCords;\n        this.selfCords = this.selfCords || this._data?.selfCords;\n    }\n\n    ngOnInit() {\n\n        const selfY = parseInt(this.selfCords.top.replace('px', ''));\n        const selfX = parseInt(this.selfCords.left.replace('px', ''));\n\n        this.coverRectCords = {\n            top: this.ownerCords.y - selfY - 16,\n            left: this.ownerCords.x - selfX - 16,\n            height: this.ownerCords.height + 32,\n            width: this.ownerCords.width + 32\n        }\n\n        if (this.template instanceof TemplateRef)\n            this.isTemplate = true;\n        else if (typeof this.template == \"function\")\n            this.isTemplate = false;\n        else\n            throw new Error(\"Unrecognized template object provided.\");\n\n        // TODO: resolve the event hook with the .void element\n        setTimeout(() => {\n            this.hasBootstrapped = true;\n            if (this.pointerIsOnVoid)\n                this.dialogRef.close();\n        }, 10);\n    }\n\n    /**\n     * Close the tooltip if these actions occur\n     */\n    @HostListener(\"window:resize\")\n    @HostListener(\"window:blur\")\n    private onClose() {\n        this.dialogRef?.close();\n    }\n\n    @HostListener(\"pointerleave\")\n    private onPointerLeave() {\n\n        this.dialogRef?.close();\n    }\n}\n","<!-- Mouse event blocker for pointer leave -->\n<div\n    *ngIf=\"coverRectCords\"\n    class=\"owner-mask\"\n    [style.top]=\"coverRectCords.top + 'px'\"\n    [style.left]=\"coverRectCords.left + 'px'\"\n    [style.height]=\"coverRectCords.height + 'px'\"\n    [style.width]=\"coverRectCords.width + 'px'\"\n    style=\"z-index: -1;\"\n></div>\n\n<div class=\"void\"\n    (pointerenter)=\"pointerIsOnVoid = true; hasBootstrapped && dialogRef.close()\"\n    (pointerleave)=\"pointerIsOnVoid = false\"\n    (pointerdown)=\"hasBootstrapped && dialogRef.close()\"\n></div>\n\n<div class=\"container\">\n    <ng-container\n        *ngIf=\"isTemplate == false\"\n        [ngComponentOutlet]=\"$any(template)\"\n    >\n    </ng-container>\n\n    <ng-container\n        *ngIf=\"isTemplate == true\"\n    >\n        <ng-container\n            [ngTemplateOutlet]=\"$any(template)\"\n            [ngTemplateOutletContext]=\"{ '$implicit': data }\"\n        ></ng-container>\n    </ng-container>\n</div>\n"]}
|