@banta/sdk 4.6.26 → 4.7.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.
Files changed (131) hide show
  1. package/{esm2015/banta-sdk.js → esm2020/banta-sdk.mjs} +0 -0
  2. package/{esm2015/lib/attachment-scraper.js → esm2020/lib/attachment-scraper.mjs} +0 -0
  3. package/esm2020/lib/banta/banta.component.mjs +204 -0
  4. package/esm2020/lib/banta-logo.component.mjs +11 -0
  5. package/esm2020/lib/banta-sdk.module.mjs +144 -0
  6. package/esm2020/lib/chat/banta-chat/banta-chat.component.mjs +185 -0
  7. package/esm2020/lib/chat/chat-message/chat-message.component.mjs +62 -0
  8. package/esm2020/lib/chat/chat-view/chat-view.component.mjs +170 -0
  9. package/esm2020/lib/chat/chat.module.mjs +51 -0
  10. package/{esm2015/lib/chat/index.js → esm2020/lib/chat/index.mjs} +0 -0
  11. package/esm2020/lib/chat/live-chat-message.component.mjs +80 -0
  12. package/{esm2015/lib/chat-backend-base.js → esm2020/lib/chat-backend-base.mjs} +0 -0
  13. package/esm2020/lib/chat-backend.mjs +83 -0
  14. package/{esm2015/lib/chat-source-base.js → esm2020/lib/chat-source-base.mjs} +0 -0
  15. package/esm2020/lib/chat-source.mjs +169 -0
  16. package/esm2020/lib/comments/attachment-button/attachment-button.component.mjs +69 -0
  17. package/esm2020/lib/comments/attachment-scraper.directive.mjs +107 -0
  18. package/esm2020/lib/comments/banta-comments/banta-comments.component.mjs +666 -0
  19. package/esm2020/lib/comments/comment/comment.component.mjs +217 -0
  20. package/esm2020/lib/comments/comment-field/comment-field.component.mjs +377 -0
  21. package/esm2020/lib/comments/comment-sort/comment-sort.component.mjs +37 -0
  22. package/esm2020/lib/comments/comment-view/comment-view.component.mjs +379 -0
  23. package/esm2020/lib/comments/comments.module.mjs +111 -0
  24. package/{esm2015/lib/comments/index.js → esm2020/lib/comments/index.mjs} +0 -0
  25. package/esm2020/lib/comments/live-comment.component.mjs +80 -0
  26. package/esm2020/lib/comments/reply-send-options.directive.mjs +13 -0
  27. package/esm2020/lib/common/attachment/attachment.component.mjs +113 -0
  28. package/esm2020/lib/common/attachments/attachments.component.mjs +72 -0
  29. package/esm2020/lib/common/common.module.mjs +59 -0
  30. package/{esm2015/lib/common/index.js → esm2020/lib/common/index.mjs} +0 -0
  31. package/{esm2015/lib/common/lazy-connection.js → esm2020/lib/common/lazy-connection.mjs} +0 -0
  32. package/esm2020/lib/common/lightbox/lightbox.component.mjs +31 -0
  33. package/esm2020/lib/common/markdown-to-html.pipe.mjs +76 -0
  34. package/esm2020/lib/common/mention-linker.pipe.mjs +35 -0
  35. package/esm2020/lib/common/timestamp.component.mjs +113 -0
  36. package/esm2020/lib/common/trust-resource-url.pipe.mjs +22 -0
  37. package/esm2020/lib/emoji/emoji-selector-button.component.mjs +113 -0
  38. package/esm2020/lib/emoji/emoji-selector-panel/emoji-selector-panel.component.mjs +88 -0
  39. package/esm2020/lib/emoji/emoji.module.mjs +55 -0
  40. package/{esm2015/lib/emoji/emojis.js → esm2020/lib/emoji/emojis.mjs} +0 -0
  41. package/{esm2015/lib/emoji/index.js → esm2020/lib/emoji/index.mjs} +0 -0
  42. package/esm2020/lib/giphy-attachments.mjs +16 -0
  43. package/{esm2015/lib/index.js → esm2020/lib/index.mjs} +0 -0
  44. package/esm2020/lib/live-message.component.mjs +96 -0
  45. package/{esm2015/lib/message-menu-item.js → esm2020/lib/message-menu-item.mjs} +0 -0
  46. package/{esm2015/lib/sdk-options.js → esm2020/lib/sdk-options.mjs} +0 -0
  47. package/esm2020/lib/tweet-attachments.mjs +13 -0
  48. package/esm2020/lib/url-attachments.mjs +42 -0
  49. package/esm2020/lib/youtube-attachments.mjs +22 -0
  50. package/{esm2015/public-api.js → esm2020/public-api.mjs} +0 -0
  51. package/fesm2015/{banta-sdk.js → banta-sdk.mjs} +2323 -2008
  52. package/fesm2015/banta-sdk.mjs.map +1 -0
  53. package/fesm2020/banta-sdk.mjs +10490 -0
  54. package/fesm2020/banta-sdk.mjs.map +1 -0
  55. package/{banta-sdk.d.ts → index.d.ts} +1 -0
  56. package/lib/banta/banta.component.d.ts +3 -0
  57. package/lib/banta-logo.component.d.ts +3 -0
  58. package/lib/banta-sdk.module.d.ts +24 -0
  59. package/lib/chat/banta-chat/banta-chat.component.d.ts +3 -0
  60. package/lib/chat/chat-message/chat-message.component.d.ts +3 -0
  61. package/lib/chat/chat-view/chat-view.component.d.ts +3 -0
  62. package/lib/chat/chat.module.d.ts +13 -0
  63. package/lib/chat/live-chat-message.component.d.ts +3 -0
  64. package/lib/chat-backend.d.ts +3 -0
  65. package/lib/chat-source.d.ts +3 -3
  66. package/lib/comments/attachment-button/attachment-button.component.d.ts +3 -0
  67. package/lib/comments/attachment-scraper.directive.d.ts +3 -0
  68. package/lib/comments/banta-comments/banta-comments.component.d.ts +3 -0
  69. package/lib/comments/comment/comment.component.d.ts +3 -0
  70. package/lib/comments/comment-field/comment-field.component.d.ts +3 -0
  71. package/lib/comments/comment-sort/comment-sort.component.d.ts +3 -0
  72. package/lib/comments/comment-view/comment-view.component.d.ts +3 -0
  73. package/lib/comments/comments.module.d.ts +28 -0
  74. package/lib/comments/live-comment.component.d.ts +3 -0
  75. package/lib/comments/reply-send-options.directive.d.ts +3 -0
  76. package/lib/common/attachment/attachment.component.d.ts +3 -0
  77. package/lib/common/attachments/attachments.component.d.ts +3 -0
  78. package/lib/common/common.module.d.ts +15 -0
  79. package/lib/common/lightbox/lightbox.component.d.ts +3 -0
  80. package/lib/common/markdown-to-html.pipe.d.ts +3 -0
  81. package/lib/common/mention-linker.pipe.d.ts +3 -0
  82. package/lib/common/timestamp.component.d.ts +3 -0
  83. package/lib/common/trust-resource-url.pipe.d.ts +3 -0
  84. package/lib/emoji/emoji-selector-button.component.d.ts +3 -0
  85. package/lib/emoji/emoji-selector-panel/emoji-selector-panel.component.d.ts +3 -0
  86. package/lib/emoji/emoji.module.d.ts +14 -0
  87. package/lib/live-message.component.d.ts +3 -0
  88. package/lib/url-attachments.d.ts +3 -0
  89. package/package.json +22 -10
  90. package/banta-sdk.metadata.json +0 -1
  91. package/bundles/banta-sdk.umd.js +0 -11895
  92. package/bundles/banta-sdk.umd.js.map +0 -1
  93. package/bundles/banta-sdk.umd.min.js +0 -2
  94. package/bundles/banta-sdk.umd.min.js.map +0 -1
  95. package/esm2015/lib/banta/banta.component.js +0 -201
  96. package/esm2015/lib/banta-logo.component.js +0 -20
  97. package/esm2015/lib/banta-sdk.module.js +0 -98
  98. package/esm2015/lib/chat/banta-chat/banta-chat.component.js +0 -175
  99. package/esm2015/lib/chat/chat-message/chat-message.component.js +0 -56
  100. package/esm2015/lib/chat/chat-view/chat-view.component.js +0 -167
  101. package/esm2015/lib/chat/chat.module.js +0 -32
  102. package/esm2015/lib/chat/live-chat-message.component.js +0 -71
  103. package/esm2015/lib/chat-backend.js +0 -92
  104. package/esm2015/lib/chat-source.js +0 -205
  105. package/esm2015/lib/comments/attachment-button/attachment-button.component.js +0 -70
  106. package/esm2015/lib/comments/attachment-scraper.directive.js +0 -109
  107. package/esm2015/lib/comments/banta-comments/banta-comments.component.js +0 -678
  108. package/esm2015/lib/comments/comment/comment.component.js +0 -186
  109. package/esm2015/lib/comments/comment-field/comment-field.component.js +0 -350
  110. package/esm2015/lib/comments/comment-sort/comment-sort.component.js +0 -34
  111. package/esm2015/lib/comments/comment-view/comment-view.component.js +0 -364
  112. package/esm2015/lib/comments/comments.module.js +0 -62
  113. package/esm2015/lib/comments/live-comment.component.js +0 -71
  114. package/esm2015/lib/comments/reply-send-options.directive.js +0 -9
  115. package/esm2015/lib/common/attachment/attachment.component.js +0 -105
  116. package/esm2015/lib/common/attachments/attachments.component.js +0 -66
  117. package/esm2015/lib/common/common.module.js +0 -36
  118. package/esm2015/lib/common/lightbox/lightbox.component.js +0 -30
  119. package/esm2015/lib/common/markdown-to-html.pipe.js +0 -74
  120. package/esm2015/lib/common/mention-linker.pipe.js +0 -31
  121. package/esm2015/lib/common/timestamp.component.js +0 -106
  122. package/esm2015/lib/common/trust-resource-url.pipe.js +0 -21
  123. package/esm2015/lib/emoji/emoji-selector-button.component.js +0 -108
  124. package/esm2015/lib/emoji/emoji-selector-panel/emoji-selector-panel.component.js +0 -86
  125. package/esm2015/lib/emoji/emoji.module.js +0 -34
  126. package/esm2015/lib/giphy-attachments.js +0 -20
  127. package/esm2015/lib/live-message.component.js +0 -74
  128. package/esm2015/lib/tweet-attachments.js +0 -16
  129. package/esm2015/lib/url-attachments.js +0 -46
  130. package/esm2015/lib/youtube-attachments.js +0 -25
  131. package/fesm2015/banta-sdk.js.map +0 -1
