@ctrl/ngx-emoji-mart 8.2.0 → 9.0.0
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 +39 -14
- package/anchors.component.d.ts +1 -1
- package/category.component.d.ts +1 -1
- package/esm2022/anchors.component.mjs +100 -0
- package/esm2022/category.component.mjs +409 -0
- package/esm2022/emoji-frequently.service.mjs +90 -0
- package/esm2022/emoji-search.service.mjs +178 -0
- package/esm2022/ngx-emoji/emoji.component.mjs +299 -0
- package/esm2022/ngx-emoji/emoji.module.mjs +17 -0
- package/esm2022/ngx-emoji/emoji.service.mjs +145 -0
- package/esm2022/picker.component.mjs +522 -0
- package/esm2022/picker.module.mjs +40 -0
- package/esm2022/preview.component.mjs +206 -0
- package/esm2022/search.component.mjs +175 -0
- package/esm2022/skins.component.mjs +107 -0
- package/{fesm2020 → fesm2022}/ctrl-ngx-emoji-mart-ngx-emoji.mjs +62 -54
- package/{fesm2015 → fesm2022}/ctrl-ngx-emoji-mart-ngx-emoji.mjs.map +1 -1
- package/{fesm2020 → fesm2022}/ctrl-ngx-emoji-mart.mjs +320 -254
- package/fesm2022/ctrl-ngx-emoji-mart.mjs.map +1 -0
- package/ngx-emoji/emoji.component.d.ts +1 -1
- package/ngx-emoji/emoji.module.d.ts +1 -2
- package/package.json +8 -16
- package/picker.component.d.ts +2 -2
- package/picker.module.d.ts +1 -4
- package/preview.component.d.ts +2 -2
- package/search.component.d.ts +1 -1
- package/skins.component.d.ts +2 -2
- package/esm2020/anchors.component.mjs +0 -83
- package/esm2020/category.component.mjs +0 -386
- package/esm2020/emoji-frequently.service.mjs +0 -88
- package/esm2020/emoji-search.service.mjs +0 -177
- package/esm2020/ngx-emoji/emoji.component.mjs +0 -288
- package/esm2020/ngx-emoji/emoji.module.mjs +0 -18
- package/esm2020/ngx-emoji/emoji.service.mjs +0 -144
- package/esm2020/picker.component.mjs +0 -508
- package/esm2020/picker.module.mjs +0 -49
- package/esm2020/preview.component.mjs +0 -187
- package/esm2020/search.component.mjs +0 -165
- package/esm2020/skins.component.mjs +0 -108
- package/fesm2015/ctrl-ngx-emoji-mart-ngx-emoji.mjs +0 -34330
- package/fesm2015/ctrl-ngx-emoji-mart.mjs +0 -1763
- package/fesm2015/ctrl-ngx-emoji-mart.mjs.map +0 -1
- package/fesm2020/ctrl-ngx-emoji-mart-ngx-emoji.mjs.map +0 -1
- package/fesm2020/ctrl-ngx-emoji-mart.mjs.map +0 -1
- /package/{esm2020 → esm2022}/ctrl-ngx-emoji-mart.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-emoji/ctrl-ngx-emoji-mart-ngx-emoji.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-emoji/data/categories.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-emoji/data/data.interfaces.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-emoji/data/emojis.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-emoji/data/skins.mjs +0 -0
- /package/{esm2020 → esm2022}/ngx-emoji/index.mjs +0 -0
- /package/{esm2020 → esm2022}/public_api.mjs +0 -0
- /package/{esm2020 → esm2022}/svgs/index.mjs +0 -0
- /package/{esm2020 → esm2022}/utils/index.mjs +0 -0
@@ -0,0 +1,409 @@
|
|
1
|
+
import { EmojiComponent } from '@ctrl/ngx-emoji-mart/ngx-emoji';
|
2
|
+
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
|
3
|
+
import { CommonModule } from '@angular/common';
|
4
|
+
import { Subject } from 'rxjs';
|
5
|
+
import * as i0 from "@angular/core";
|
6
|
+
import * as i1 from "@ctrl/ngx-emoji-mart/ngx-emoji";
|
7
|
+
import * as i2 from "./emoji-frequently.service";
|
8
|
+
import * as i3 from "@angular/common";
|
9
|
+
class CategoryComponent {
|
10
|
+
ref;
|
11
|
+
emojiService;
|
12
|
+
frequently;
|
13
|
+
emojis = null;
|
14
|
+
hasStickyPosition = true;
|
15
|
+
name = '';
|
16
|
+
perLine = 9;
|
17
|
+
totalFrequentLines = 4;
|
18
|
+
recent = [];
|
19
|
+
custom = [];
|
20
|
+
i18n;
|
21
|
+
id;
|
22
|
+
hideObsolete = true;
|
23
|
+
notFoundEmoji;
|
24
|
+
virtualize = false;
|
25
|
+
virtualizeOffset = 0;
|
26
|
+
emojiIsNative;
|
27
|
+
emojiSkin;
|
28
|
+
emojiSize;
|
29
|
+
emojiSet;
|
30
|
+
emojiSheetSize;
|
31
|
+
emojiForceSize;
|
32
|
+
emojiTooltip;
|
33
|
+
emojiBackgroundImageFn;
|
34
|
+
emojiImageUrlFn;
|
35
|
+
emojiUseButton;
|
36
|
+
emojiClick = new EventEmitter();
|
37
|
+
/**
|
38
|
+
* Note: the suffix is added explicitly so we know the event is dispatched outside of the Angular zone.
|
39
|
+
*/
|
40
|
+
emojiOverOutsideAngular = new EventEmitter();
|
41
|
+
emojiLeaveOutsideAngular = new EventEmitter();
|
42
|
+
container;
|
43
|
+
label;
|
44
|
+
containerStyles = {};
|
45
|
+
emojisToDisplay = [];
|
46
|
+
filteredEmojisSubject = new Subject();
|
47
|
+
filteredEmojis$ = this.filteredEmojisSubject.asObservable();
|
48
|
+
labelStyles = {};
|
49
|
+
labelSpanStyles = {};
|
50
|
+
margin = 0;
|
51
|
+
minMargin = 0;
|
52
|
+
maxMargin = 0;
|
53
|
+
top = 0;
|
54
|
+
rows = 0;
|
55
|
+
constructor(ref, emojiService, frequently) {
|
56
|
+
this.ref = ref;
|
57
|
+
this.emojiService = emojiService;
|
58
|
+
this.frequently = frequently;
|
59
|
+
}
|
60
|
+
ngOnInit() {
|
61
|
+
this.updateRecentEmojis();
|
62
|
+
this.emojisToDisplay = this.filterEmojis();
|
63
|
+
if (this.noEmojiToDisplay) {
|
64
|
+
this.containerStyles = { display: 'none' };
|
65
|
+
}
|
66
|
+
if (!this.hasStickyPosition) {
|
67
|
+
this.labelStyles = { height: 28 };
|
68
|
+
// this.labelSpanStyles = { position: 'absolute' };
|
69
|
+
}
|
70
|
+
}
|
71
|
+
ngOnChanges(changes) {
|
72
|
+
if (changes.emojis?.currentValue?.length !== changes.emojis?.previousValue?.length) {
|
73
|
+
this.emojisToDisplay = this.filterEmojis();
|
74
|
+
this.ngAfterViewInit();
|
75
|
+
}
|
76
|
+
}
|
77
|
+
ngAfterViewInit() {
|
78
|
+
if (!this.virtualize) {
|
79
|
+
return;
|
80
|
+
}
|
81
|
+
const { width } = this.container.nativeElement.getBoundingClientRect();
|
82
|
+
const perRow = Math.floor(width / (this.emojiSize + 12));
|
83
|
+
this.rows = Math.ceil(this.emojisToDisplay.length / perRow);
|
84
|
+
this.containerStyles = {
|
85
|
+
...this.containerStyles,
|
86
|
+
minHeight: `${this.rows * (this.emojiSize + 12) + 28}px`,
|
87
|
+
};
|
88
|
+
this.ref.detectChanges();
|
89
|
+
this.handleScroll(this.container.nativeElement.parentNode.parentNode.scrollTop);
|
90
|
+
}
|
91
|
+
get noEmojiToDisplay() {
|
92
|
+
return this.emojisToDisplay.length === 0;
|
93
|
+
}
|
94
|
+
memoizeSize() {
|
95
|
+
const parent = this.container.nativeElement.parentNode.parentNode;
|
96
|
+
const { top, height } = this.container.nativeElement.getBoundingClientRect();
|
97
|
+
const parentTop = parent.getBoundingClientRect().top;
|
98
|
+
const labelHeight = this.label.nativeElement.getBoundingClientRect().height;
|
99
|
+
this.top = top - parentTop + parent.scrollTop;
|
100
|
+
if (height === 0) {
|
101
|
+
this.maxMargin = 0;
|
102
|
+
}
|
103
|
+
else {
|
104
|
+
this.maxMargin = height - labelHeight;
|
105
|
+
}
|
106
|
+
}
|
107
|
+
handleScroll(scrollTop) {
|
108
|
+
let margin = scrollTop - this.top;
|
109
|
+
margin = margin < this.minMargin ? this.minMargin : margin;
|
110
|
+
margin = margin > this.maxMargin ? this.maxMargin : margin;
|
111
|
+
if (this.virtualize) {
|
112
|
+
const { top, height } = this.container.nativeElement.getBoundingClientRect();
|
113
|
+
const parentHeight = this.container.nativeElement.parentNode.parentNode.clientHeight;
|
114
|
+
if (parentHeight + (parentHeight + this.virtualizeOffset) >= top &&
|
115
|
+
-height - (parentHeight + this.virtualizeOffset) <= top) {
|
116
|
+
this.filteredEmojisSubject.next(this.emojisToDisplay);
|
117
|
+
}
|
118
|
+
else {
|
119
|
+
this.filteredEmojisSubject.next([]);
|
120
|
+
}
|
121
|
+
}
|
122
|
+
if (margin === this.margin) {
|
123
|
+
this.ref.detectChanges();
|
124
|
+
return false;
|
125
|
+
}
|
126
|
+
if (!this.hasStickyPosition) {
|
127
|
+
this.label.nativeElement.style.top = `${margin}px`;
|
128
|
+
}
|
129
|
+
this.margin = margin;
|
130
|
+
this.ref.detectChanges();
|
131
|
+
return true;
|
132
|
+
}
|
133
|
+
updateRecentEmojis() {
|
134
|
+
if (this.name !== 'Recent') {
|
135
|
+
return;
|
136
|
+
}
|
137
|
+
let frequentlyUsed = this.recent || this.frequently.get(this.perLine, this.totalFrequentLines);
|
138
|
+
if (!frequentlyUsed || !frequentlyUsed.length) {
|
139
|
+
frequentlyUsed = this.frequently.get(this.perLine, this.totalFrequentLines);
|
140
|
+
}
|
141
|
+
if (!frequentlyUsed.length) {
|
142
|
+
return;
|
143
|
+
}
|
144
|
+
this.emojis = frequentlyUsed
|
145
|
+
.map(id => {
|
146
|
+
const emoji = this.custom.filter((e) => e.id === id)[0];
|
147
|
+
if (emoji) {
|
148
|
+
return emoji;
|
149
|
+
}
|
150
|
+
return id;
|
151
|
+
})
|
152
|
+
.filter(id => !!this.emojiService.getData(id));
|
153
|
+
}
|
154
|
+
updateDisplay(display) {
|
155
|
+
this.containerStyles.display = display;
|
156
|
+
this.updateRecentEmojis();
|
157
|
+
this.ref.detectChanges();
|
158
|
+
}
|
159
|
+
trackById(index, item) {
|
160
|
+
return item;
|
161
|
+
}
|
162
|
+
filterEmojis() {
|
163
|
+
const newEmojis = [];
|
164
|
+
for (const emoji of this.emojis || []) {
|
165
|
+
if (!emoji) {
|
166
|
+
continue;
|
167
|
+
}
|
168
|
+
const data = this.emojiService.getData(emoji);
|
169
|
+
if (!data || (data.obsoletedBy && this.hideObsolete) || (!data.unified && !data.custom)) {
|
170
|
+
continue;
|
171
|
+
}
|
172
|
+
newEmojis.push(emoji);
|
173
|
+
}
|
174
|
+
return newEmojis;
|
175
|
+
}
|
176
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CategoryComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.EmojiService }, { token: i2.EmojiFrequentlyService }], target: i0.ɵɵFactoryTarget.Component });
|
177
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.0", type: CategoryComponent, isStandalone: true, selector: "emoji-category", inputs: { emojis: "emojis", hasStickyPosition: "hasStickyPosition", name: "name", perLine: "perLine", totalFrequentLines: "totalFrequentLines", recent: "recent", custom: "custom", i18n: "i18n", id: "id", hideObsolete: "hideObsolete", notFoundEmoji: "notFoundEmoji", virtualize: "virtualize", virtualizeOffset: "virtualizeOffset", emojiIsNative: "emojiIsNative", emojiSkin: "emojiSkin", emojiSize: "emojiSize", emojiSet: "emojiSet", emojiSheetSize: "emojiSheetSize", emojiForceSize: "emojiForceSize", emojiTooltip: "emojiTooltip", emojiBackgroundImageFn: "emojiBackgroundImageFn", emojiImageUrlFn: "emojiImageUrlFn", emojiUseButton: "emojiUseButton" }, outputs: { emojiClick: "emojiClick", emojiOverOutsideAngular: "emojiOverOutsideAngular", emojiLeaveOutsideAngular: "emojiLeaveOutsideAngular" }, viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true, static: true }, { propertyName: "label", first: true, predicate: ["label"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: `
|
178
|
+
<section
|
179
|
+
#container
|
180
|
+
class="emoji-mart-category"
|
181
|
+
[attr.aria-label]="i18n.categories[id]"
|
182
|
+
[class.emoji-mart-no-results]="noEmojiToDisplay"
|
183
|
+
[ngStyle]="containerStyles"
|
184
|
+
>
|
185
|
+
<div class="emoji-mart-category-label" [ngStyle]="labelStyles" [attr.data-name]="name">
|
186
|
+
<!-- already labeled by the section aria-label -->
|
187
|
+
<span #label [ngStyle]="labelSpanStyles" aria-hidden="true">
|
188
|
+
{{ i18n.categories[id] }}
|
189
|
+
</span>
|
190
|
+
</div>
|
191
|
+
|
192
|
+
<div *ngIf="virtualize; else normalRenderTemplate">
|
193
|
+
<div *ngIf="filteredEmojis$ | async as filteredEmojis">
|
194
|
+
<ngx-emoji
|
195
|
+
*ngFor="let emoji of filteredEmojis; trackBy: trackById"
|
196
|
+
[emoji]="emoji"
|
197
|
+
[size]="emojiSize"
|
198
|
+
[skin]="emojiSkin"
|
199
|
+
[isNative]="emojiIsNative"
|
200
|
+
[set]="emojiSet"
|
201
|
+
[sheetSize]="emojiSheetSize"
|
202
|
+
[forceSize]="emojiForceSize"
|
203
|
+
[tooltip]="emojiTooltip"
|
204
|
+
[backgroundImageFn]="emojiBackgroundImageFn"
|
205
|
+
[imageUrlFn]="emojiImageUrlFn"
|
206
|
+
[hideObsolete]="hideObsolete"
|
207
|
+
[useButton]="emojiUseButton"
|
208
|
+
(emojiOverOutsideAngular)="emojiOverOutsideAngular.emit($event)"
|
209
|
+
(emojiLeaveOutsideAngular)="emojiLeaveOutsideAngular.emit($event)"
|
210
|
+
(emojiClick)="emojiClick.emit($event)"
|
211
|
+
></ngx-emoji>
|
212
|
+
</div>
|
213
|
+
</div>
|
214
|
+
|
215
|
+
<div *ngIf="noEmojiToDisplay">
|
216
|
+
<div>
|
217
|
+
<ngx-emoji
|
218
|
+
[emoji]="notFoundEmoji"
|
219
|
+
[size]="38"
|
220
|
+
[skin]="emojiSkin"
|
221
|
+
[isNative]="emojiIsNative"
|
222
|
+
[set]="emojiSet"
|
223
|
+
[sheetSize]="emojiSheetSize"
|
224
|
+
[forceSize]="emojiForceSize"
|
225
|
+
[tooltip]="emojiTooltip"
|
226
|
+
[backgroundImageFn]="emojiBackgroundImageFn"
|
227
|
+
[useButton]="emojiUseButton"
|
228
|
+
></ngx-emoji>
|
229
|
+
</div>
|
230
|
+
|
231
|
+
<div class="emoji-mart-no-results-label">
|
232
|
+
{{ i18n.notfound }}
|
233
|
+
</div>
|
234
|
+
</div>
|
235
|
+
</section>
|
236
|
+
|
237
|
+
<ng-template #normalRenderTemplate>
|
238
|
+
<ngx-emoji
|
239
|
+
*ngFor="let emoji of emojisToDisplay; trackBy: trackById"
|
240
|
+
[emoji]="emoji"
|
241
|
+
[size]="emojiSize"
|
242
|
+
[skin]="emojiSkin"
|
243
|
+
[isNative]="emojiIsNative"
|
244
|
+
[set]="emojiSet"
|
245
|
+
[sheetSize]="emojiSheetSize"
|
246
|
+
[forceSize]="emojiForceSize"
|
247
|
+
[tooltip]="emojiTooltip"
|
248
|
+
[backgroundImageFn]="emojiBackgroundImageFn"
|
249
|
+
[imageUrlFn]="emojiImageUrlFn"
|
250
|
+
[hideObsolete]="hideObsolete"
|
251
|
+
[useButton]="emojiUseButton"
|
252
|
+
(emojiOverOutsideAngular)="emojiOverOutsideAngular.emit($event)"
|
253
|
+
(emojiLeaveOutsideAngular)="emojiLeaveOutsideAngular.emit($event)"
|
254
|
+
(emojiClick)="emojiClick.emit($event)"
|
255
|
+
></ngx-emoji>
|
256
|
+
</ng-template>
|
257
|
+
`, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "component", type: EmojiComponent, selector: "ngx-emoji", inputs: ["skin", "set", "sheetSize", "isNative", "forceSize", "tooltip", "size", "emoji", "fallback", "hideObsolete", "sheetRows", "sheetColumns", "useButton", "backgroundImageFn", "imageUrlFn"], outputs: ["emojiClick", "emojiOver", "emojiOverOutsideAngular", "emojiLeave", "emojiLeaveOutsideAngular"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
258
|
+
}
|
259
|
+
export { CategoryComponent };
|
260
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: CategoryComponent, decorators: [{
|
261
|
+
type: Component,
|
262
|
+
args: [{
|
263
|
+
selector: 'emoji-category',
|
264
|
+
template: `
|
265
|
+
<section
|
266
|
+
#container
|
267
|
+
class="emoji-mart-category"
|
268
|
+
[attr.aria-label]="i18n.categories[id]"
|
269
|
+
[class.emoji-mart-no-results]="noEmojiToDisplay"
|
270
|
+
[ngStyle]="containerStyles"
|
271
|
+
>
|
272
|
+
<div class="emoji-mart-category-label" [ngStyle]="labelStyles" [attr.data-name]="name">
|
273
|
+
<!-- already labeled by the section aria-label -->
|
274
|
+
<span #label [ngStyle]="labelSpanStyles" aria-hidden="true">
|
275
|
+
{{ i18n.categories[id] }}
|
276
|
+
</span>
|
277
|
+
</div>
|
278
|
+
|
279
|
+
<div *ngIf="virtualize; else normalRenderTemplate">
|
280
|
+
<div *ngIf="filteredEmojis$ | async as filteredEmojis">
|
281
|
+
<ngx-emoji
|
282
|
+
*ngFor="let emoji of filteredEmojis; trackBy: trackById"
|
283
|
+
[emoji]="emoji"
|
284
|
+
[size]="emojiSize"
|
285
|
+
[skin]="emojiSkin"
|
286
|
+
[isNative]="emojiIsNative"
|
287
|
+
[set]="emojiSet"
|
288
|
+
[sheetSize]="emojiSheetSize"
|
289
|
+
[forceSize]="emojiForceSize"
|
290
|
+
[tooltip]="emojiTooltip"
|
291
|
+
[backgroundImageFn]="emojiBackgroundImageFn"
|
292
|
+
[imageUrlFn]="emojiImageUrlFn"
|
293
|
+
[hideObsolete]="hideObsolete"
|
294
|
+
[useButton]="emojiUseButton"
|
295
|
+
(emojiOverOutsideAngular)="emojiOverOutsideAngular.emit($event)"
|
296
|
+
(emojiLeaveOutsideAngular)="emojiLeaveOutsideAngular.emit($event)"
|
297
|
+
(emojiClick)="emojiClick.emit($event)"
|
298
|
+
></ngx-emoji>
|
299
|
+
</div>
|
300
|
+
</div>
|
301
|
+
|
302
|
+
<div *ngIf="noEmojiToDisplay">
|
303
|
+
<div>
|
304
|
+
<ngx-emoji
|
305
|
+
[emoji]="notFoundEmoji"
|
306
|
+
[size]="38"
|
307
|
+
[skin]="emojiSkin"
|
308
|
+
[isNative]="emojiIsNative"
|
309
|
+
[set]="emojiSet"
|
310
|
+
[sheetSize]="emojiSheetSize"
|
311
|
+
[forceSize]="emojiForceSize"
|
312
|
+
[tooltip]="emojiTooltip"
|
313
|
+
[backgroundImageFn]="emojiBackgroundImageFn"
|
314
|
+
[useButton]="emojiUseButton"
|
315
|
+
></ngx-emoji>
|
316
|
+
</div>
|
317
|
+
|
318
|
+
<div class="emoji-mart-no-results-label">
|
319
|
+
{{ i18n.notfound }}
|
320
|
+
</div>
|
321
|
+
</div>
|
322
|
+
</section>
|
323
|
+
|
324
|
+
<ng-template #normalRenderTemplate>
|
325
|
+
<ngx-emoji
|
326
|
+
*ngFor="let emoji of emojisToDisplay; trackBy: trackById"
|
327
|
+
[emoji]="emoji"
|
328
|
+
[size]="emojiSize"
|
329
|
+
[skin]="emojiSkin"
|
330
|
+
[isNative]="emojiIsNative"
|
331
|
+
[set]="emojiSet"
|
332
|
+
[sheetSize]="emojiSheetSize"
|
333
|
+
[forceSize]="emojiForceSize"
|
334
|
+
[tooltip]="emojiTooltip"
|
335
|
+
[backgroundImageFn]="emojiBackgroundImageFn"
|
336
|
+
[imageUrlFn]="emojiImageUrlFn"
|
337
|
+
[hideObsolete]="hideObsolete"
|
338
|
+
[useButton]="emojiUseButton"
|
339
|
+
(emojiOverOutsideAngular)="emojiOverOutsideAngular.emit($event)"
|
340
|
+
(emojiLeaveOutsideAngular)="emojiLeaveOutsideAngular.emit($event)"
|
341
|
+
(emojiClick)="emojiClick.emit($event)"
|
342
|
+
></ngx-emoji>
|
343
|
+
</ng-template>
|
344
|
+
`,
|
345
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
346
|
+
preserveWhitespaces: false,
|
347
|
+
standalone: true,
|
348
|
+
imports: [CommonModule, EmojiComponent],
|
349
|
+
}]
|
350
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.EmojiService }, { type: i2.EmojiFrequentlyService }]; }, propDecorators: { emojis: [{
|
351
|
+
type: Input
|
352
|
+
}], hasStickyPosition: [{
|
353
|
+
type: Input
|
354
|
+
}], name: [{
|
355
|
+
type: Input
|
356
|
+
}], perLine: [{
|
357
|
+
type: Input
|
358
|
+
}], totalFrequentLines: [{
|
359
|
+
type: Input
|
360
|
+
}], recent: [{
|
361
|
+
type: Input
|
362
|
+
}], custom: [{
|
363
|
+
type: Input
|
364
|
+
}], i18n: [{
|
365
|
+
type: Input
|
366
|
+
}], id: [{
|
367
|
+
type: Input
|
368
|
+
}], hideObsolete: [{
|
369
|
+
type: Input
|
370
|
+
}], notFoundEmoji: [{
|
371
|
+
type: Input
|
372
|
+
}], virtualize: [{
|
373
|
+
type: Input
|
374
|
+
}], virtualizeOffset: [{
|
375
|
+
type: Input
|
376
|
+
}], emojiIsNative: [{
|
377
|
+
type: Input
|
378
|
+
}], emojiSkin: [{
|
379
|
+
type: Input
|
380
|
+
}], emojiSize: [{
|
381
|
+
type: Input
|
382
|
+
}], emojiSet: [{
|
383
|
+
type: Input
|
384
|
+
}], emojiSheetSize: [{
|
385
|
+
type: Input
|
386
|
+
}], emojiForceSize: [{
|
387
|
+
type: Input
|
388
|
+
}], emojiTooltip: [{
|
389
|
+
type: Input
|
390
|
+
}], emojiBackgroundImageFn: [{
|
391
|
+
type: Input
|
392
|
+
}], emojiImageUrlFn: [{
|
393
|
+
type: Input
|
394
|
+
}], emojiUseButton: [{
|
395
|
+
type: Input
|
396
|
+
}], emojiClick: [{
|
397
|
+
type: Output
|
398
|
+
}], emojiOverOutsideAngular: [{
|
399
|
+
type: Output
|
400
|
+
}], emojiLeaveOutsideAngular: [{
|
401
|
+
type: Output
|
402
|
+
}], container: [{
|
403
|
+
type: ViewChild,
|
404
|
+
args: ['container', { static: true }]
|
405
|
+
}], label: [{
|
406
|
+
type: ViewChild,
|
407
|
+
args: ['label', { static: true }]
|
408
|
+
}] } });
|
409
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"category.component.js","sourceRoot":"","sources":["../../src/lib/picker/category.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Emoji, EmojiComponent, EmojiService } from '@ctrl/ngx-emoji-mart/ngx-emoji';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { Observable, Subject } from 'rxjs';\n\nimport { EmojiFrequentlyService } from './emoji-frequently.service';\n\n@Component({\n  selector: 'emoji-category',\n  template: `\n    <section\n      #container\n      class=\"emoji-mart-category\"\n      [attr.aria-label]=\"i18n.categories[id]\"\n      [class.emoji-mart-no-results]=\"noEmojiToDisplay\"\n      [ngStyle]=\"containerStyles\"\n    >\n      <div class=\"emoji-mart-category-label\" [ngStyle]=\"labelStyles\" [attr.data-name]=\"name\">\n        <!-- already labeled by the section aria-label -->\n        <span #label [ngStyle]=\"labelSpanStyles\" aria-hidden=\"true\">\n          {{ i18n.categories[id] }}\n        </span>\n      </div>\n\n      <div *ngIf=\"virtualize; else normalRenderTemplate\">\n        <div *ngIf=\"filteredEmojis$ | async as filteredEmojis\">\n          <ngx-emoji\n            *ngFor=\"let emoji of filteredEmojis; trackBy: trackById\"\n            [emoji]=\"emoji\"\n            [size]=\"emojiSize\"\n            [skin]=\"emojiSkin\"\n            [isNative]=\"emojiIsNative\"\n            [set]=\"emojiSet\"\n            [sheetSize]=\"emojiSheetSize\"\n            [forceSize]=\"emojiForceSize\"\n            [tooltip]=\"emojiTooltip\"\n            [backgroundImageFn]=\"emojiBackgroundImageFn\"\n            [imageUrlFn]=\"emojiImageUrlFn\"\n            [hideObsolete]=\"hideObsolete\"\n            [useButton]=\"emojiUseButton\"\n            (emojiOverOutsideAngular)=\"emojiOverOutsideAngular.emit($event)\"\n            (emojiLeaveOutsideAngular)=\"emojiLeaveOutsideAngular.emit($event)\"\n            (emojiClick)=\"emojiClick.emit($event)\"\n          ></ngx-emoji>\n        </div>\n      </div>\n\n      <div *ngIf=\"noEmojiToDisplay\">\n        <div>\n          <ngx-emoji\n            [emoji]=\"notFoundEmoji\"\n            [size]=\"38\"\n            [skin]=\"emojiSkin\"\n            [isNative]=\"emojiIsNative\"\n            [set]=\"emojiSet\"\n            [sheetSize]=\"emojiSheetSize\"\n            [forceSize]=\"emojiForceSize\"\n            [tooltip]=\"emojiTooltip\"\n            [backgroundImageFn]=\"emojiBackgroundImageFn\"\n            [useButton]=\"emojiUseButton\"\n          ></ngx-emoji>\n        </div>\n\n        <div class=\"emoji-mart-no-results-label\">\n          {{ i18n.notfound }}\n        </div>\n      </div>\n    </section>\n\n    <ng-template #normalRenderTemplate>\n      <ngx-emoji\n        *ngFor=\"let emoji of emojisToDisplay; trackBy: trackById\"\n        [emoji]=\"emoji\"\n        [size]=\"emojiSize\"\n        [skin]=\"emojiSkin\"\n        [isNative]=\"emojiIsNative\"\n        [set]=\"emojiSet\"\n        [sheetSize]=\"emojiSheetSize\"\n        [forceSize]=\"emojiForceSize\"\n        [tooltip]=\"emojiTooltip\"\n        [backgroundImageFn]=\"emojiBackgroundImageFn\"\n        [imageUrlFn]=\"emojiImageUrlFn\"\n        [hideObsolete]=\"hideObsolete\"\n        [useButton]=\"emojiUseButton\"\n        (emojiOverOutsideAngular)=\"emojiOverOutsideAngular.emit($event)\"\n        (emojiLeaveOutsideAngular)=\"emojiLeaveOutsideAngular.emit($event)\"\n        (emojiClick)=\"emojiClick.emit($event)\"\n      ></ngx-emoji>\n    </ng-template>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  preserveWhitespaces: false,\n  standalone: true,\n  imports: [CommonModule, EmojiComponent],\n})\nexport class CategoryComponent implements OnChanges, OnInit, AfterViewInit {\n  @Input() emojis: any[] | null = null;\n  @Input() hasStickyPosition = true;\n  @Input() name = '';\n  @Input() perLine = 9;\n  @Input() totalFrequentLines = 4;\n  @Input() recent: string[] = [];\n  @Input() custom: any[] = [];\n  @Input() i18n: any;\n  @Input() id: any;\n  @Input() hideObsolete = true;\n  @Input() notFoundEmoji?: string;\n  @Input() virtualize = false;\n  @Input() virtualizeOffset = 0;\n  @Input() emojiIsNative?: Emoji['isNative'];\n  @Input() emojiSkin!: Emoji['skin'];\n  @Input() emojiSize!: Emoji['size'];\n  @Input() emojiSet!: Emoji['set'];\n  @Input() emojiSheetSize!: Emoji['sheetSize'];\n  @Input() emojiForceSize!: Emoji['forceSize'];\n  @Input() emojiTooltip!: Emoji['tooltip'];\n  @Input() emojiBackgroundImageFn?: Emoji['backgroundImageFn'];\n  @Input() emojiImageUrlFn?: Emoji['imageUrlFn'];\n  @Input() emojiUseButton?: boolean;\n  @Output() emojiClick: Emoji['emojiClick'] = new EventEmitter();\n  /**\n   * Note: the suffix is added explicitly so we know the event is dispatched outside of the Angular zone.\n   */\n  @Output() emojiOverOutsideAngular: Emoji['emojiOver'] = new EventEmitter();\n  @Output() emojiLeaveOutsideAngular: Emoji['emojiLeave'] = new EventEmitter();\n  @ViewChild('container', { static: true }) container!: ElementRef;\n  @ViewChild('label', { static: true }) label!: ElementRef;\n  containerStyles: any = {};\n  emojisToDisplay: any[] = [];\n  private filteredEmojisSubject = new Subject<any[] | null | undefined>();\n  filteredEmojis$: Observable<any[] | null | undefined> = this.filteredEmojisSubject.asObservable();\n  labelStyles: any = {};\n  labelSpanStyles: any = {};\n  margin = 0;\n  minMargin = 0;\n  maxMargin = 0;\n  top = 0;\n  rows = 0;\n\n  constructor(\n    public ref: ChangeDetectorRef,\n    private emojiService: EmojiService,\n    private frequently: EmojiFrequentlyService,\n  ) {}\n\n  ngOnInit() {\n    this.updateRecentEmojis();\n    this.emojisToDisplay = this.filterEmojis();\n\n    if (this.noEmojiToDisplay) {\n      this.containerStyles = { display: 'none' };\n    }\n\n    if (!this.hasStickyPosition) {\n      this.labelStyles = { height: 28 };\n      // this.labelSpanStyles = { position: 'absolute' };\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes.emojis?.currentValue?.length !== changes.emojis?.previousValue?.length) {\n      this.emojisToDisplay = this.filterEmojis();\n      this.ngAfterViewInit();\n    }\n  }\n\n  ngAfterViewInit() {\n    if (!this.virtualize) {\n      return;\n    }\n\n    const { width } = this.container.nativeElement.getBoundingClientRect();\n\n    const perRow = Math.floor(width / (this.emojiSize + 12));\n    this.rows = Math.ceil(this.emojisToDisplay.length / perRow);\n\n    this.containerStyles = {\n      ...this.containerStyles,\n      minHeight: `${this.rows * (this.emojiSize + 12) + 28}px`,\n    };\n\n    this.ref.detectChanges();\n\n    this.handleScroll(this.container.nativeElement.parentNode.parentNode.scrollTop);\n  }\n\n  get noEmojiToDisplay(): boolean {\n    return this.emojisToDisplay.length === 0;\n  }\n\n  memoizeSize() {\n    const parent = this.container.nativeElement.parentNode.parentNode;\n    const { top, height } = this.container.nativeElement.getBoundingClientRect();\n    const parentTop = parent.getBoundingClientRect().top;\n    const labelHeight = this.label.nativeElement.getBoundingClientRect().height;\n\n    this.top = top - parentTop + parent.scrollTop;\n\n    if (height === 0) {\n      this.maxMargin = 0;\n    } else {\n      this.maxMargin = height - labelHeight;\n    }\n  }\n\n  handleScroll(scrollTop: number): boolean {\n    let margin = scrollTop - this.top;\n    margin = margin < this.minMargin ? this.minMargin : margin;\n    margin = margin > this.maxMargin ? this.maxMargin : margin;\n\n    if (this.virtualize) {\n      const { top, height } = this.container.nativeElement.getBoundingClientRect();\n      const parentHeight = this.container.nativeElement.parentNode.parentNode.clientHeight;\n\n      if (\n        parentHeight + (parentHeight + this.virtualizeOffset) >= top &&\n        -height - (parentHeight + this.virtualizeOffset) <= top\n      ) {\n        this.filteredEmojisSubject.next(this.emojisToDisplay);\n      } else {\n        this.filteredEmojisSubject.next([]);\n      }\n    }\n\n    if (margin === this.margin) {\n      this.ref.detectChanges();\n      return false;\n    }\n\n    if (!this.hasStickyPosition) {\n      this.label.nativeElement.style.top = `${margin}px`;\n    }\n\n    this.margin = margin;\n    this.ref.detectChanges();\n    return true;\n  }\n\n  updateRecentEmojis() {\n    if (this.name !== 'Recent') {\n      return;\n    }\n\n    let frequentlyUsed = this.recent || this.frequently.get(this.perLine, this.totalFrequentLines);\n    if (!frequentlyUsed || !frequentlyUsed.length) {\n      frequentlyUsed = this.frequently.get(this.perLine, this.totalFrequentLines);\n    }\n    if (!frequentlyUsed.length) {\n      return;\n    }\n    this.emojis = frequentlyUsed\n      .map(id => {\n        const emoji = this.custom.filter((e: any) => e.id === id)[0];\n        if (emoji) {\n          return emoji;\n        }\n\n        return id;\n      })\n      .filter(id => !!this.emojiService.getData(id));\n  }\n\n  updateDisplay(display: 'none' | 'block') {\n    this.containerStyles.display = display;\n    this.updateRecentEmojis();\n    this.ref.detectChanges();\n  }\n\n  trackById(index: number, item: any) {\n    return item;\n  }\n\n  private filterEmojis(): any[] {\n    const newEmojis = [];\n    for (const emoji of this.emojis || []) {\n      if (!emoji) {\n        continue;\n      }\n      const data = this.emojiService.getData(emoji);\n      if (!data || (data.obsoletedBy && this.hideObsolete) || (!data.unified && !data.custom)) {\n        continue;\n      }\n      newEmojis.push(emoji);\n    }\n    return newEmojis;\n  }\n}\n"]}
|
@@ -0,0 +1,90 @@
|
|
1
|
+
import { isPlatformBrowser } from '@angular/common';
|
2
|
+
import { Inject, Injectable, PLATFORM_ID } from '@angular/core';
|
3
|
+
import * as i0 from "@angular/core";
|
4
|
+
class EmojiFrequentlyService {
|
5
|
+
platformId;
|
6
|
+
NAMESPACE = 'emoji-mart';
|
7
|
+
frequently = null;
|
8
|
+
defaults = {};
|
9
|
+
initialized = false;
|
10
|
+
DEFAULTS = [
|
11
|
+
'+1',
|
12
|
+
'grinning',
|
13
|
+
'kissing_heart',
|
14
|
+
'heart_eyes',
|
15
|
+
'laughing',
|
16
|
+
'stuck_out_tongue_winking_eye',
|
17
|
+
'sweat_smile',
|
18
|
+
'joy',
|
19
|
+
'scream',
|
20
|
+
'disappointed',
|
21
|
+
'unamused',
|
22
|
+
'weary',
|
23
|
+
'sob',
|
24
|
+
'sunglasses',
|
25
|
+
'heart',
|
26
|
+
'poop',
|
27
|
+
];
|
28
|
+
constructor(platformId) {
|
29
|
+
this.platformId = platformId;
|
30
|
+
}
|
31
|
+
init() {
|
32
|
+
this.frequently = JSON.parse((isPlatformBrowser(this.platformId) &&
|
33
|
+
localStorage.getItem(`${this.NAMESPACE}.frequently`)) ||
|
34
|
+
'null');
|
35
|
+
this.initialized = true;
|
36
|
+
}
|
37
|
+
add(emoji) {
|
38
|
+
if (!this.initialized) {
|
39
|
+
this.init();
|
40
|
+
}
|
41
|
+
if (!this.frequently) {
|
42
|
+
this.frequently = this.defaults;
|
43
|
+
}
|
44
|
+
if (!this.frequently[emoji.id]) {
|
45
|
+
this.frequently[emoji.id] = 0;
|
46
|
+
}
|
47
|
+
this.frequently[emoji.id] += 1;
|
48
|
+
if (isPlatformBrowser(this.platformId)) {
|
49
|
+
localStorage.setItem(`${this.NAMESPACE}.last`, emoji.id);
|
50
|
+
localStorage.setItem(`${this.NAMESPACE}.frequently`, JSON.stringify(this.frequently));
|
51
|
+
}
|
52
|
+
}
|
53
|
+
get(perLine, totalLines) {
|
54
|
+
if (!this.initialized) {
|
55
|
+
this.init();
|
56
|
+
}
|
57
|
+
if (this.frequently === null) {
|
58
|
+
this.defaults = {};
|
59
|
+
const result = [];
|
60
|
+
for (let i = 0; i < perLine; i++) {
|
61
|
+
this.defaults[this.DEFAULTS[i]] = perLine - i;
|
62
|
+
result.push(this.DEFAULTS[i]);
|
63
|
+
}
|
64
|
+
return result;
|
65
|
+
}
|
66
|
+
const quantity = perLine * totalLines;
|
67
|
+
const frequentlyKeys = Object.keys(this.frequently);
|
68
|
+
const sorted = frequentlyKeys
|
69
|
+
.sort((a, b) => this.frequently[a] - this.frequently[b])
|
70
|
+
.reverse();
|
71
|
+
const sliced = sorted.slice(0, quantity);
|
72
|
+
const last = isPlatformBrowser(this.platformId) && localStorage.getItem(`${this.NAMESPACE}.last`);
|
73
|
+
if (last && !sliced.includes(last)) {
|
74
|
+
sliced.pop();
|
75
|
+
sliced.push(last);
|
76
|
+
}
|
77
|
+
return sliced;
|
78
|
+
}
|
79
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: EmojiFrequentlyService, deps: [{ token: PLATFORM_ID }], target: i0.ɵɵFactoryTarget.Injectable });
|
80
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: EmojiFrequentlyService, providedIn: 'root' });
|
81
|
+
}
|
82
|
+
export { EmojiFrequentlyService };
|
83
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: EmojiFrequentlyService, decorators: [{
|
84
|
+
type: Injectable,
|
85
|
+
args: [{ providedIn: 'root' }]
|
86
|
+
}], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
87
|
+
type: Inject,
|
88
|
+
args: [PLATFORM_ID]
|
89
|
+
}] }]; } });
|
90
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW1vamktZnJlcXVlbnRseS5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2xpYi9waWNrZXIvZW1vamktZnJlcXVlbnRseS5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQ3BELE9BQU8sRUFBRSxNQUFNLEVBQUUsVUFBVSxFQUFFLFdBQVcsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFJaEUsTUFDYSxzQkFBc0I7SUF1QlE7SUF0QnpDLFNBQVMsR0FBRyxZQUFZLENBQUM7SUFDekIsVUFBVSxHQUFxQyxJQUFJLENBQUM7SUFDcEQsUUFBUSxHQUE4QixFQUFFLENBQUM7SUFDekMsV0FBVyxHQUFHLEtBQUssQ0FBQztJQUNwQixRQUFRLEdBQUc7UUFDVCxJQUFJO1FBQ0osVUFBVTtRQUNWLGVBQWU7UUFDZixZQUFZO1FBQ1osVUFBVTtRQUNWLDhCQUE4QjtRQUM5QixhQUFhO1FBQ2IsS0FBSztRQUNMLFFBQVE7UUFDUixjQUFjO1FBQ2QsVUFBVTtRQUNWLE9BQU87UUFDUCxLQUFLO1FBQ0wsWUFBWTtRQUNaLE9BQU87UUFDUCxNQUFNO0tBQ1AsQ0FBQztJQUNGLFlBQXlDLFVBQWtCO1FBQWxCLGVBQVUsR0FBVixVQUFVLENBQVE7SUFBRyxDQUFDO0lBQy9ELElBQUk7UUFDRixJQUFJLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQzFCLENBQUMsaUJBQWlCLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQztZQUNqQyxZQUFZLENBQUMsT0FBTyxDQUFDLEdBQUcsSUFBSSxDQUFDLFNBQVMsYUFBYSxDQUFDLENBQUM7WUFDckQsTUFBTSxDQUNULENBQUM7UUFDRixJQUFJLENBQUMsV0FBVyxHQUFHLElBQUksQ0FBQztJQUMxQixDQUFDO0lBQ0QsR0FBRyxDQUFDLEtBQWdCO1FBQ2xCLElBQUksQ0FBQyxJQUFJLENBQUMs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V0FBVyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGlzUGxhdGZvcm1Ccm93c2VyIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEluamVjdCwgSW5qZWN0YWJsZSwgUExBVEZPUk1fSUQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuaW1wb3J0IHsgRW1vamlEYXRhIH0gZnJvbSAnQGN0cmwvbmd4LWVtb2ppLW1hcnQvbmd4LWVtb2ppJztcblxuQEluamVjdGFibGUoeyBwcm92aWRlZEluOiAncm9vdCcgfSlcbmV4cG9ydCBjbGFzcyBFbW9qaUZyZXF1ZW50bHlTZXJ2aWNlIHtcbiAgTkFNRVNQQUNFID0gJ2Vtb2ppLW1hcnQnO1xuICBmcmVxdWVudGx5OiB7IFtrZXk6IHN0cmluZ106IG51bWJlciB9IHwgbnVsbCA9IG51bGw7XG4gIGRlZmF1bHRzOiB7IFtrZXk6IHN0cmluZ106IG51bWJlciB9ID0ge307XG4gIGluaXRpYWxpemVkID0gZmFsc2U7XG4gIERFRkFVTFRTID0gW1xuICAgICcrMScsXG4gICAgJ2dyaW5uaW5nJyxcbiAgICAna2lzc2luZ19oZWFydCcsXG4gICAgJ2hlYXJ0X2V5ZXMnLFxuICAgICdsYXVnaGluZycsXG4gICAgJ3N0dWNrX291dF90b25ndWVfd2lua2luZ19leWUnLFxuICAgICdzd2VhdF9zbWlsZScsXG4gICAgJ2pveScsXG4gICAgJ3NjcmVhbScsXG4gICAgJ2Rpc2FwcG9pbnRlZCcsXG4gICAgJ3VuYW11c2VkJyxcbiAgICAnd2VhcnknLFxuICAgICdzb2InLFxuICAgICdzdW5nbGFzc2VzJyxcbiAgICAnaGVhcnQnLFxuICAgICdwb29wJyxcbiAgXTtcbiAgY29uc3RydWN0b3IoQEluamVjdChQTEFURk9STV9JRCkgcHJpdmF0ZSBwbGF0Zm9ybUlkOiBzdHJpbmcpIHt9XG4gIGluaXQoKSB7XG4gICAgdGhpcy5mcmVxdWVudGx5ID0gSlNPTi5wYXJzZShcbiAgICAgIChpc1BsYXRmb3JtQnJvd3Nlcih0aGlzLnBsYXRmb3JtSWQpICYmXG4gICAgICAgIGxvY2FsU3RvcmFnZS5nZXRJdGVtKGAke3RoaXMuTkFNRVNQQUNFfS5mcmVxdWVudGx5YCkpIHx8XG4gICAgICAgICdudWxsJyxcbiAgICApO1xuICAgIHRoaXMuaW5pdGlhbGl6ZWQgPSB0cnVlO1xuICB9XG4gIGFkZChlbW9qaTogRW1vamlEYXRhKSB7XG4gICAgaWYgKCF0aGlzLmluaXRpYWxpemVkKSB7XG4gICAgICB0aGlzLmluaXQoKTtcbiAgICB9XG4gICAgaWYgKCF0aGlzLmZyZXF1ZW50bHkpIHtcbiAgICAgIHRoaXMuZnJlcXVlbnRseSA9IHRoaXMuZGVmYXVsdHM7XG4gICAgfVxuICAgIGlmICghdGhpcy5mcmVxdWVudGx5W2Vtb2ppLmlkXSkge1xuICAgICAgdGhpcy5mcmVxdWVudGx5W2Vtb2ppLmlkXSA9IDA7XG4gICAgfVxuICAgIHRoaXMuZnJlcXVlbnRseVtlbW9qaS5pZF0gKz0gMTtcblxuICAgIGlmIChpc1BsYXRmb3JtQnJvd3Nlcih0aGlzLnBsYXRmb3JtSWQpKSB7XG4gICAgICBsb2NhbFN0b3JhZ2Uuc2V0SXRlbShgJHt0aGlzLk5BTUVTUEFDRX0ubGFzdGAsIGVtb2ppLmlkKTtcbiAgICAgIGxvY2FsU3RvcmFnZS5zZXRJdGVtKGAke3RoaXMuTkFNRVNQQUNFfS5mcmVxdWVudGx5YCwgSlNPTi5zdHJpbmdpZnkodGhpcy5mcmVxdWVudGx5KSk7XG4gICAgfVxuICB9XG4gIGdldChwZXJMaW5lOiBudW1iZXIsIHRvdGFsTGluZXM6IG51bWJlcikge1xuICAgIGlmICghdGhpcy5pbml0aWFsaXplZCkge1xuICAgICAgdGhpcy5pbml0KCk7XG4gICAgfVxuICAgIGlmICh0aGlzLmZyZXF1ZW50bHkgPT09IG51bGwpIHtcbiAgICAgIHRoaXMuZGVmYXVsdHMgPSB7fTtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IFtdO1xuXG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IHBlckxpbmU7IGkrKykge1xuICAgICAgICB0aGlzLmRlZmF1bHRzW3RoaXMuREVGQVVMVFNbaV1dID0gcGVyTGluZSAtIGk7XG4gICAgICAgIHJlc3VsdC5wdXNoKHRoaXMuREVGQVVMVFNbaV0pO1xuICAgICAgfVxuICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICB9XG5cbiAgICBjb25zdCBxdWFudGl0eSA9IHBlckxpbmUgKiB0b3RhbExpbmVzO1xuICAgIGNvbnN0IGZyZXF1ZW50bHlLZXlzID0gT2JqZWN0LmtleXModGhpcy5mcmVxdWVudGx5KTtcblxuICAgIGNvbnN0IHNvcnRlZCA9IGZyZXF1ZW50bHlLZXlzXG4gICAgICAuc29ydCgoYSwgYikgPT4gdGhpcy5mcmVxdWVudGx5IVthXSAtIHRoaXMuZnJlcXVlbnRseSFbYl0pXG4gICAgICAucmV2ZXJzZSgpO1xuICAgIGNvbnN0IHNsaWNlZCA9IHNvcnRlZC5zbGljZSgwLCBxdWFudGl0eSk7XG5cbiAgICBjb25zdCBsYXN0ID1cbiAgICAgIGlzUGxhdGZvcm1Ccm93c2VyKHRoaXMucGxhdGZvcm1JZCkgJiYgbG9jYWxTdG9yYWdlLmdldEl0ZW0oYCR7dGhpcy5OQU1FU1BBQ0V9Lmxhc3RgKTtcblxuICAgIGlmIChsYXN0ICYmICFzbGljZWQuaW5jbHVkZXMobGFzdCkpIHtcbiAgICAgIHNsaWNlZC5wb3AoKTtcbiAgICAgIHNsaWNlZC5wdXNoKGxhc3QpO1xuICAgIH1cbiAgICByZXR1cm4gc2xpY2VkO1xuICB9XG59XG4iXX0=
|