@@ -0,0 +1,379 @@
1
+ import { Component, Input, ViewChild, Output, HostBinding, ViewChildren } from "@angular/core";
2
+ import { CommentsOrder } from '@banta/common';
3
+ import { Subject, Subscription } from 'rxjs';
4
+ import { CommentComponent } from "../comment/comment.component";
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "../../chat-backend-base";
7
+ import * as i2 from "@angular/common";
8
+ import * as i3 from "@angular/material/icon";
9
+ import * as i4 from "@angular/material/button";
10
+ import * as i5 from "@angular/material/progress-spinner";
11
+ import * as i6 from "../comment/comment.component";
12
+ export class CommentViewComponent {
13
+ constructor(backend) {
14
+ this.backend = backend;
15
+ this._sourceSubs = new Subscription();
16
+ this._selected = new Subject();
17
+ this._liked = new Subject();
18
+ this._unliked = new Subject();
19
+ this._reported = new Subject();
20
+ this._userSelected = new Subject();
21
+ this._usernameSelected = new Subject();
22
+ this._avatarSelected = new Subject();
23
+ this._shared = new Subject();
24
+ this._deleted = new Subject();
25
+ this._messageEdited = new Subject();
26
+ this.showEmptyState = true;
27
+ this.allowReplies = true;
28
+ this.customMenuItems = [];
29
+ this.menuMessage = null;
30
+ this.messages = [];
31
+ this.customSortEnabled = false;
32
+ this.sourceLoaded = new Promise(r => this.markSourceLoaded = r);
33
+ this.maxMessages = 2000;
34
+ this.maxVisibleMessages = 200;
35
+ this.newestLast = false;
36
+ this.isViewingMore = false;
37
+ this.isLoadingMore = false;
38
+ this.hasMore = false;
39
+ this.newMessages = [];
40
+ this.olderMessages = [];
41
+ this.sortOrderChanged = new Subject();
42
+ }
43
+ get comments() {
44
+ return Array.from(this.commentsQuery);
45
+ }
46
+ /**
47
+ * Returns true if this message can be found within one of the message buffers (older, current, newer)
48
+ * @param message
49
+ */
50
+ isMessageLoadedInContext(message) {
51
+ return this.olderMessages.find(x => x.id === message.id)
52
+ || this.messages.find(x => x.id === message.id)
53
+ || this.newMessages.find(x => x.id === message.id);
54
+ }
55
+ async loadMessageInContext(message) {
56
+ await this.sourceLoaded;
57
+ console.log(`Loading message ${message.id} in context...`);
58
+ while (this.hasMore && !this.isMessageLoadedInContext(message)) {
59
+ console.log(`...Need to load more comments to find ${message.id}`);
60
+ await this.showMore();
61
+ }
62
+ if (!this.isMessageLoadedInContext(message)) {
63
+ console.error(`Error while loading message in context: Failed to find message ${message.id}, maybe it was deleted!`);
64
+ return false;
65
+ }
66
+ let pageSize = this.maxVisibleMessages;
67
+ let items = [].concat(this.olderMessages, this.messages, this.newMessages);
68
+ let index = items.findIndex(x => x.id === message.id);
69
+ if (index < 0) {
70
+ console.error(`Error while loading message in context: Message was not present in message list!`);
71
+ return false;
72
+ }
73
+ let startIndex = Math.max(0, index - pageSize / 2);
74
+ this.newMessages = items.splice(0, startIndex);
75
+ this.messages = items.splice(0, pageSize);
76
+ this.olderMessages = items;
77
+ this.isViewingMore = true;
78
+ }
79
+ /**
80
+ * Get the CommentComponent instantiated for the given ChatMessage,
81
+ * if it exists in the current view. Note that messages which are not
82
+ * currently shown to the user will not return a CommentComponent.
83
+ * @param message
84
+ * @returns
85
+ */
86
+ getCommentComponentForMessage(message) {
87
+ if (!message)
88
+ throw new Error(`You must pass a valid ChatMessage`);
89
+ return this.comments.find(x => x.message?.id === message.id);
90
+ }
91
+ get selected() {
92
+ return this._selected;
93
+ }
94
+ get messageEdited() {
95
+ return this._messageEdited.asObservable();
96
+ }
97
+ saveEdit(message, newMessage) {
98
+ this._messageEdited.next({ message, newMessage });
99
+ }
100
+ get userSelected() { return this._userSelected; }
101
+ get reported() { return this._reported; }
102
+ get liked() { return this._liked; }
103
+ get unliked() { return this._unliked; }
104
+ get usernameSelected() { return this._usernameSelected; }
105
+ get avatarSelected() { return this._avatarSelected; }
106
+ get shared() { return this._shared; }
107
+ get deleted() { return this._deleted; }
108
+ get source() {
109
+ return this._source;
110
+ }
111
+ likeMessage(message) {
112
+ this._liked.next(message);
113
+ }
114
+ unlikeMessage(message) {
115
+ this._unliked.next(message);
116
+ }
117
+ reportMessage(message) {
118
+ this._reported.next(message);
119
+ }
120
+ selectMessage(message) {
121
+ this._selected.next(message);
122
+ }
123
+ selectMessageUser(message) {
124
+ this._userSelected.next(message);
125
+ }
126
+ selectUsername(user) {
127
+ this._usernameSelected.next(user);
128
+ }
129
+ selectAvatar(user) {
130
+ this._avatarSelected.next(user);
131
+ }
132
+ sharedMessage(message) {
133
+ this._shared.next(message);
134
+ }
135
+ startEditing(message) {
136
+ this.messages.forEach(m => m.transientState.editing = false);
137
+ message.transientState.editing = true;
138
+ }
139
+ deleteMessage(message) {
140
+ this._deleted.next(message);
141
+ }
142
+ set source(value) {
143
+ this.customSortEnabled = value?.sortOrder !== CommentsOrder.NEWEST;
144
+ this.newMessages = [];
145
+ this.olderMessages = [];
146
+ window.bantaSourceDebug = value;
147
+ if (this._sourceSubs) {
148
+ this._sourceSubs.unsubscribe();
149
+ this._sourceSubs = null;
150
+ }
151
+ this._source = value;
152
+ if (value) {
153
+ const messages = (value.messages || []).slice();
154
+ this.messages = messages;
155
+ this.olderMessages = messages.splice(this.maxVisibleMessages, messages.length);
156
+ this.hasMore = true; //this.olderMessages.length > 0;
157
+ this._sourceSubs = new Subscription();
158
+ this._sourceSubs.add(this._source.messageReceived.subscribe(msg => this.messageReceived(msg)));
159
+ this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));
160
+ this._sourceSubs.add(this.backend.userChanged.subscribe(user => this.currentUser = user));
161
+ this.getInitialMessages();
162
+ }
163
+ }
164
+ async getInitialMessages() {
165
+ let messages = (await this._source.getExistingMessages());
166
+ messages.forEach(m => m.transientState ?? (m.transientState = {}));
167
+ this.messages = this.newestLast ? messages.slice().reverse() : messages;
168
+ this.sortMessages();
169
+ if (this.markSourceLoaded)
170
+ this.markSourceLoaded();
171
+ }
172
+ messageIdentity(index, chatMessage) {
173
+ return chatMessage.id;
174
+ }
175
+ async showNew() {
176
+ let naturalOrder = CommentsOrder.NEWEST;
177
+ if (this.source && this.source.sortOrder !== naturalOrder) {
178
+ this.sortOrderChanged.next(naturalOrder);
179
+ return;
180
+ }
181
+ this.isViewingMore = false;
182
+ if (this.newestLast)
183
+ this.messages = this.messages.concat(this.newMessages.splice(0, this.newMessages.length));
184
+ else
185
+ this.messages = this.newMessages.splice(0, this.newMessages.length).concat(this.messages);
186
+ let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);
187
+ this.olderMessages = overflow.concat(this.olderMessages);
188
+ this.olderMessages.splice(this.maxMessages - this.maxVisibleMessages, this.olderMessages.length);
189
+ this.hasMore = this.olderMessages.length > 0;
190
+ }
191
+ async showMore() {
192
+ this.isViewingMore = true;
193
+ if (this.olderMessages.length > 0) {
194
+ this.isLoadingMore = false;
195
+ this.messages = this.messages.concat(this.olderMessages.splice(0, 50));
196
+ }
197
+ this.isLoadingMore = true;
198
+ let nextPageSize = 20;
199
+ let lastMessage;
200
+ if (this.newestLast) {
201
+ lastMessage = this.olderMessages[0] ?? this.messages[0];
202
+ }
203
+ else {
204
+ lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];
205
+ }
206
+ if (!lastMessage) {
207
+ this.isLoadingMore = false;
208
+ this.hasMore = false;
209
+ return;
210
+ }
211
+ let messages = await this.source.loadAfter(lastMessage, nextPageSize);
212
+ if (this.newestLast)
213
+ messages = messages.slice().reverse();
214
+ messages.forEach(m => m.transientState ?? (m.transientState = {}));
215
+ if (this.newestLast)
216
+ this.messages = messages.concat(this.messages);
217
+ else
218
+ this.messages = this.messages.concat(messages);
219
+ this.isLoadingMore = false;
220
+ if (messages.length === 0) {
221
+ console.log(`Reached the end of the list.`);
222
+ this.hasMore = false;
223
+ }
224
+ }
225
+ addMessage(message) {
226
+ if (!message.transientState)
227
+ message.transientState ?? (message.transientState = {});
228
+ let destination = this.messages;
229
+ let bucket = this.olderMessages;
230
+ let newestLast = this.newestLast;
231
+ if (this.isViewingMore) {
232
+ destination = this.newMessages;
233
+ bucket = null;
234
+ }
235
+ if (newestLast) {
236
+ destination.push(message);
237
+ let overflow = destination.splice(this.maxVisibleMessages, destination.length);
238
+ bucket?.push(...overflow);
239
+ }
240
+ else {
241
+ destination.unshift(message);
242
+ let overflow = destination.splice(this.maxVisibleMessages, destination.length);
243
+ bucket?.unshift(...overflow);
244
+ }
245
+ if (bucket?.length > 0)
246
+ this.hasMore = true;
247
+ message.pagingCursor = String(this.incrementPagingCursors());
248
+ this.sortMessages();
249
+ }
250
+ incrementPagingCursors() {
251
+ if (this.source.sortOrder !== CommentsOrder.NEWEST)
252
+ return;
253
+ let maxPagingCursor = 0;
254
+ for (let group of [this.messages, this.olderMessages, this.newMessages]) {
255
+ for (let message of group) {
256
+ if (message.pagingCursor) {
257
+ let pagingCursor = Number(message.pagingCursor) + 1;
258
+ if (pagingCursor > maxPagingCursor)
259
+ maxPagingCursor = pagingCursor;
260
+ message.pagingCursor = String(pagingCursor);
261
+ }
262
+ }
263
+ }
264
+ return maxPagingCursor;
265
+ }
266
+ /**
267
+ * Wait for all currently visible comments to be fully loaded, including all attachments.
268
+ * Doing this will prevent layout shift when scrolling to a specific comment.
269
+ */
270
+ async waitForAllCommentsToLoad() {
271
+ await new Promise(r => setTimeout(r, 100));
272
+ await this.sourceLoaded;
273
+ await Promise.all(this.comments.map(x => x.waitForLoad()));
274
+ }
275
+ sortMessages() {
276
+ if (!this.source)
277
+ return;
278
+ let sorter;
279
+ if (this.source.sortOrder === CommentsOrder.LIKES)
280
+ sorter = (a, b) => b.likes - a.likes;
281
+ else if (this.source.sortOrder === CommentsOrder.NEWEST)
282
+ sorter = (a, b) => (b.sentAt - a.sentAt) * (this.newestLast ? -1 : 1);
283
+ else if (this.source.sortOrder === CommentsOrder.OLDEST)
284
+ sorter = (a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? -1 : 1);
285
+ this.messages.sort(sorter);
286
+ this.olderMessages.sort(sorter);
287
+ this.newMessages.sort(sorter);
288
+ }
289
+ messageReceived(message) {
290
+ this.addMessage(message);
291
+ if (this.isScrolledToLatest())
292
+ setTimeout(() => this.scrollToLatest());
293
+ }
294
+ isScrolledToLatest() {
295
+ if (!this.messageContainer)
296
+ return false;
297
+ const el = this.messageContainer.nativeElement;
298
+ const currentScroll = el.scrollTop;
299
+ const currentTotal = el.scrollHeight - el.offsetHeight;
300
+ return currentScroll > currentTotal - 10;
301
+ }
302
+ messageSent(message) {
303
+ this.addMessage(message);
304
+ if (!this.messageContainer)
305
+ return;
306
+ this.scrollToLatest();
307
+ }
308
+ scrollToLatest() {
309
+ if (!this.messageContainer)
310
+ return;
311
+ const el = this.messageContainer.nativeElement;
312
+ el.scrollTop = el.scrollHeight;
313
+ }
314
+ mentionsMe(message) {
315
+ if (!this.currentUser)
316
+ return false;
317
+ if (message.message.includes(`@${this.currentUser.username}`))
318
+ return true;
319
+ return false;
320
+ }
321
+ }
322
+ CommentViewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CommentViewComponent, deps: [{ token: i1.ChatBackendBase }], target: i0.ɵɵFactoryTarget.Component });
323
+ CommentViewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: CommentViewComponent, selector: "banta-comment-view", inputs: { showEmptyState: "showEmptyState", allowReplies: "allowReplies", customMenuItems: "customMenuItems", fixedHeight: "fixedHeight", selectedMessage: "selectedMessage", source: "source", genericAvatarUrl: "genericAvatarUrl", maxMessages: "maxMessages", maxVisibleMessages: "maxVisibleMessages", newestLast: "newestLast" }, outputs: { selected: "selected", messageEdited: "messageEdited", userSelected: "userSelected", reported: "reported", liked: "liked", unliked: "unliked", usernameSelected: "usernameSelected", avatarSelected: "avatarSelected", shared: "shared", deleted: "deleted", sortOrderChanged: "sortOrderChanged" }, host: { properties: { "class.fixed-height": "this.fixedHeight" } }, viewQueries: [{ propertyName: "messageContainer", first: true, predicate: ["messageContainer"], descendants: true }, { propertyName: "commentsQuery", predicate: CommentComponent, descendants: true }], ngImport: i0, template: "<div class=\"message-container\">\r\n <ng-content select=\"[data-before]\"></ng-content>\r\n\r\n <a mat-button class=\"nav\" [class.visible]=\"!newestLast && (isViewingMore || customSortEnabled)\" href=\"javascript:;\" (click)=\"showNew()\">\r\n <mat-icon>file_upload</mat-icon>\r\n New\r\n <ng-container *ngIf=\"newMessages.length >= 1\">\r\n ({{newMessages.length}})\r\n </ng-container>\r\n </a>\r\n <a mat-button class=\"nav\" [class.visible]=\"newestLast && hasMore && !isLoadingMore\" href=\"javascript:;\" (click)=\"showMore()\">Show earlier</a>\r\n\r\n <ng-container *ngIf=\"messages.length === 0\">\r\n <div class=\"empty-state\" *ngIf=\"showEmptyState\">\r\n Be the first to comment!\r\n </div>\r\n </ng-container>\r\n <ng-container *ngFor=\"let message of messages; trackBy: messageIdentity\">\r\n <banta-comment\r\n *ngIf=\"!message.hidden\"\r\n class=\"abbreviated\"\r\n \r\n [customMenuItems]=\"customMenuItems\"\r\n [message]=\"message\"\r\n [mine]=\"currentUser?.id === message.user?.id\"\r\n [permissions]=\"source?.permissions\"\r\n [showReplyAction]=\"allowReplies\"\r\n [editing]=\"message.transientState.editing\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (click)=\"isViewingMore = true\"\r\n (editStarted)=\"startEditing(message)\"\r\n (deleted)=\"deleteMessage(message)\"\r\n (editEnded)=\"message.transientState.editing = false\"\r\n (edited)=\"saveEdit(message, $event)\"\r\n (userSelected)=\"selectMessageUser(message)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (liked)=\"likeMessage(message)\"\r\n (unliked)=\"unlikeMessage(message)\"\r\n (reported)=\"reportMessage(message)\"\r\n (selected)=\"selectMessage(message)\"\r\n (shared)=\"sharedMessage($event)\"\r\n ></banta-comment>\r\n <div class=\"inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n <ng-content select=\".inline-replies\"></ng-content>\r\n </div>\r\n </ng-container>\r\n\r\n <a mat-button class=\"nav\" [class.visible]=\"newestLast && (isViewingMore || customSortEnabled)\" href=\"javascript:;\" (click)=\"showNew()\">\r\n <mat-icon>file_download</mat-icon>\r\n New\r\n <ng-container *ngIf=\"newMessages.length >= 1\">\r\n ({{newMessages.length}})\r\n </ng-container>\r\n </a>\r\n <a mat-button class=\"nav\" [class.visible]=\"!newestLast && hasMore && !isLoadingMore\" href=\"javascript:;\" (click)=\"showMore()\">Show more</a>\r\n\r\n <div class=\"loading-more\" *ngIf=\"isLoadingMore\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n\r\n <!-- <div style=\"color: #666\">\r\n n={{newMessages.length}}, m={{messages.length}}, o={{olderMessages.length}},\r\n v={{maxVisibleMessages}}, M={{maxMessages}}\r\n </div> -->\r\n\r\n <ng-content select=\":not([data-before]):not(.inline-replies)\"></ng-content>\r\n</div>\r\n", styles: [":host{flex-grow:1;display:flex;flex-direction:column;opacity:1;transition:.2s opacity ease-in}.message-container{flex-grow:1;overflow-x:hidden;color:#111;background:white;padding:.5em 1em 3em .5em;opacity:1;transition:.5s opacity ease-in-out;position:relative}.message-container.no-scroll{height:auto;overflow-y:visible}.message-container.faded{opacity:.25}.message-container .overlay{position:absolute;inset:0;z-index:10}:host.fixed-height .message-container{overflow-y:auto}:host-context(.mat-dark-theme) .message-container{color:#fff;background:#111111}.empty-state{text-align:center;margin:3em;color:#666}:host-context(.mat-dark-theme) .empty-state{color:#666}a.nav{position:absolute;right:.5em;z-index:10;text-align:center;opacity:0;transition:.4s opacity ease-in-out;pointer-events:none;border-radius:2em;background:#222}a.nav.visible{opacity:1;pointer-events:initial}.loading-more{padding:2em;text-align:center;margin:0 auto;width:-moz-fit-content;width:fit-content}@media (max-width: 400px){.message-container{padding:0 0 3em}}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i4.MatAnchor, selector: "a[mat-button], a[mat-raised-button], a[mat-icon-button], a[mat-fab], a[mat-mini-fab], a[mat-stroked-button], a[mat-flat-button]", inputs: ["disabled", "disableRipple", "color", "tabIndex"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i5.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "diameter", "strokeWidth", "mode", "value"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: i6.CommentComponent, selector: "banta-comment", inputs: ["maxLength", "message", "customMenuItems", "showReplyAction", "permissions", "mine", "editing", "genericAvatarUrl"], outputs: ["loaded", "userSelected", "usernameSelected", "avatarSelected", "reported", "liked", "unliked", "selected", "edited", "deleted", "editStarted", "editEnded", "shared"] }] });
324
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CommentViewComponent, decorators: [{
325
+ type: Component,
326
+ args: [{ selector: 'banta-comment-view', template: "<div class=\"message-container\">\r\n <ng-content select=\"[data-before]\"></ng-content>\r\n\r\n <a mat-button class=\"nav\" [class.visible]=\"!newestLast && (isViewingMore || customSortEnabled)\" href=\"javascript:;\" (click)=\"showNew()\">\r\n <mat-icon>file_upload</mat-icon>\r\n New\r\n <ng-container *ngIf=\"newMessages.length >= 1\">\r\n ({{newMessages.length}})\r\n </ng-container>\r\n </a>\r\n <a mat-button class=\"nav\" [class.visible]=\"newestLast && hasMore && !isLoadingMore\" href=\"javascript:;\" (click)=\"showMore()\">Show earlier</a>\r\n\r\n <ng-container *ngIf=\"messages.length === 0\">\r\n <div class=\"empty-state\" *ngIf=\"showEmptyState\">\r\n Be the first to comment!\r\n </div>\r\n </ng-container>\r\n <ng-container *ngFor=\"let message of messages; trackBy: messageIdentity\">\r\n <banta-comment\r\n *ngIf=\"!message.hidden\"\r\n class=\"abbreviated\"\r\n \r\n [customMenuItems]=\"customMenuItems\"\r\n [message]=\"message\"\r\n [mine]=\"currentUser?.id === message.user?.id\"\r\n [permissions]=\"source?.permissions\"\r\n [showReplyAction]=\"allowReplies\"\r\n [editing]=\"message.transientState.editing\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (click)=\"isViewingMore = true\"\r\n (editStarted)=\"startEditing(message)\"\r\n (deleted)=\"deleteMessage(message)\"\r\n (editEnded)=\"message.transientState.editing = false\"\r\n (edited)=\"saveEdit(message, $event)\"\r\n (userSelected)=\"selectMessageUser(message)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (liked)=\"likeMessage(message)\"\r\n (unliked)=\"unlikeMessage(message)\"\r\n (reported)=\"reportMessage(message)\"\r\n (selected)=\"selectMessage(message)\"\r\n (shared)=\"sharedMessage($event)\"\r\n ></banta-comment>\r\n <div class=\"inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n <ng-content select=\".inline-replies\"></ng-content>\r\n </div>\r\n </ng-container>\r\n\r\n <a mat-button class=\"nav\" [class.visible]=\"newestLast && (isViewingMore || customSortEnabled)\" href=\"javascript:;\" (click)=\"showNew()\">\r\n <mat-icon>file_download</mat-icon>\r\n New\r\n <ng-container *ngIf=\"newMessages.length >= 1\">\r\n ({{newMessages.length}})\r\n </ng-container>\r\n </a>\r\n <a mat-button class=\"nav\" [class.visible]=\"!newestLast && hasMore && !isLoadingMore\" href=\"javascript:;\" (click)=\"showMore()\">Show more</a>\r\n\r\n <div class=\"loading-more\" *ngIf=\"isLoadingMore\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n\r\n <!-- <div style=\"color: #666\">\r\n n={{newMessages.length}}, m={{messages.length}}, o={{olderMessages.length}},\r\n v={{maxVisibleMessages}}, M={{maxMessages}}\r\n </div> -->\r\n\r\n <ng-content select=\":not([data-before]):not(.inline-replies)\"></ng-content>\r\n</div>\r\n", styles: [":host{flex-grow:1;display:flex;flex-direction:column;opacity:1;transition:.2s opacity ease-in}.message-container{flex-grow:1;overflow-x:hidden;color:#111;background:white;padding:.5em 1em 3em .5em;opacity:1;transition:.5s opacity ease-in-out;position:relative}.message-container.no-scroll{height:auto;overflow-y:visible}.message-container.faded{opacity:.25}.message-container .overlay{position:absolute;inset:0;z-index:10}:host.fixed-height .message-container{overflow-y:auto}:host-context(.mat-dark-theme) .message-container{color:#fff;background:#111111}.empty-state{text-align:center;margin:3em;color:#666}:host-context(.mat-dark-theme) .empty-state{color:#666}a.nav{position:absolute;right:.5em;z-index:10;text-align:center;opacity:0;transition:.4s opacity ease-in-out;pointer-events:none;border-radius:2em;background:#222}a.nav.visible{opacity:1;pointer-events:initial}.loading-more{padding:2em;text-align:center;margin:0 auto;width:-moz-fit-content;width:fit-content}@media (max-width: 400px){.message-container{padding:0 0 3em}}\n"] }]
327
+ }], ctorParameters: function () { return [{ type: i1.ChatBackendBase }]; }, propDecorators: { showEmptyState: [{
328
+ type: Input
329
+ }], allowReplies: [{
330
+ type: Input
331
+ }], customMenuItems: [{
332
+ type: Input
333
+ }], commentsQuery: [{
334
+ type: ViewChildren,
335
+ args: [CommentComponent]
336
+ }], fixedHeight: [{
337
+ type: Input
338
+ }, {
339
+ type: HostBinding,
340
+ args: ['class.fixed-height']
341
+ }], selectedMessage: [{
342
+ type: Input
343
+ }], selected: [{
344
+ type: Output
345
+ }], messageEdited: [{
346
+ type: Output
347
+ }], userSelected: [{
348
+ type: Output
349
+ }], reported: [{
350
+ type: Output
351
+ }], liked: [{
352
+ type: Output
353
+ }], unliked: [{
354
+ type: Output
355
+ }], usernameSelected: [{
356
+ type: Output
357
+ }], avatarSelected: [{
358
+ type: Output
359
+ }], shared: [{
360
+ type: Output
361
+ }], deleted: [{
362
+ type: Output
363
+ }], source: [{
364
+ type: Input
365
+ }], genericAvatarUrl: [{
366
+ type: Input
367
+ }], messageContainer: [{
368
+ type: ViewChild,
369
+ args: ['messageContainer']
370
+ }], maxMessages: [{
371
+ type: Input
372
+ }], maxVisibleMessages: [{
373
+ type: Input
374
+ }], newestLast: [{
375
+ type: Input
376
+ }], sortOrderChanged: [{
377
+ type: Output
378
+ }] } });
379
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment-view.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/comments/comment-view/comment-view.component.ts","../../../../../../projects/sdk/src/lib/comments/comment-view/comment-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output, HostBinding, ViewChildren, QueryList } from \"@angular/core\";\r\nimport { User, ChatMessage, CommentsOrder } from '@banta/common';\r\nimport { Subject, Subscription } from 'rxjs';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\nimport { MessageMenuItem } from \"../../message-menu-item\";\r\nimport { CommentComponent } from \"../comment/comment.component\";\r\n\r\nexport interface EditEvent {\r\n    message: ChatMessage;\r\n    newMessage: string;\r\n}\r\n\r\n@Component({\r\n    selector: 'banta-comment-view',\r\n    templateUrl: './comment-view.component.html',\r\n    styleUrls: ['./comment-view.component.scss']\r\n})\r\nexport class CommentViewComponent {\r\n    constructor(\r\n        private backend: ChatBackendBase\r\n    ) {\r\n\r\n    }\r\n\r\n    private _sourceSubs = new Subscription();\r\n    private _source: ChatSourceBase;\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _liked = new Subject<ChatMessage>();\r\n    private _unliked = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _usernameSelected = new Subject<User>();\r\n    private _avatarSelected = new Subject<User>();\r\n    private _shared = new Subject<ChatMessage>();\r\n    private _deleted = new Subject<ChatMessage>();\r\n    private _messageEdited = new Subject<EditEvent>();\r\n\r\n    @Input()\r\n    showEmptyState = true;\r\n\r\n    @Input()\r\n    allowReplies = true;\r\n\r\n    @Input() customMenuItems: MessageMenuItem[] = [];\r\n\r\n    @ViewChildren(CommentComponent)\r\n    commentsQuery: QueryList<CommentComponent>;\r\n\r\n    get comments() {\r\n        return Array.from(this.commentsQuery);\r\n    }\r\n\r\n    /**\r\n     * Returns true if this message can be found within one of the message buffers (older, current, newer)\r\n     * @param message \r\n     */\r\n    isMessageLoadedInContext(message: ChatMessage) {\r\n        return this.olderMessages.find(x => x.id === message.id)\r\n            || this.messages.find(x => x.id === message.id)\r\n            || this.newMessages.find(x => x.id === message.id);\r\n    }\r\n\r\n    async loadMessageInContext(message: ChatMessage) {\r\n        await this.sourceLoaded;\r\n        \r\n        console.log(`Loading message ${message.id} in context...`);\r\n        while (this.hasMore && !this.isMessageLoadedInContext(message)) {\r\n            console.log(`...Need to load more comments to find ${message.id}`);\r\n            await this.showMore();\r\n        }\r\n\r\n        if (!this.isMessageLoadedInContext(message)) {\r\n            console.error(`Error while loading message in context: Failed to find message ${message.id}, maybe it was deleted!`);\r\n            return false;\r\n        }\r\n\r\n        let pageSize = this.maxVisibleMessages;\r\n        let items = [].concat(this.olderMessages, this.messages, this.newMessages);\r\n        let index = items.findIndex(x => x.id === message.id);\r\n\r\n        if (index < 0) {\r\n            console.error(`Error while loading message in context: Message was not present in message list!`);\r\n            return false;\r\n        }\r\n\r\n        let startIndex = Math.max(0, index - pageSize / 2);        \r\n        this.newMessages = items.splice(0, startIndex);\r\n        this.messages = items.splice(0, pageSize);\r\n        this.olderMessages = items;\r\n        this.isViewingMore = true;\r\n    }\r\n\r\n    /**\r\n     * Get the CommentComponent instantiated for the given ChatMessage,\r\n     * if it exists in the current view. Note that messages which are not \r\n     * currently shown to the user will not return a CommentComponent.\r\n     * @param message \r\n     * @returns \r\n     */\r\n    getCommentComponentForMessage(message: ChatMessage) {\r\n        if (!message)\r\n            throw new Error(`You must pass a valid ChatMessage`);\r\n        return this.comments.find(x => x.message?.id === message.id);\r\n    }\r\n\r\n    @Input()\r\n    @HostBinding('class.fixed-height')\r\n    fixedHeight: boolean;\r\n\r\n    @Input()\r\n    selectedMessage: ChatMessage;\r\n    \r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output() \r\n    get messageEdited() {\r\n        return this._messageEdited.asObservable();\r\n    }\r\n\r\n    saveEdit(message: ChatMessage, newMessage: string) {\r\n        this._messageEdited.next({ message, newMessage });\r\n    }\r\n\r\n    @Output() get userSelected() { return this._userSelected; }\r\n    @Output() get reported() { return this._reported; }\r\n    @Output() get liked() { return this._liked; }\r\n    @Output() get unliked() { return this._unliked; }\r\n    @Output() get usernameSelected() { return this._usernameSelected; }\r\n    @Output() get avatarSelected() { return this._avatarSelected; }\r\n    @Output() get shared() { return this._shared; }\r\n    @Output() get deleted() { return this._deleted; }\r\n\r\n    menuMessage: ChatMessage = null;\r\n    messages: ChatMessage[] = [];\r\n    currentUser: User;\r\n\r\n    @Input()\r\n    get source() {\r\n        return this._source;\r\n    }\r\n\r\n    likeMessage(message: ChatMessage) {\r\n        this._liked.next(message);\r\n    }\r\n\r\n    unlikeMessage(message: ChatMessage) {\r\n        this._unliked.next(message);\r\n    }\r\n\r\n    reportMessage(message: ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    selectMessage(message: ChatMessage) {\r\n        this._selected.next(message);\r\n    }\r\n\r\n    selectMessageUser(message: ChatMessage) {\r\n        this._userSelected.next(message);\r\n    }\r\n\r\n    selectUsername(user: User) {\r\n        this._usernameSelected.next(user);\r\n    }\r\n\r\n    selectAvatar(user: User) {\r\n        this._avatarSelected.next(user);\r\n    }\r\n\r\n    sharedMessage(message: ChatMessage) {\r\n        this._shared.next(message);\r\n    }\r\n\r\n    startEditing(message: ChatMessage) {\r\n        this.messages.forEach(m => m.transientState.editing = false);\r\n        message.transientState.editing = true;\r\n    }\r\n\r\n    deleteMessage(message: ChatMessage) {\r\n        this._deleted.next(message);\r\n    }\r\n\r\n    customSortEnabled = false;\r\n\r\n    markSourceLoaded: () => void;\r\n    sourceLoaded = new Promise<void>(r => this.markSourceLoaded = r);\r\n\r\n    set source(value) {\r\n        this.customSortEnabled = value?.sortOrder !== CommentsOrder.NEWEST;\r\n        this.newMessages = [];\r\n        this.olderMessages = [];\r\n\r\n        (window as any).bantaSourceDebug = value;\r\n\r\n        if (this._sourceSubs) {\r\n            this._sourceSubs.unsubscribe();\r\n            this._sourceSubs = null;\r\n        }\r\n        this._source = value;\r\n\r\n        if (value) {\r\n            const messages = (value.messages || []).slice();\r\n            this.messages = messages;\r\n            this.olderMessages = messages.splice(this.maxVisibleMessages, messages.length);\r\n            this.hasMore = true; //this.olderMessages.length > 0;\r\n\r\n            this._sourceSubs = new Subscription();\r\n            this._sourceSubs.add(this._source.messageReceived.subscribe(msg => this.messageReceived(msg)));\r\n            this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));\r\n\r\n            this._sourceSubs.add(\r\n                this.backend.userChanged.subscribe(user => this.currentUser = user)\r\n            );\r\n\r\n            this.getInitialMessages();\r\n        }\r\n    }\r\n\r\n    private async getInitialMessages() {\r\n        let messages = (await this._source.getExistingMessages());\r\n        messages.forEach(m => m.transientState ??= {});\r\n        this.messages = this.newestLast ? messages.slice().reverse() : messages;\r\n        this.sortMessages();\r\n        if (this.markSourceLoaded)\r\n            this.markSourceLoaded();\r\n    }\r\n\r\n    @Input()\r\n    genericAvatarUrl: string;\r\n\r\n    @ViewChild('messageContainer')\r\n    messageContainer: ElementRef<HTMLElement>;\r\n\r\n    @Input()\r\n    maxMessages = 2000;\r\n\r\n    @Input()\r\n    maxVisibleMessages: number = 200;\r\n\r\n    @Input()\r\n    newestLast = false;\r\n\r\n    isViewingMore = false;\r\n    isLoadingMore = false;\r\n    hasMore = false;\r\n\r\n    newMessages: ChatMessage[] = [];\r\n    olderMessages: ChatMessage[] = [];\r\n\r\n    messageIdentity(index: number, chatMessage: ChatMessage) {\r\n        return chatMessage.id;\r\n    }\r\n\r\n    @Output()\r\n    sortOrderChanged = new Subject<CommentsOrder>();\r\n\r\n    async showNew() {\r\n        let naturalOrder = CommentsOrder.NEWEST;\r\n        if (this.source && this.source.sortOrder !== naturalOrder) {\r\n            this.sortOrderChanged.next(naturalOrder);\r\n            return;\r\n        }\r\n\r\n        this.isViewingMore = false;\r\n\r\n        if (this.newestLast)\r\n            this.messages = this.messages.concat(this.newMessages.splice(0, this.newMessages.length));\r\n        else\r\n            this.messages = this.newMessages.splice(0, this.newMessages.length).concat(this.messages);\r\n        let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);\r\n        this.olderMessages = overflow.concat(this.olderMessages);\r\n        this.olderMessages.splice(this.maxMessages - this.maxVisibleMessages, this.olderMessages.length);\r\n        this.hasMore = this.olderMessages.length > 0;\r\n    }\r\n\r\n    async showMore() {\r\n        this.isViewingMore = true;\r\n\r\n        if (this.olderMessages.length > 0) {\r\n            this.isLoadingMore = false;\r\n            this.messages = this.messages.concat(this.olderMessages.splice(0, 50));\r\n        }\r\n        this.isLoadingMore = true;\r\n\r\n        let nextPageSize = 20;\r\n        let lastMessage: ChatMessage;\r\n\r\n        if (this.newestLast) {\r\n            lastMessage = this.olderMessages[0] ?? this.messages[0];\r\n        } else {\r\n            lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];\r\n        }\r\n\r\n        if (!lastMessage) {\r\n            this.isLoadingMore = false;\r\n            this.hasMore = false;\r\n            return;\r\n        }\r\n\r\n        let messages = await this.source.loadAfter(lastMessage, nextPageSize);\r\n\r\n        if (this.newestLast)\r\n            messages = messages.slice().reverse();\r\n        \r\n        messages.forEach(m => m.transientState ??= {});\r\n\r\n        if (this.newestLast)\r\n            this.messages = messages.concat(this.messages);\r\n        else\r\n            this.messages = this.messages.concat(messages);\r\n        this.isLoadingMore = false;\r\n        if (messages.length === 0) {\r\n            console.log(`Reached the end of the list.`);\r\n            this.hasMore = false;\r\n        }\r\n    }\r\n\r\n    private addMessage(message: ChatMessage) {\r\n\r\n        if (!message.transientState)\r\n            message.transientState ??= {};\r\n        \r\n        let destination = this.messages;\r\n        let bucket = this.olderMessages;\r\n        let newestLast = this.newestLast;\r\n\r\n        if (this.isViewingMore) {\r\n            destination = this.newMessages;\r\n            bucket = null;\r\n        }\r\n\r\n        \r\n        if (newestLast) {\r\n            destination.push(message);\r\n            let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n            bucket?.push(...overflow);\r\n        } else {\r\n            destination.unshift(message);\r\n            let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n            bucket?.unshift(...overflow);\r\n        }\r\n\r\n        if (bucket?.length > 0)\r\n            this.hasMore = true;\r\n\r\n        message.pagingCursor = String(this.incrementPagingCursors());\r\n        this.sortMessages();\r\n    }\r\n\r\n    private incrementPagingCursors() {\r\n        if (this.source.sortOrder !== CommentsOrder.NEWEST)\r\n            return;\r\n        \r\n        let maxPagingCursor = 0;\r\n        for (let group of [this.messages, this.olderMessages, this.newMessages]) {\r\n            for (let message of group) {\r\n                if (message.pagingCursor) {\r\n                    let pagingCursor = Number(message.pagingCursor) + 1;\r\n                    if (pagingCursor > maxPagingCursor)\r\n                        maxPagingCursor = pagingCursor;\r\n                    message.pagingCursor = String(pagingCursor);\r\n                }\r\n            }\r\n        }\r\n\r\n        return maxPagingCursor;\r\n    }\r\n\r\n    /**\r\n     * Wait for all currently visible comments to be fully loaded, including all attachments.\r\n     * Doing this will prevent layout shift when scrolling to a specific comment.\r\n     */\r\n    async waitForAllCommentsToLoad() {\r\n        await new Promise(r => setTimeout(r, 100));\r\n        await this.sourceLoaded;\r\n        await Promise.all(this.comments.map(x => x.waitForLoad()));\r\n    }\r\n\r\n    private sortMessages() {\r\n        if (!this.source)\r\n            return;\r\n        \r\n        let sorter: (a: ChatMessage, b: ChatMessage) => number;\r\n\r\n        if (this.source.sortOrder === CommentsOrder.LIKES)\r\n            sorter = (a, b) => b.likes - a.likes;\r\n        else if (this.source.sortOrder === CommentsOrder.NEWEST)\r\n            sorter = (a, b) => (b.sentAt - a.sentAt) * (this.newestLast ? -1 : 1);\r\n        else if (this.source.sortOrder === CommentsOrder.OLDEST)\r\n            sorter = (a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? -1 : 1);\r\n\r\n        this.messages.sort(sorter);\r\n        this.olderMessages.sort(sorter);\r\n        this.newMessages.sort(sorter);\r\n    }\r\n\r\n    private messageReceived(message: ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (this.isScrolledToLatest())\r\n            setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    isScrolledToLatest() {\r\n        if (!this.messageContainer)\r\n            return false;\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n        const currentScroll = el.scrollTop;\r\n        const currentTotal = el.scrollHeight - el.offsetHeight;\r\n\r\n        return currentScroll > currentTotal - 10;\r\n    }\r\n\r\n    private messageSent(message: ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (!this.messageContainer)\r\n            return;\r\n\r\n        this.scrollToLatest();\r\n    }\r\n\r\n    scrollToLatest() {\r\n        if (!this.messageContainer)\r\n            return;\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n        el.scrollTop = el.scrollHeight;\r\n    }\r\n\r\n    mentionsMe(message: ChatMessage) {\r\n        if (!this.currentUser)\r\n            return false;\r\n\r\n        if (message.message.includes(`@${this.currentUser.username}`))\r\n            return true;\r\n\r\n        return false;\r\n    }\r\n}\r\n","<div class=\"message-container\">\r\n    <ng-content select=\"[data-before]\"></ng-content>\r\n\r\n    <a mat-button class=\"nav\" [class.visible]=\"!newestLast && (isViewingMore || customSortEnabled)\" href=\"javascript:;\" (click)=\"showNew()\">\r\n        <mat-icon>file_upload</mat-icon>\r\n        New\r\n        <ng-container *ngIf=\"newMessages.length >= 1\">\r\n            ({{newMessages.length}})\r\n        </ng-container>\r\n    </a>\r\n    <a mat-button class=\"nav\" [class.visible]=\"newestLast && hasMore && !isLoadingMore\" href=\"javascript:;\" (click)=\"showMore()\">Show earlier</a>\r\n\r\n    <ng-container *ngIf=\"messages.length === 0\">\r\n        <div class=\"empty-state\" *ngIf=\"showEmptyState\">\r\n            Be the first to comment!\r\n        </div>\r\n    </ng-container>\r\n    <ng-container *ngFor=\"let message of messages; trackBy: messageIdentity\">\r\n        <banta-comment\r\n            *ngIf=\"!message.hidden\"\r\n            class=\"abbreviated\"\r\n            \r\n            [customMenuItems]=\"customMenuItems\"\r\n            [message]=\"message\"\r\n            [mine]=\"currentUser?.id === message.user?.id\"\r\n            [permissions]=\"source?.permissions\"\r\n            [showReplyAction]=\"allowReplies\"\r\n            [editing]=\"message.transientState.editing\"\r\n            [genericAvatarUrl]=\"genericAvatarUrl\"\r\n            (click)=\"isViewingMore = true\"\r\n            (editStarted)=\"startEditing(message)\"\r\n            (deleted)=\"deleteMessage(message)\"\r\n            (editEnded)=\"message.transientState.editing = false\"\r\n            (edited)=\"saveEdit(message, $event)\"\r\n            (userSelected)=\"selectMessageUser(message)\"\r\n            (avatarSelected)=\"selectAvatar($event)\"\r\n            (usernameSelected)=\"selectUsername($event)\"\r\n            (liked)=\"likeMessage(message)\"\r\n            (unliked)=\"unlikeMessage(message)\"\r\n            (reported)=\"reportMessage(message)\"\r\n            (selected)=\"selectMessage(message)\"\r\n            (shared)=\"sharedMessage($event)\"\r\n            ></banta-comment>\r\n        <div class=\"inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n            <ng-content select=\".inline-replies\"></ng-content>\r\n        </div>\r\n    </ng-container>\r\n\r\n    <a mat-button class=\"nav\" [class.visible]=\"newestLast && (isViewingMore || customSortEnabled)\" href=\"javascript:;\" (click)=\"showNew()\">\r\n        <mat-icon>file_download</mat-icon>\r\n        New\r\n        <ng-container *ngIf=\"newMessages.length >= 1\">\r\n            ({{newMessages.length}})\r\n        </ng-container>\r\n    </a>\r\n    <a mat-button class=\"nav\" [class.visible]=\"!newestLast && hasMore && !isLoadingMore\" href=\"javascript:;\" (click)=\"showMore()\">Show more</a>\r\n\r\n    <div class=\"loading-more\" *ngIf=\"isLoadingMore\">\r\n        <mat-spinner></mat-spinner>\r\n    </div>\r\n\r\n    <!-- <div style=\"color: #666\">\r\n        n={{newMessages.length}}, m={{messages.length}}, o={{olderMessages.length}},\r\n        v={{maxVisibleMessages}}, M={{maxMessages}}\r\n    </div> -->\r\n\r\n    <ng-content select=\":not([data-before]):not(.inline-replies)\"></ng-content>\r\n</div>\r\n"]}
@@ -0,0 +1,111 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommentComponent } from './comment/comment.component';
3
+ import { CommentViewComponent } from './comment-view/comment-view.component';
4
+ import { BantaCommentsComponent } from './banta-comments/banta-comments.component';
5
+ import { CommonModule } from '@angular/common';
6
+ import { FormsModule } from '@angular/forms';
7
+ import { BantaCommonModule } from '../common';
8
+ import { LiveCommentComponent } from './live-comment.component';
9
+ import { EmojiModule } from '../emoji';
10
+ import { MatIconModule } from '@angular/material/icon';
11
+ import { MatButtonModule } from '@angular/material/button';
12
+ import { MatMenuModule } from '@angular/material/menu';
13
+ import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
14
+ import { MatFormFieldModule } from '@angular/material/form-field';
15
+ import { MatInputModule } from '@angular/material/input';
16
+ import { TextFieldModule } from '@angular/cdk/text-field';
17
+ import { CommentFieldComponent } from './comment-field/comment-field.component';
18
+ import { MatTooltipModule } from '@angular/material/tooltip';
19
+ import { CommentSortComponent } from "./comment-sort/comment-sort.component";
20
+ import { MatSelectModule } from "@angular/material/select";
21
+ import { AttachmentButtonComponent } from './attachment-button/attachment-button.component';
22
+ import { BantaReplySendOptionsDirective } from './reply-send-options.directive';
23
+ import { OverlayModule } from '@angular/cdk/overlay';
24
+ import { PortalModule } from '@angular/cdk/portal';
25
+ import { AttachmentScraperDirective } from './attachment-scraper.directive';
26
+ import * as i0 from "@angular/core";
27
+ const COMPONENTS = [
28
+ CommentComponent,
29
+ CommentViewComponent,
30
+ BantaCommentsComponent,
31
+ LiveCommentComponent,
32
+ CommentFieldComponent,
33
+ CommentSortComponent,
34
+ AttachmentButtonComponent,
35
+ BantaReplySendOptionsDirective,
36
+ AttachmentScraperDirective
37
+ ];
38
+ export class CommentsModule {
39
+ }
40
+ CommentsModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CommentsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
41
+ CommentsModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "14.2.12", ngImport: i0, type: CommentsModule, declarations: [CommentComponent,
42
+ CommentViewComponent,
43
+ BantaCommentsComponent,
44
+ LiveCommentComponent,
45
+ CommentFieldComponent,
46
+ CommentSortComponent,
47
+ AttachmentButtonComponent,
48
+ BantaReplySendOptionsDirective,
49
+ AttachmentScraperDirective], imports: [CommonModule,
50
+ TextFieldModule,
51
+ FormsModule,
52
+ MatIconModule,
53
+ MatFormFieldModule,
54
+ MatInputModule,
55
+ MatButtonModule,
56
+ MatMenuModule,
57
+ MatProgressSpinnerModule,
58
+ BantaCommonModule,
59
+ EmojiModule,
60
+ MatTooltipModule,
61
+ MatSelectModule,
62
+ OverlayModule,
63
+ PortalModule], exports: [CommentComponent,
64
+ CommentViewComponent,
65
+ BantaCommentsComponent,
66
+ LiveCommentComponent,
67
+ CommentFieldComponent,
68
+ CommentSortComponent,
69
+ AttachmentButtonComponent,
70
+ BantaReplySendOptionsDirective,
71
+ AttachmentScraperDirective] });
72
+ CommentsModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CommentsModule, imports: [CommonModule,
73
+ TextFieldModule,
74
+ FormsModule,
75
+ MatIconModule,
76
+ MatFormFieldModule,
77
+ MatInputModule,
78
+ MatButtonModule,
79
+ MatMenuModule,
80
+ MatProgressSpinnerModule,
81
+ BantaCommonModule,
82
+ EmojiModule,
83
+ MatTooltipModule,
84
+ MatSelectModule,
85
+ OverlayModule,
86
+ PortalModule] });
87
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CommentsModule, decorators: [{
88
+ type: NgModule,
89
+ args: [{
90
+ declarations: COMPONENTS,
91
+ imports: [
92
+ CommonModule,
93
+ TextFieldModule,
94
+ FormsModule,
95
+ MatIconModule,
96
+ MatFormFieldModule,
97
+ MatInputModule,
98
+ MatButtonModule,
99
+ MatMenuModule,
100
+ MatProgressSpinnerModule,
101
+ BantaCommonModule,
102
+ EmojiModule,
103
+ MatTooltipModule,
104
+ MatSelectModule,
105
+ OverlayModule,
106
+ PortalModule
107
+ ],
108
+ exports: COMPONENTS
109
+ }]
110
+ }] });
111
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbWVudHMubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2RrL3NyYy9saWIvY29tbWVudHMvY29tbWVudHMubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDL0QsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7QUFDN0UsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sMkNBQTJDLENBQUM7QUFDbkYsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUM3QyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFDOUMsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDaEUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUN2QyxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDdkQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQzNELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUN2RCxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUM5RSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUNsRSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDekQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQzFELE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLHlDQUF5QyxDQUFDO0FBQ2hGLE9BQU8sRUFBQyxnQkFBZ0IsRUFBQyxNQUFNLDJCQUEyQixDQUFDO0FBQzNELE9BQU8sRUFBQyxvQkFBb0IsRUFBQyxNQUFNLHVDQUF1QyxDQUFDO0FBQzNFLE9BQU8sRUFBQyxlQUFlLEVBQUMsTUFBTSwwQkFBMEIsQ0FBQztBQUN6RCxPQUFPLEVBQUUseUJBQXlCLEVBQUUsTUFBTSxpREFBaUQsQ0FBQztBQUM1RixPQUFPLEVBQUUsOEJBQThCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNoRixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDckQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ25ELE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxNQUFNLGdDQUFnQyxDQUFDOztBQUU1RSxNQUFNLFVBQVUsR0FBRztJQUNmLGdCQUFnQjtJQUNoQixvQkFBb0I7SUFDcEIsc0JBQXNCO0lBQ3RCLG9CQUFvQjtJQUNwQixxQkFBcUI7SUFDckIsb0JBQW9CO0lBQ3BCLHlCQUF5QjtJQUN6Qiw4QkFBOEI7SUFDOUIsMEJBQTBCO0NBQzdCLENBQUM7QUF1QkYsTUFBTSxPQUFPLGNBQWM7OzRHQUFkLGNBQWM7NkdBQWQsY0FBYyxpQkFoQ3ZCLGdCQUFnQjtRQUNoQixvQkFBb0I7UUFDcEIsc0JBQXNCO1FBQ3RCLG9CQUFvQjtRQUNwQixxQkFBcUI7UUFDckIsb0JBQW9CO1FBQ3BCLHlCQUF5QjtRQUN6Qiw4QkFBOEI7UUFDOUIsMEJBQTBCLGFBTTFCLFlBQVk7UUFDWixlQUFlO1FBQ2YsV0FBVztRQUNYLGFBQWE7UUFDYixrQkFBa0I7UUFDbEIsY0FBYztRQUNkLGVBQWU7UUFDZixhQUFhO1FBQ2Isd0JBQXdCO1FBQ3hCLGlCQUFpQjtRQUNqQixXQUFXO1FBQ1gsZ0JBQWdCO1FBQ2hCLGVBQWU7UUFDZixhQUFhO1FBQ2IsWUFBWSxhQTVCWixnQkFBZ0I7UUFDaEIsb0JBQW9CO1FBQ3BCLHNCQUFzQjtRQUN0QixvQkFBb0I7UUFDcEIscUJBQXFCO1FBQ3JCLG9CQUFvQjtRQUNwQix5QkFBeUI7UUFDekIsOEJBQThCO1FBQzlCLDBCQUEwQjs2R0F3QmpCLGNBQWMsWUFsQnZCLFlBQVk7UUFDWixlQUFlO1FBQ2YsV0FBVztRQUNYLGFBQWE7UUFDYixrQkFBa0I7UUFDbEIsY0FBYztRQUNkLGVBQWU7UUFDZixhQUFhO1FBQ2Isd0JBQXdCO1FBQ3hCLGlCQUFpQjtRQUNqQixXQUFXO1FBQ1gsZ0JBQWdCO1FBQ2hCLGVBQWU7UUFDZixhQUFhO1FBQ2IsWUFBWTs0RkFJSCxjQUFjO2tCQXJCMUIsUUFBUTttQkFBQztvQkFDTixZQUFZLEVBQUUsVUFBVTtvQkFDMUIsT0FBTyxFQUFFO3dCQUNQLFlBQVk7d0JBQ1osZUFBZTt3QkFDZixXQUFXO3dCQUNYLGFBQWE7d0JBQ2Isa0JBQWtCO3dCQUNsQixjQUFjO3dCQUNkLGVBQWU7d0JBQ2YsYUFBYTt3QkFDYix3QkFBd0I7d0JBQ3hCLGlCQUFpQjt3QkFDakIsV0FBVzt3QkFDWCxnQkFBZ0I7d0JBQ2hCLGVBQWU7d0JBQ2YsYUFBYTt3QkFDYixZQUFZO3FCQUNiO29CQUNDLE9BQU8sRUFBRSxVQUFVO2lCQUN0QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IENvbW1lbnRDb21wb25lbnQgfSBmcm9tICcuL2NvbW1lbnQvY29tbWVudC5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBDb21tZW50Vmlld0NvbXBvbmVudCB9IGZyb20gJy4vY29tbWVudC12aWV3L2NvbW1lbnQtdmlldy5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBCYW50YUNvbW1lbnRzQ29tcG9uZW50IH0gZnJvbSAnLi9iYW50YS1jb21tZW50cy9iYW50YS1jb21tZW50cy5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5pbXBvcnQgeyBGb3Jtc01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcclxuaW1wb3J0IHsgQmFudGFDb21tb25Nb2R1bGUgfSBmcm9tICcuLi9jb21tb24nO1xyXG5pbXBvcnQgeyBMaXZlQ29tbWVudENvbXBvbmVudCB9IGZyb20gJy4vbGl2ZS1jb21tZW50LmNvbXBvbmVudCc7XHJcbmltcG9ydCB7IEVtb2ppTW9kdWxlIH0gZnJvbSAnLi4vZW1vamknO1xyXG5pbXBvcnQgeyBNYXRJY29uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvaWNvbic7XHJcbmltcG9ydCB7IE1hdEJ1dHRvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2J1dHRvbic7XHJcbmltcG9ydCB7IE1hdE1lbnVNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9tZW51JztcclxuaW1wb3J0IHsgTWF0UHJvZ3Jlc3NTcGlubmVyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvcHJvZ3Jlc3Mtc3Bpbm5lcic7XHJcbmltcG9ydCB7IE1hdEZvcm1GaWVsZE1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2Zvcm0tZmllbGQnO1xyXG5pbXBvcnQgeyBNYXRJbnB1dE1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2lucHV0JztcclxuaW1wb3J0IHsgVGV4dEZpZWxkTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL3RleHQtZmllbGQnO1xyXG5pbXBvcnQgeyBDb21tZW50RmllbGRDb21wb25lbnQgfSBmcm9tICcuL2NvbW1lbnQtZmllbGQvY29tbWVudC1maWVsZC5jb21wb25lbnQnO1xyXG5pbXBvcnQge01hdFRvb2x0aXBNb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL3Rvb2x0aXAnO1xyXG5pbXBvcnQge0NvbW1lbnRTb3J0Q29tcG9uZW50fSBmcm9tIFwiLi9jb21tZW50LXNvcnQvY29tbWVudC1zb3J0LmNvbXBvbmVudFwiO1xyXG5pbXBvcnQge01hdFNlbGVjdE1vZHVsZX0gZnJvbSBcIkBhbmd1bGFyL21hdGVyaWFsL3NlbGVjdFwiO1xyXG5pbXBvcnQgeyBBdHRhY2htZW50QnV0dG9uQ29tcG9uZW50IH0gZnJvbSAnLi9hdHRhY2htZW50LWJ1dHRvbi9hdHRhY2htZW50LWJ1dHRvbi5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBCYW50YVJlcGx5U2VuZE9wdGlvbnNEaXJlY3RpdmUgfSBmcm9tICcuL3JlcGx5LXNlbmQtb3B0aW9ucy5kaXJlY3RpdmUnO1xyXG5pbXBvcnQgeyBPdmVybGF5TW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY2RrL292ZXJsYXknO1xyXG5pbXBvcnQgeyBQb3J0YWxNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jZGsvcG9ydGFsJztcclxuaW1wb3J0IHsgQXR0YWNobWVudFNjcmFwZXJEaXJlY3RpdmUgfSBmcm9tICcuL2F0dGFjaG1lbnQtc2NyYXBlci5kaXJlY3RpdmUnO1xyXG5cclxuY29uc3QgQ09NUE9ORU5UUyA9IFtcclxuICAgIENvbW1lbnRDb21wb25lbnQsXHJcbiAgICBDb21tZW50Vmlld0NvbXBvbmVudCxcclxuICAgIEJhbnRhQ29tbWVudHNDb21wb25lbnQsXHJcbiAgICBMaXZlQ29tbWVudENvbXBvbmVudCxcclxuICAgIENvbW1lbnRGaWVsZENvbXBvbmVudCxcclxuICAgIENvbW1lbnRTb3J0Q29tcG9uZW50LFxyXG4gICAgQXR0YWNobWVudEJ1dHRvbkNvbXBvbmVudCxcclxuICAgIEJhbnRhUmVwbHlTZW5kT3B0aW9uc0RpcmVjdGl2ZSxcclxuICAgIEF0dGFjaG1lbnRTY3JhcGVyRGlyZWN0aXZlXHJcbl07XHJcblxyXG5ATmdNb2R1bGUoe1xyXG4gICAgZGVjbGFyYXRpb25zOiBDT01QT05FTlRTLFxyXG4gIGltcG9ydHM6IFtcclxuICAgIENvbW1vbk1vZHVsZSxcclxuICAgIFRleHRGaWVsZE1vZHVsZSxcclxuICAgIEZvcm1zTW9kdWxlLFxyXG4gICAgTWF0SWNvbk1vZHVsZSxcclxuICAgIE1hdEZvcm1GaWVsZE1vZHVsZSxcclxuICAgIE1hdElucHV0TW9kdWxlLFxyXG4gICAgTWF0QnV0dG9uTW9kdWxlLFxyXG4gICAgTWF0TWVudU1vZHVsZSxcclxuICAgIE1hdFByb2dyZXNzU3Bpbm5lck1vZHVsZSxcclxuICAgIEJhbnRhQ29tbW9uTW9kdWxlLFxyXG4gICAgRW1vamlNb2R1bGUsXHJcbiAgICBNYXRUb29sdGlwTW9kdWxlLFxyXG4gICAgTWF0U2VsZWN0TW9kdWxlLFxyXG4gICAgT3ZlcmxheU1vZHVsZSxcclxuICAgIFBvcnRhbE1vZHVsZVxyXG4gIF0sXHJcbiAgICBleHBvcnRzOiBDT01QT05FTlRTXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBDb21tZW50c01vZHVsZSB7XHJcbn1cclxuIl19