@banta/sdk 6.0.3 → 7.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/fesm2022/banta-sdk.mjs +327 -6762
- package/fesm2022/banta-sdk.mjs.map +1 -1
- package/index.d.ts +1472 -3
- package/package.json +11 -13
- package/esm2022/banta-sdk.mjs +0 -5
- package/esm2022/lib/attachment-scraper.mjs +0 -2
- package/esm2022/lib/banta/banta.component.mjs +0 -207
- package/esm2022/lib/banta-logo.component.mjs +0 -11
- package/esm2022/lib/banta-sdk.module.mjs +0 -135
- package/esm2022/lib/chat/banta-chat/banta-chat.component.mjs +0 -209
- package/esm2022/lib/chat/chat-message/chat-message.component.mjs +0 -62
- package/esm2022/lib/chat/chat-view/chat-view.component.mjs +0 -166
- package/esm2022/lib/chat/chat.module.mjs +0 -51
- package/esm2022/lib/chat/index.mjs +0 -6
- package/esm2022/lib/chat/live-chat-message.component.mjs +0 -80
- package/esm2022/lib/chat-backend-base.mjs +0 -31
- package/esm2022/lib/chat-backend.mjs +0 -199
- package/esm2022/lib/chat-source-base.mjs +0 -2
- package/esm2022/lib/chat-source.mjs +0 -282
- package/esm2022/lib/comments/attachment-button/attachment-button.component.mjs +0 -75
- package/esm2022/lib/comments/attachment-scraper.directive.mjs +0 -101
- package/esm2022/lib/comments/banta-comments/banta-comments.component.mjs +0 -817
- package/esm2022/lib/comments/comment/comment.component.mjs +0 -224
- package/esm2022/lib/comments/comment-field/comment-field.component.mjs +0 -411
- package/esm2022/lib/comments/comment-sort/comment-sort.component.mjs +0 -37
- package/esm2022/lib/comments/comment-view/comment-view.component.mjs +0 -780
- package/esm2022/lib/comments/comments.module.mjs +0 -127
- package/esm2022/lib/comments/index.mjs +0 -12
- package/esm2022/lib/comments/inline-replies.directive.mjs +0 -13
- package/esm2022/lib/comments/live-comment.component.mjs +0 -80
- package/esm2022/lib/comments/reply-send-options.directive.mjs +0 -13
- package/esm2022/lib/common/attachment/attachment.component.mjs +0 -128
- package/esm2022/lib/common/attachments/attachments.component.mjs +0 -75
- package/esm2022/lib/common/common.module.mjs +0 -68
- package/esm2022/lib/common/index.mjs +0 -11
- package/esm2022/lib/common/lazy-connection.mjs +0 -15
- package/esm2022/lib/common/lightbox/lightbox.component.mjs +0 -31
- package/esm2022/lib/common/markdown-to-html.pipe.mjs +0 -83
- package/esm2022/lib/common/mention-linker.pipe.mjs +0 -35
- package/esm2022/lib/common/timer-pool.service.mjs +0 -85
- package/esm2022/lib/common/timestamp.component.mjs +0 -124
- package/esm2022/lib/common/trust-resource-url.pipe.mjs +0 -22
- package/esm2022/lib/emoji/emoji-selector-button.component.mjs +0 -115
- package/esm2022/lib/emoji/emoji-selector-panel/emoji-selector-panel.component.mjs +0 -93
- package/esm2022/lib/emoji/emoji.module.mjs +0 -55
- package/esm2022/lib/emoji/emojis.mjs +0 -6508
- package/esm2022/lib/emoji/index.mjs +0 -5
- package/esm2022/lib/giphy-attachments.mjs +0 -16
- package/esm2022/lib/index.mjs +0 -20
- package/esm2022/lib/live-message.component.mjs +0 -96
- package/esm2022/lib/message-menu-item.mjs +0 -2
- package/esm2022/lib/sdk-options.mjs +0 -3
- package/esm2022/lib/static-chat-source.mjs +0 -101
- package/esm2022/lib/tweet-attachments.mjs +0 -13
- package/esm2022/lib/url-attachments.mjs +0 -42
- package/esm2022/lib/youtube-attachments.mjs +0 -29
- package/esm2022/public-api.mjs +0 -5
- package/lib/attachment-scraper.d.ts +0 -15
- package/lib/banta/banta.component.d.ts +0 -59
- package/lib/banta-logo.component.d.ts +0 -5
- package/lib/banta-sdk.module.d.ts +0 -32
- package/lib/chat/banta-chat/banta-chat.component.d.ts +0 -79
- package/lib/chat/chat-message/chat-message.component.d.ts +0 -21
- package/lib/chat/chat-view/chat-view.component.d.ts +0 -52
- package/lib/chat/chat.module.d.ts +0 -15
- package/lib/chat/index.d.ts +0 -5
- package/lib/chat/live-chat-message.component.d.ts +0 -23
- package/lib/chat-backend-base.d.ts +0 -72
- package/lib/chat-backend.d.ts +0 -66
- package/lib/chat-source-base.d.ts +0 -51
- package/lib/chat-source.d.ts +0 -85
- package/lib/comments/attachment-button/attachment-button.component.d.ts +0 -17
- package/lib/comments/attachment-scraper.directive.d.ts +0 -21
- package/lib/comments/banta-comments/banta-comments.component.d.ts +0 -216
- package/lib/comments/comment/comment.component.d.ts +0 -90
- package/lib/comments/comment-field/comment-field.component.d.ts +0 -92
- package/lib/comments/comment-sort/comment-sort.component.d.ts +0 -16
- package/lib/comments/comment-view/comment-view.component.d.ts +0 -205
- package/lib/comments/comments.module.d.ts +0 -34
- package/lib/comments/index.d.ts +0 -11
- package/lib/comments/inline-replies.directive.d.ts +0 -5
- package/lib/comments/live-comment.component.d.ts +0 -23
- package/lib/comments/reply-send-options.directive.d.ts +0 -5
- package/lib/common/attachment/attachment.component.d.ts +0 -34
- package/lib/common/attachments/attachments.component.d.ts +0 -26
- package/lib/common/common.module.d.ts +0 -19
- package/lib/common/index.d.ts +0 -10
- package/lib/common/lazy-connection.d.ts +0 -6
- package/lib/common/lightbox/lightbox.component.d.ts +0 -14
- package/lib/common/markdown-to-html.pipe.d.ts +0 -13
- package/lib/common/mention-linker.pipe.d.ts +0 -13
- package/lib/common/timer-pool.service.d.ts +0 -15
- package/lib/common/timestamp.component.d.ts +0 -19
- package/lib/common/trust-resource-url.pipe.d.ts +0 -10
- package/lib/emoji/emoji-selector-button.component.d.ts +0 -30
- package/lib/emoji/emoji-selector-panel/emoji-selector-panel.component.d.ts +0 -23
- package/lib/emoji/emoji.module.d.ts +0 -16
- package/lib/emoji/emojis.d.ts +0 -6507
- package/lib/emoji/index.d.ts +0 -4
- package/lib/giphy-attachments.d.ts +0 -5
- package/lib/index.d.ts +0 -19
- package/lib/live-message.component.d.ts +0 -22
- package/lib/message-menu-item.d.ts +0 -6
- package/lib/sdk-options.d.ts +0 -8
- package/lib/static-chat-source.d.ts +0 -49
- package/lib/tweet-attachments.d.ts +0 -5
- package/lib/url-attachments.d.ts +0 -14
- package/lib/youtube-attachments.d.ts +0 -5
- package/public-api.d.ts +0 -1
|
@@ -1,780 +0,0 @@
|
|
|
1
|
-
import { Component, Input, ViewChild, Output, HostBinding, ViewChildren, TemplateRef, ContentChild } from "@angular/core";
|
|
2
|
-
import { CommentsOrder, FilterMode } from '@banta/common';
|
|
3
|
-
import { Subject, Subscription } from 'rxjs';
|
|
4
|
-
import { CommentComponent } from "../comment/comment.component";
|
|
5
|
-
import { BantaInlineRepliesDirective } from "../inline-replies.directive";
|
|
6
|
-
import * as i0 from "@angular/core";
|
|
7
|
-
import * as i1 from "../../chat-backend-base";
|
|
8
|
-
import * as i2 from "@angular/common";
|
|
9
|
-
import * as i3 from "@angular/material/icon";
|
|
10
|
-
import * as i4 from "@angular/material/button";
|
|
11
|
-
import * as i5 from "@angular/material/progress-spinner";
|
|
12
|
-
import * as i6 from "../comment/comment.component";
|
|
13
|
-
const DEFAULT_MAX_MESSAGES = 2000;
|
|
14
|
-
const DEFAULT_MAX_VISIBLE_MESSAGES = 200;
|
|
15
|
-
export class CommentViewComponent {
|
|
16
|
-
constructor(backend, elementRef) {
|
|
17
|
-
this.backend = backend;
|
|
18
|
-
this.elementRef = elementRef;
|
|
19
|
-
//#endregion
|
|
20
|
-
//#region Fields
|
|
21
|
-
this._sourceSubs = new Subscription();
|
|
22
|
-
this.menuMessage = null;
|
|
23
|
-
this.pinnedMessages = [];
|
|
24
|
-
this.messages = [];
|
|
25
|
-
this.customSortEnabled = false;
|
|
26
|
-
this.sourceLoaded = new Promise(r => this.markSourceLoaded = r);
|
|
27
|
-
this.isViewingMore = false;
|
|
28
|
-
this.isLoadingMore = false;
|
|
29
|
-
this.hasMore = false;
|
|
30
|
-
this.messageClicked = false;
|
|
31
|
-
/**
|
|
32
|
-
* While this is called "new" messages, it really represents the messages that would be visible *at the beginning
|
|
33
|
-
* of the sort order*, which can be flipped by the newestLast feature (used for replies mode).
|
|
34
|
-
*
|
|
35
|
-
* So, when newestLast is false (top-level comments), regardless of the current sortOrder, newMessages are conceptually
|
|
36
|
-
* *above* the visible set of messages.
|
|
37
|
-
*
|
|
38
|
-
* When newestLast is true (as in replies mode), regardless of the current sortOrder, newMessages are conceptually
|
|
39
|
-
* *below* the visible set of messages.
|
|
40
|
-
*/
|
|
41
|
-
this.newMessages = [];
|
|
42
|
-
/**
|
|
43
|
-
* While this is called "older" messages, it really represents the messages that would be visible *at the end of the
|
|
44
|
-
* sort order*, which can be flipped by the newestLast feature (useds for replies mode).
|
|
45
|
-
*
|
|
46
|
-
* So, when newestLast is false, regardless of the current sortOrder, olderMessages are conceptually *below*
|
|
47
|
-
* the visible set of messages.
|
|
48
|
-
*
|
|
49
|
-
* When newestLast is true (as in replies mode), regardless of the current sortOrder, olderMessages are conceptually
|
|
50
|
-
* *above* the visible set of messages.
|
|
51
|
-
*/
|
|
52
|
-
this.olderMessages = [];
|
|
53
|
-
this.collapsePins = false;
|
|
54
|
-
this.newestLast = false;
|
|
55
|
-
this.holdNewMessages = false;
|
|
56
|
-
this.showEmptyState = true;
|
|
57
|
-
this.emptyStateMessage = 'Be the first to comment!';
|
|
58
|
-
this.allowReplies = true;
|
|
59
|
-
this.enableHoldOnClick = false;
|
|
60
|
-
this.enableHoldOnScroll = true;
|
|
61
|
-
this.customMenuItems = [];
|
|
62
|
-
//#endregion
|
|
63
|
-
//#region Outputs
|
|
64
|
-
this._selected = new Subject();
|
|
65
|
-
this._liked = new Subject();
|
|
66
|
-
this._unliked = new Subject();
|
|
67
|
-
this._pinned = new Subject();
|
|
68
|
-
this._unpinned = new Subject();
|
|
69
|
-
this._reported = new Subject();
|
|
70
|
-
this._userSelected = new Subject();
|
|
71
|
-
this._usernameSelected = new Subject();
|
|
72
|
-
this._avatarSelected = new Subject();
|
|
73
|
-
this._shared = new Subject();
|
|
74
|
-
this._deleted = new Subject();
|
|
75
|
-
this._messageEdited = new Subject();
|
|
76
|
-
this._sortOrderChanged = new Subject();
|
|
77
|
-
this._filterModeChanged = new Subject();
|
|
78
|
-
this.userSelected = this._userSelected.asObservable();
|
|
79
|
-
this.reported = this._reported.asObservable();
|
|
80
|
-
this.liked = this._liked.asObservable();
|
|
81
|
-
this.unliked = this._unliked.asObservable();
|
|
82
|
-
this.pinned = this._pinned.asObservable();
|
|
83
|
-
this.unpinned = this._unpinned.asObservable();
|
|
84
|
-
this.usernameSelected = this._usernameSelected.asObservable();
|
|
85
|
-
this.avatarSelected = this._avatarSelected.asObservable();
|
|
86
|
-
this.shared = this._shared.asObservable();
|
|
87
|
-
this.deleted = this._deleted.asObservable();
|
|
88
|
-
this.selected = this._selected.asObservable();
|
|
89
|
-
this.messageEdited = this._messageEdited.asObservable();
|
|
90
|
-
this.sortOrderChanged = this._sortOrderChanged.asObservable();
|
|
91
|
-
this.filterModeChanged = this._filterModeChanged.asObservable();
|
|
92
|
-
this.heldMessages = [];
|
|
93
|
-
}
|
|
94
|
-
get source() { return this._source; }
|
|
95
|
-
set source(value) { this.setSource(value); }
|
|
96
|
-
get previousMessages() { return this.newestLast ? this.olderMessages : this.newMessages; }
|
|
97
|
-
get nextMessages() { return this.newestLast ? this.newMessages : this.olderMessages; }
|
|
98
|
-
set maxMessages(value) { this._maxMessages = value; }
|
|
99
|
-
get maxMessages() { return this._maxMessages ?? DEFAULT_MAX_MESSAGES; }
|
|
100
|
-
set maxVisibleMessages(value) { this._maxVisibleMessages = value; }
|
|
101
|
-
get maxVisibleMessages() { return this._maxVisibleMessages ?? DEFAULT_MAX_VISIBLE_MESSAGES; }
|
|
102
|
-
get comments() { return Array.from(this.commentsQuery); }
|
|
103
|
-
//#endregion
|
|
104
|
-
/**
|
|
105
|
-
* Returns true if this message can be found within one of the message buffers (older, current, newer)
|
|
106
|
-
* @param message
|
|
107
|
-
*/
|
|
108
|
-
isMessageLoadedInContext(message) {
|
|
109
|
-
return this.olderMessages.find(x => x.id === message.id)
|
|
110
|
-
|| this.messages.find(x => x.id === message.id)
|
|
111
|
-
|| this.newMessages.find(x => x.id === message.id);
|
|
112
|
-
}
|
|
113
|
-
async loadMessageInContext(message) {
|
|
114
|
-
await this.sourceLoaded;
|
|
115
|
-
console.log(`Loading message ${message.id} in context...`);
|
|
116
|
-
while (this.hasMore && !this.isMessageLoadedInContext(message)) {
|
|
117
|
-
console.log(`...Need to load more comments to find ${message.id}`);
|
|
118
|
-
await this.showMore();
|
|
119
|
-
}
|
|
120
|
-
if (!this.isMessageLoadedInContext(message)) {
|
|
121
|
-
console.error(`Error while loading message in context: Failed to find message ${message.id}, maybe it was deleted!`);
|
|
122
|
-
return false;
|
|
123
|
-
}
|
|
124
|
-
let items = [].concat(this.olderMessages, this.messages, this.newMessages);
|
|
125
|
-
let index = items.findIndex(x => x.id === message.id);
|
|
126
|
-
if (index < 0) {
|
|
127
|
-
console.error(`Error while loading message in context: Message was not present in message list!`);
|
|
128
|
-
return false;
|
|
129
|
-
}
|
|
130
|
-
let startIndex = Math.max(0, index - this.maxVisibleMessages / 2);
|
|
131
|
-
this.newMessages = items.splice(0, startIndex);
|
|
132
|
-
this.messages = items.splice(0, this.maxVisibleMessages);
|
|
133
|
-
this.olderMessages = items;
|
|
134
|
-
this.isViewingMore = true;
|
|
135
|
-
}
|
|
136
|
-
get shouldShowNewMessageIndicator() {
|
|
137
|
-
return this.isViewingMore
|
|
138
|
-
|| this.customSortEnabled
|
|
139
|
-
|| this.sourceFilterMode !== FilterMode.ALL
|
|
140
|
-
|| this.heldMessages.length > 0;
|
|
141
|
-
}
|
|
142
|
-
get shouldHoldNewMessages() {
|
|
143
|
-
if (this.customSortEnabled)
|
|
144
|
-
return true;
|
|
145
|
-
if (this.holdNewMessages || this.isViewingMore) {
|
|
146
|
-
console.log(`holding due to settings`);
|
|
147
|
-
return true;
|
|
148
|
-
}
|
|
149
|
-
if (this.enableHoldOnClick && this.messageClicked)
|
|
150
|
-
return true;
|
|
151
|
-
if (this.enableHoldOnScroll) {
|
|
152
|
-
let keyMessage;
|
|
153
|
-
if (this.newestLast)
|
|
154
|
-
keyMessage = this.messages[this.messages.length - 1];
|
|
155
|
-
else
|
|
156
|
-
keyMessage = this.messages[0];
|
|
157
|
-
if (keyMessage) {
|
|
158
|
-
const messageElement = this.getElementForComment(keyMessage.id);
|
|
159
|
-
if (messageElement) {
|
|
160
|
-
if (!this.isElementVisible(messageElement)) {
|
|
161
|
-
console.log(`key element is not visible`);
|
|
162
|
-
return true;
|
|
163
|
-
}
|
|
164
|
-
else {
|
|
165
|
-
console.log(`key element is visible`);
|
|
166
|
-
}
|
|
167
|
-
}
|
|
168
|
-
else {
|
|
169
|
-
console.log(`could not find key element`);
|
|
170
|
-
}
|
|
171
|
-
}
|
|
172
|
-
else {
|
|
173
|
-
console.log(`could not find key message`);
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
return false;
|
|
177
|
-
}
|
|
178
|
-
isElementVisible(element) {
|
|
179
|
-
const elementRect = element.getBoundingClientRect();
|
|
180
|
-
return !!elementRect
|
|
181
|
-
&& elementRect.bottom >= 0
|
|
182
|
-
&& elementRect.right >= 0
|
|
183
|
-
&& elementRect.left <= document.documentElement.clientWidth
|
|
184
|
-
&& elementRect.top <= document.documentElement.clientHeight;
|
|
185
|
-
}
|
|
186
|
-
/**
|
|
187
|
-
* Get the CommentComponent instantiated for the given ChatMessage,
|
|
188
|
-
* if it exists in the current view. Note that messages which are not
|
|
189
|
-
* currently shown to the user will not return a CommentComponent.
|
|
190
|
-
* @param message
|
|
191
|
-
* @returns
|
|
192
|
-
*/
|
|
193
|
-
getCommentComponentForMessage(message) {
|
|
194
|
-
if (!message)
|
|
195
|
-
throw new Error(`You must pass a valid ChatMessage`);
|
|
196
|
-
return this.comments.find(x => x.message?.id === message.id);
|
|
197
|
-
}
|
|
198
|
-
saveEdit(message, newMessage) {
|
|
199
|
-
this._messageEdited.next({ message, newMessage });
|
|
200
|
-
}
|
|
201
|
-
likeMessage(message) {
|
|
202
|
-
this._liked.next(message);
|
|
203
|
-
}
|
|
204
|
-
unlikeMessage(message) {
|
|
205
|
-
this._unliked.next(message);
|
|
206
|
-
}
|
|
207
|
-
pinMessage(message, options) {
|
|
208
|
-
this._pinned.next({ message, options });
|
|
209
|
-
}
|
|
210
|
-
unpinMessage(message) {
|
|
211
|
-
this._unpinned.next(message);
|
|
212
|
-
}
|
|
213
|
-
reportMessage(message) {
|
|
214
|
-
this._reported.next(message);
|
|
215
|
-
}
|
|
216
|
-
selectMessage(message) {
|
|
217
|
-
this._selected.next(message);
|
|
218
|
-
}
|
|
219
|
-
selectMessageUser(message) {
|
|
220
|
-
this._userSelected.next(message);
|
|
221
|
-
}
|
|
222
|
-
selectUsername(user) {
|
|
223
|
-
this._usernameSelected.next(user);
|
|
224
|
-
}
|
|
225
|
-
selectAvatar(user) {
|
|
226
|
-
this._avatarSelected.next(user);
|
|
227
|
-
}
|
|
228
|
-
sharedMessage(message) {
|
|
229
|
-
this._shared.next(message);
|
|
230
|
-
}
|
|
231
|
-
startEditing(message) {
|
|
232
|
-
this.messages.forEach(m => m.transientState.editing = false);
|
|
233
|
-
message.transientState.editing = true;
|
|
234
|
-
}
|
|
235
|
-
deleteMessage(message) {
|
|
236
|
-
this._deleted.next(message);
|
|
237
|
-
}
|
|
238
|
-
setSource(value) {
|
|
239
|
-
this.customSortEnabled = (value?.sortOrder ?? CommentsOrder.NEWEST) !== CommentsOrder.NEWEST;
|
|
240
|
-
this.newMessages = [];
|
|
241
|
-
this.olderMessages = [];
|
|
242
|
-
this.heldMessages = [];
|
|
243
|
-
window.bantaSourceDebug = value;
|
|
244
|
-
if (this._sourceSubs) {
|
|
245
|
-
this._sourceSubs.unsubscribe();
|
|
246
|
-
this._sourceSubs = null;
|
|
247
|
-
}
|
|
248
|
-
this._source = value;
|
|
249
|
-
if (value) {
|
|
250
|
-
const messages = (value.messages || []).slice();
|
|
251
|
-
this.messages = messages;
|
|
252
|
-
this.olderMessages = messages.splice(this.maxVisibleMessages, messages.length);
|
|
253
|
-
this.hasMore = true; //this.olderMessages.length > 0;
|
|
254
|
-
this._sourceSubs = new Subscription();
|
|
255
|
-
this._sourceSubs.add(this._source.messageReceived.subscribe(msg => this.messageReceived(msg)));
|
|
256
|
-
this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));
|
|
257
|
-
this._sourceSubs.add(this._source.messageUpdated.subscribe(msg => this.messageUpdated(msg)));
|
|
258
|
-
this._sourceSubs.add(this.backend.userChanged.subscribe(user => this.currentUser = user));
|
|
259
|
-
this.getInitialMessages();
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
async getInitialMessages() {
|
|
263
|
-
// Get the pinned messages
|
|
264
|
-
let pinnedMessages = await this._source.getPinnedMessages();
|
|
265
|
-
pinnedMessages.forEach(m => m.transientState ??= {});
|
|
266
|
-
this.pinnedMessages = pinnedMessages;
|
|
267
|
-
let messages = await this._source.getExistingMessages();
|
|
268
|
-
messages.forEach(m => m.transientState ??= {});
|
|
269
|
-
this.messages = this.newestLast ? messages.slice().reverse() : messages;
|
|
270
|
-
if (this.messages.length > this.maxVisibleMessages) {
|
|
271
|
-
if (this.newestLast) {
|
|
272
|
-
this.previousMessages.push(...this.messages.splice(0, this.messages.length - this.maxVisibleMessages));
|
|
273
|
-
}
|
|
274
|
-
else {
|
|
275
|
-
this.nextMessages.unshift(...this.messages.splice(this.maxVisibleMessages, this.messages.length));
|
|
276
|
-
}
|
|
277
|
-
}
|
|
278
|
-
this.debugMessages();
|
|
279
|
-
this.sortMessages();
|
|
280
|
-
if (this.markSourceLoaded)
|
|
281
|
-
this.markSourceLoaded();
|
|
282
|
-
}
|
|
283
|
-
debugMessages() {
|
|
284
|
-
if (!this.showDebug)
|
|
285
|
-
return;
|
|
286
|
-
// console.log([
|
|
287
|
-
// ...this.previousMessages.map(x => x.message),
|
|
288
|
-
// '[[',
|
|
289
|
-
// ...this.messages.map(x => x.message),
|
|
290
|
-
// ']]',
|
|
291
|
-
// ...this.nextMessages.map(x => x.message)
|
|
292
|
-
// ].map(x => /\d+/.test(x) ? this.zeroPad(x, 2) : x).join(" "));
|
|
293
|
-
}
|
|
294
|
-
zeroPad(number, count = 2) {
|
|
295
|
-
let str;
|
|
296
|
-
if (typeof number === 'number')
|
|
297
|
-
str = String(number);
|
|
298
|
-
else
|
|
299
|
-
str = number;
|
|
300
|
-
while (str.length < count)
|
|
301
|
-
str = '0' + str;
|
|
302
|
-
return str;
|
|
303
|
-
}
|
|
304
|
-
leftPad(str, count = 2) {
|
|
305
|
-
while (str.length < count)
|
|
306
|
-
str = ' ' + str;
|
|
307
|
-
return str;
|
|
308
|
-
}
|
|
309
|
-
messageIdentity(index, chatMessage) {
|
|
310
|
-
return chatMessage.id;
|
|
311
|
-
}
|
|
312
|
-
get sourceSortOrder() {
|
|
313
|
-
return this.source?.sortOrder ?? CommentsOrder.NEWEST;
|
|
314
|
-
}
|
|
315
|
-
get sourceFilterMode() {
|
|
316
|
-
return this.source?.filterMode ?? FilterMode.ALL;
|
|
317
|
-
}
|
|
318
|
-
/**
|
|
319
|
-
* Show the newest content.
|
|
320
|
-
* - If an unnatural sort order is active (ie Oldest or Likes), it will be changed to Newest.
|
|
321
|
-
* - The new content will be placed where new content goes based on newestLast (replies mode), so if it is true, the content is
|
|
322
|
-
* placed at the end, otherwise it is placed at the beginning.
|
|
323
|
-
*
|
|
324
|
-
* @param event
|
|
325
|
-
* @returns
|
|
326
|
-
*/
|
|
327
|
-
async showNewest(event) {
|
|
328
|
-
// Regardless of how we handle this, clear out our held messages
|
|
329
|
-
this.heldMessages = [];
|
|
330
|
-
this.messageClicked = false;
|
|
331
|
-
// If the sort order is not already Newest, switch to Newest and stop.
|
|
332
|
-
// The act of changing the sort order will cause the newest content to be loaded.
|
|
333
|
-
let naturalOrder = CommentsOrder.NEWEST;
|
|
334
|
-
if (this.sourceSortOrder !== naturalOrder || this.sourceFilterMode !== FilterMode.ALL) {
|
|
335
|
-
if (this.sourceSortOrder !== naturalOrder)
|
|
336
|
-
this._sortOrderChanged.next(naturalOrder);
|
|
337
|
-
if (this.sourceFilterMode !== FilterMode.ALL)
|
|
338
|
-
this._filterModeChanged.next(FilterMode.ALL);
|
|
339
|
-
return;
|
|
340
|
-
}
|
|
341
|
-
// On this path, we are already on Newest, but there is newer content available (such as when new content is
|
|
342
|
-
// being buffered due to user engagement on a comment)
|
|
343
|
-
this.isViewingMore = false;
|
|
344
|
-
// Move all newerMessages into messages, respecting the newestLast direction (normal or replies mode)
|
|
345
|
-
if (this.newestLast)
|
|
346
|
-
this.messages = this.messages.concat(this.newMessages.splice(0, this.newMessages.length));
|
|
347
|
-
else
|
|
348
|
-
this.messages = this.newMessages.splice(0, this.newMessages.length).concat(this.messages);
|
|
349
|
-
let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);
|
|
350
|
-
this.olderMessages = overflow.concat(this.olderMessages);
|
|
351
|
-
this.olderMessages.splice(this.maxMessages - this.maxVisibleMessages, this.olderMessages.length);
|
|
352
|
-
this.hasMore = this.olderMessages.length > 0;
|
|
353
|
-
// Scroll to the newest comment.
|
|
354
|
-
if (this.messages.length > 0) {
|
|
355
|
-
if (this.newestLast) {
|
|
356
|
-
this.scrollToComment(this.messages[this.messages.length - 1].id);
|
|
357
|
-
}
|
|
358
|
-
else {
|
|
359
|
-
this.scrollToComment(this.messages[0].id);
|
|
360
|
-
}
|
|
361
|
-
}
|
|
362
|
-
}
|
|
363
|
-
get showDebug() {
|
|
364
|
-
if (typeof window === 'undefined')
|
|
365
|
-
return false;
|
|
366
|
-
return localStorage['banta:debug'] === '1';
|
|
367
|
-
}
|
|
368
|
-
get shouldShowNext() {
|
|
369
|
-
if (!this.newestLast) {
|
|
370
|
-
return this.hasMore || this.nextMessages.length > 0;
|
|
371
|
-
}
|
|
372
|
-
return this.nextMessages.length > 0;
|
|
373
|
-
}
|
|
374
|
-
get shouldShowPrevious() {
|
|
375
|
-
if (this.newestLast) {
|
|
376
|
-
return this.hasMore || this.previousMessages.length > 0;
|
|
377
|
-
}
|
|
378
|
-
return this.previousMessages.length > 0;
|
|
379
|
-
}
|
|
380
|
-
get pageSize() {
|
|
381
|
-
return Math.min(20, this.maxVisibleMessages || 20);
|
|
382
|
-
}
|
|
383
|
-
async showPrevious() {
|
|
384
|
-
this.isViewingMore = true;
|
|
385
|
-
let nextPageSize = this.pageSize;
|
|
386
|
-
this.isLoadingMore = false;
|
|
387
|
-
if (isNaN(nextPageSize))
|
|
388
|
-
throw new Error(`Not safe to load more with NaN page size`);
|
|
389
|
-
if (this.previousMessages.length > 0) {
|
|
390
|
-
const storedMessages = this.previousMessages.splice(Math.max(0, this.previousMessages.length - nextPageSize), nextPageSize);
|
|
391
|
-
this.messages = [...storedMessages, ...this.messages];
|
|
392
|
-
nextPageSize -= storedMessages.length;
|
|
393
|
-
}
|
|
394
|
-
// Load more from backend if needed
|
|
395
|
-
// Note: Backend only supports fetching more content in one direction.
|
|
396
|
-
let lastMessage = this.previousMessages[0] ?? this.messages[0];
|
|
397
|
-
if (nextPageSize > 0 && this.newestLast && lastMessage) {
|
|
398
|
-
this.isLoadingMore = true;
|
|
399
|
-
let messages = await this.source.loadAfter(lastMessage, nextPageSize);
|
|
400
|
-
messages = messages.slice().reverse(); // because newestLast === true
|
|
401
|
-
messages.forEach(m => m.transientState ??= {});
|
|
402
|
-
// In replies mode (newestLast), we want to put these new messages onto the *top* of the set of visible messages.
|
|
403
|
-
// Otherwise we want to put them on the *bottom*.
|
|
404
|
-
this.messages = [...messages, ...this.messages];
|
|
405
|
-
// If we didn't receive any messages at all, there's no more to fetch.
|
|
406
|
-
if (messages.length === 0)
|
|
407
|
-
this.hasMore = false;
|
|
408
|
-
this.isLoadingMore = false;
|
|
409
|
-
}
|
|
410
|
-
// Extract the messages that do not fit in the maxVisibleMessages buffer.
|
|
411
|
-
if (this.messages.length > this.maxVisibleMessages) {
|
|
412
|
-
let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);
|
|
413
|
-
this.nextMessages.unshift(...overflow);
|
|
414
|
-
if (this.nextMessages.length > this.maxMessages)
|
|
415
|
-
this.nextMessages.splice(this.maxMessages, this.nextMessages.length);
|
|
416
|
-
}
|
|
417
|
-
this.debugMessages();
|
|
418
|
-
}
|
|
419
|
-
get sortNextLabel() {
|
|
420
|
-
if (this.sourceSortOrder === 'newest') {
|
|
421
|
-
return 'Older';
|
|
422
|
-
}
|
|
423
|
-
else if (this.sourceSortOrder === 'oldest') {
|
|
424
|
-
return 'Newer';
|
|
425
|
-
}
|
|
426
|
-
else if (this.sourceSortOrder === 'likes') {
|
|
427
|
-
return 'Less Likes';
|
|
428
|
-
}
|
|
429
|
-
return 'More';
|
|
430
|
-
}
|
|
431
|
-
get sortPreviousLabel() {
|
|
432
|
-
if (this.sourceSortOrder === 'newest') {
|
|
433
|
-
return 'Newer';
|
|
434
|
-
}
|
|
435
|
-
else if (this.sourceSortOrder === 'oldest') {
|
|
436
|
-
return 'Older';
|
|
437
|
-
}
|
|
438
|
-
else if (this.sourceSortOrder === 'likes') {
|
|
439
|
-
return 'More Likes';
|
|
440
|
-
}
|
|
441
|
-
return 'More';
|
|
442
|
-
}
|
|
443
|
-
get nextLabel() { return this.newestLast ? this.sortPreviousLabel : this.sortNextLabel; }
|
|
444
|
-
get previousLabel() { return this.newestLast ? this.sortNextLabel : this.sortPreviousLabel; }
|
|
445
|
-
/**
|
|
446
|
-
* Show more content
|
|
447
|
-
* - When in replies mode (newestLast), the content is added at the top
|
|
448
|
-
* - When in normal mode, the content is added at the bottom
|
|
449
|
-
* - The current sort order does *not* factor in here, which is why it is showMore() not showEarlier().
|
|
450
|
-
*
|
|
451
|
-
* @returns
|
|
452
|
-
*/
|
|
453
|
-
async showNext() {
|
|
454
|
-
this.isViewingMore = true;
|
|
455
|
-
let nextPageSize = this.pageSize;
|
|
456
|
-
if (isNaN(nextPageSize))
|
|
457
|
-
throw new Error(`Not safe to load more with NaN page size`);
|
|
458
|
-
this.isLoadingMore = false;
|
|
459
|
-
if (this.nextMessages.length > 0) {
|
|
460
|
-
const storedMessages = this.nextMessages.splice(0, nextPageSize);
|
|
461
|
-
this.messages = [...this.messages, ...storedMessages];
|
|
462
|
-
nextPageSize -= storedMessages.length;
|
|
463
|
-
}
|
|
464
|
-
const lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];
|
|
465
|
-
if (nextPageSize > 0 && !this.newestLast && lastMessage) {
|
|
466
|
-
// Load more from backend
|
|
467
|
-
this.isLoadingMore = true;
|
|
468
|
-
let messages = await this.source.loadAfter(lastMessage, nextPageSize);
|
|
469
|
-
messages.forEach(m => m.transientState ??= {});
|
|
470
|
-
this.messages = [...this.messages, ...messages];
|
|
471
|
-
// If we didn't receive any messages at all, there's no more to fetch.
|
|
472
|
-
if (messages.length === 0)
|
|
473
|
-
this.hasMore = false;
|
|
474
|
-
this.isLoadingMore = false;
|
|
475
|
-
}
|
|
476
|
-
// Extract the messages that do not fit in the maxVisibleMessages buffer.
|
|
477
|
-
if (this.messages.length > this.maxVisibleMessages) {
|
|
478
|
-
let overflow = this.messages.splice(0, this.messages.length - this.maxVisibleMessages);
|
|
479
|
-
// Regardless of the order (newestLast), newMessages represents the direction that is being pushed, since it's definition
|
|
480
|
-
// depends on that order. Move overflowing messages into newMessages.
|
|
481
|
-
this.previousMessages.push(...overflow);
|
|
482
|
-
if (this.previousMessages.length > this.maxMessages)
|
|
483
|
-
this.previousMessages.splice(0, this.previousMessages.length - this.maxMessages);
|
|
484
|
-
}
|
|
485
|
-
this.debugMessages();
|
|
486
|
-
}
|
|
487
|
-
/**
|
|
488
|
-
* Show more content
|
|
489
|
-
* - When in replies mode (newestLast), the content is added at the top
|
|
490
|
-
* - When in normal mode, the content is added at the bottom
|
|
491
|
-
* - The current sort order does *not* factor in here, which is why it is showMore() not showEarlier().
|
|
492
|
-
*
|
|
493
|
-
* @returns
|
|
494
|
-
*/
|
|
495
|
-
async showMore() {
|
|
496
|
-
this.isViewingMore = true;
|
|
497
|
-
let nextPageSize = this.pageSize;
|
|
498
|
-
if (isNaN(nextPageSize))
|
|
499
|
-
throw new Error(`Not safe to load more with NaN page size`);
|
|
500
|
-
this.isLoadingMore = false;
|
|
501
|
-
if (this.olderMessages.length > 0) {
|
|
502
|
-
const storedMessages = this.olderMessages.splice(0, nextPageSize);
|
|
503
|
-
this.messages = this.messages.concat(storedMessages);
|
|
504
|
-
nextPageSize -= storedMessages.length;
|
|
505
|
-
this.hasMore = this.olderMessages.length > 0;
|
|
506
|
-
}
|
|
507
|
-
let lastMessage;
|
|
508
|
-
if (this.newestLast) {
|
|
509
|
-
lastMessage = this.olderMessages[0] ?? this.messages[0];
|
|
510
|
-
}
|
|
511
|
-
else {
|
|
512
|
-
lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];
|
|
513
|
-
}
|
|
514
|
-
if (nextPageSize > 0 && lastMessage) {
|
|
515
|
-
// Load more from backend
|
|
516
|
-
this.isLoadingMore = true;
|
|
517
|
-
if (!lastMessage) {
|
|
518
|
-
this.isLoadingMore = false;
|
|
519
|
-
this.hasMore = false;
|
|
520
|
-
return;
|
|
521
|
-
}
|
|
522
|
-
let messages = await this.source.loadAfter(lastMessage, nextPageSize);
|
|
523
|
-
if (this.newestLast)
|
|
524
|
-
messages = messages.slice().reverse();
|
|
525
|
-
messages.forEach(m => m.transientState ??= {});
|
|
526
|
-
// In replies mode (newestLast), we want to put these new messages onto the *top* of the set of visible messages.
|
|
527
|
-
// Otherwise we want to put them on the *bottom*.
|
|
528
|
-
if (this.newestLast) {
|
|
529
|
-
this.messages = messages.concat(this.messages);
|
|
530
|
-
}
|
|
531
|
-
else {
|
|
532
|
-
this.messages = this.messages.concat(messages);
|
|
533
|
-
}
|
|
534
|
-
// If we didn't receive any messages at all, there's no more to fetch.
|
|
535
|
-
if (messages.length === 0) {
|
|
536
|
-
this.hasMore = false;
|
|
537
|
-
}
|
|
538
|
-
this.isLoadingMore = false;
|
|
539
|
-
}
|
|
540
|
-
// Extract the messages that do not fit in the maxVisibleMessages buffer.
|
|
541
|
-
if (this.messages.length > this.maxVisibleMessages) {
|
|
542
|
-
let overflow = [];
|
|
543
|
-
// Move overflowing messages into newMessages.
|
|
544
|
-
// Regardless of the order (newestLast), newMessages represents the direction that is being loaded,
|
|
545
|
-
// since it's definition depends on that order.
|
|
546
|
-
if (this.newestLast) {
|
|
547
|
-
overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);
|
|
548
|
-
this.newMessages = overflow.concat(this.newMessages);
|
|
549
|
-
this.newMessages.splice(this.maxMessages - this.maxVisibleMessages, this.newMessages.length);
|
|
550
|
-
}
|
|
551
|
-
else {
|
|
552
|
-
overflow = this.messages.splice(0, this.messages.length - this.maxVisibleMessages);
|
|
553
|
-
this.newMessages.push(...overflow);
|
|
554
|
-
if (this.newMessages.length > this.maxMessages - this.maxVisibleMessages)
|
|
555
|
-
this.newMessages.splice(0, this.newMessages.length - (this.maxMessages - this.maxVisibleMessages));
|
|
556
|
-
}
|
|
557
|
-
}
|
|
558
|
-
}
|
|
559
|
-
addMessage(message) {
|
|
560
|
-
if (!message.transientState)
|
|
561
|
-
message.transientState ??= {};
|
|
562
|
-
let destination = this.messages;
|
|
563
|
-
let bucket = this.olderMessages;
|
|
564
|
-
let newestLast = this.newestLast;
|
|
565
|
-
if (this.shouldHoldNewMessages) {
|
|
566
|
-
this.heldMessages.push(message);
|
|
567
|
-
destination = this.newMessages;
|
|
568
|
-
bucket = null;
|
|
569
|
-
}
|
|
570
|
-
// If we aren't on the newest sort order, new messages shouldn't be added at all
|
|
571
|
-
if (this.sourceSortOrder !== CommentsOrder.NEWEST)
|
|
572
|
-
return;
|
|
573
|
-
if (newestLast) {
|
|
574
|
-
destination.push(message);
|
|
575
|
-
if (this.maxVisibleMessages > 0) {
|
|
576
|
-
let overflow = destination.splice(this.maxVisibleMessages, destination.length);
|
|
577
|
-
bucket?.push(...overflow);
|
|
578
|
-
}
|
|
579
|
-
}
|
|
580
|
-
else {
|
|
581
|
-
destination.unshift(message);
|
|
582
|
-
if (this.maxVisibleMessages > 0) {
|
|
583
|
-
let overflow = destination.splice(this.maxVisibleMessages, destination.length);
|
|
584
|
-
bucket?.unshift(...overflow);
|
|
585
|
-
}
|
|
586
|
-
}
|
|
587
|
-
if (bucket?.length > 0)
|
|
588
|
-
this.hasMore = true;
|
|
589
|
-
message.pagingCursor = String(this.incrementPagingCursors());
|
|
590
|
-
this.sortMessages();
|
|
591
|
-
}
|
|
592
|
-
incrementPagingCursors() {
|
|
593
|
-
if (this.source.sortOrder !== CommentsOrder.NEWEST)
|
|
594
|
-
return;
|
|
595
|
-
let maxPagingCursor = 0;
|
|
596
|
-
for (let group of [this.messages, this.olderMessages, this.newMessages]) {
|
|
597
|
-
for (let message of group) {
|
|
598
|
-
if (message.pagingCursor) {
|
|
599
|
-
let pagingCursor = Number(message.pagingCursor) + 1;
|
|
600
|
-
if (pagingCursor > maxPagingCursor)
|
|
601
|
-
maxPagingCursor = pagingCursor;
|
|
602
|
-
message.pagingCursor = String(pagingCursor);
|
|
603
|
-
}
|
|
604
|
-
}
|
|
605
|
-
}
|
|
606
|
-
return maxPagingCursor;
|
|
607
|
-
}
|
|
608
|
-
/**
|
|
609
|
-
* Wait for all currently visible comments to be fully loaded, including all attachments.
|
|
610
|
-
* Doing this will prevent layout shift when scrolling to a specific comment.
|
|
611
|
-
*/
|
|
612
|
-
async waitForAllCommentsToLoad() {
|
|
613
|
-
await new Promise(r => setTimeout(r, 100));
|
|
614
|
-
await this.sourceLoaded;
|
|
615
|
-
await Promise.all(this.comments.map(x => x.waitForLoad()));
|
|
616
|
-
}
|
|
617
|
-
sortMessages() {
|
|
618
|
-
if (!this.source)
|
|
619
|
-
return;
|
|
620
|
-
let sorter;
|
|
621
|
-
if (this.source.sortOrder === CommentsOrder.LIKES)
|
|
622
|
-
sorter = (a, b) => b.likes - a.likes;
|
|
623
|
-
else if (this.source.sortOrder === CommentsOrder.NEWEST)
|
|
624
|
-
sorter = (a, b) => (b.sentAt - a.sentAt) * (this.newestLast ? -1 : 1);
|
|
625
|
-
else if (this.source.sortOrder === CommentsOrder.OLDEST)
|
|
626
|
-
sorter = (a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? -1 : 1);
|
|
627
|
-
this.messages.sort(sorter);
|
|
628
|
-
this.olderMessages.sort(sorter);
|
|
629
|
-
this.newMessages.sort(sorter);
|
|
630
|
-
}
|
|
631
|
-
messageReceived(message) {
|
|
632
|
-
this.addMessage(message);
|
|
633
|
-
}
|
|
634
|
-
isScrolledToLatest() {
|
|
635
|
-
if (!this.messageContainer)
|
|
636
|
-
return false;
|
|
637
|
-
const el = this.messageContainer.nativeElement;
|
|
638
|
-
const currentScroll = el.scrollTop;
|
|
639
|
-
const currentTotal = el.scrollHeight - el.offsetHeight;
|
|
640
|
-
return currentScroll > currentTotal - 10;
|
|
641
|
-
}
|
|
642
|
-
messageSent(message) {
|
|
643
|
-
this.addMessage(message);
|
|
644
|
-
if (!this.messageContainer)
|
|
645
|
-
return;
|
|
646
|
-
this.scrollToLatest();
|
|
647
|
-
}
|
|
648
|
-
isPinned(message) {
|
|
649
|
-
return message.pinned && (!message.pinnedUntil || message.pinnedUntil > Date.now());
|
|
650
|
-
}
|
|
651
|
-
messageUpdated(message) {
|
|
652
|
-
let pinned = this.isPinned(message);
|
|
653
|
-
let inPins = this.pinnedMessages.some(x => x.id === message.id);
|
|
654
|
-
if (pinned && !inPins) {
|
|
655
|
-
this.pinnedMessages.unshift(message);
|
|
656
|
-
let index = this.messages.indexOf(message);
|
|
657
|
-
if (index >= 0) {
|
|
658
|
-
this.messages.splice(index, 1);
|
|
659
|
-
}
|
|
660
|
-
}
|
|
661
|
-
else if (!pinned && inPins) {
|
|
662
|
-
this.messages.push(message);
|
|
663
|
-
let index = this.pinnedMessages.indexOf(message);
|
|
664
|
-
if (index >= 0) {
|
|
665
|
-
this.pinnedMessages.splice(index, 1);
|
|
666
|
-
}
|
|
667
|
-
this.sortMessages();
|
|
668
|
-
}
|
|
669
|
-
}
|
|
670
|
-
scrollToLatest() {
|
|
671
|
-
if (!this.messageContainer) {
|
|
672
|
-
return;
|
|
673
|
-
}
|
|
674
|
-
const el = this.messageContainer.nativeElement;
|
|
675
|
-
el.scrollIntoView({ block: 'start' });
|
|
676
|
-
//el.scrollTop = el.scrollHeight;
|
|
677
|
-
}
|
|
678
|
-
get element() {
|
|
679
|
-
return this.elementRef.nativeElement;
|
|
680
|
-
}
|
|
681
|
-
async scrollToComment(commentId) {
|
|
682
|
-
if (typeof window === 'undefined')
|
|
683
|
-
return;
|
|
684
|
-
await this.waitForAllCommentsToLoad();
|
|
685
|
-
const comment = this.getElementForComment(commentId);
|
|
686
|
-
if (comment) {
|
|
687
|
-
comment.scrollIntoView({
|
|
688
|
-
inline: 'center',
|
|
689
|
-
block: 'center'
|
|
690
|
-
});
|
|
691
|
-
}
|
|
692
|
-
}
|
|
693
|
-
getElementForComment(commentId) {
|
|
694
|
-
return this.element.querySelector(`[data-comment-id="${commentId}"]`);
|
|
695
|
-
}
|
|
696
|
-
mentionsMe(message) {
|
|
697
|
-
if (!this.currentUser)
|
|
698
|
-
return false;
|
|
699
|
-
if (message.message.includes(`@${this.currentUser.username}`))
|
|
700
|
-
return true;
|
|
701
|
-
return false;
|
|
702
|
-
}
|
|
703
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.9", ngImport: i0, type: CommentViewComponent, deps: [{ token: i1.ChatBackendBase }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
704
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.9", type: CommentViewComponent, selector: "banta-comment-view", inputs: { source: "source", maxMessages: "maxMessages", maxVisibleMessages: "maxVisibleMessages", collapsePins: "collapsePins", newestLast: "newestLast", holdNewMessages: "holdNewMessages", showEmptyState: "showEmptyState", emptyStateMessage: "emptyStateMessage", allowReplies: "allowReplies", enableHoldOnClick: "enableHoldOnClick", enableHoldOnScroll: "enableHoldOnScroll", customMenuItems: "customMenuItems", fixedHeight: "fixedHeight", selectedMessage: "selectedMessage", genericAvatarUrl: "genericAvatarUrl" }, outputs: { userSelected: "userSelected", reported: "reported", liked: "liked", unliked: "unliked", pinned: "pinned", unpinned: "unpinned", usernameSelected: "usernameSelected", avatarSelected: "avatarSelected", shared: "shared", deleted: "deleted", selected: "selected", messageEdited: "messageEdited", sortOrderChanged: "sortOrderChanged", filterModeChanged: "filterModeChanged" }, host: { properties: { "class.fixed-height": "this.fixedHeight" } }, queries: [{ propertyName: "inlineRepliesTemplate", first: true, predicate: BantaInlineRepliesDirective, descendants: true, read: TemplateRef }], viewQueries: [{ propertyName: "messageContainer", first: true, predicate: ["messageContainer"], descendants: true }, { propertyName: "commentsQuery", predicate: CommentComponent, descendants: true }], ngImport: i0, template: "<div class=\"banta-message-container\" #messageContainer>\r\n <ng-content select=\"[data-before]\"></ng-content>\r\n\r\n @if (!collapsePins) {\r\n @for (message of pinnedMessages; track message.id) {\r\n @if (!message.hidden) {\r\n <banta-comment\r\n class=\"abbreviated\"\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 [readonly]=\"source?.readonly\"\r\n (click)=\"messageClicked = 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 (pinned)=\"pinMessage(message, $event.options)\"\r\n (unpinned)=\"unpinMessage(message)\"\r\n (unliked)=\"unlikeMessage(message)\"\r\n (reported)=\"reportMessage(message)\"\r\n (selected)=\"selectMessage(message)\"\r\n (shared)=\"sharedMessage($event)\"\r\n />\r\n <div class=\"banta-inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n <ng-container *ngTemplateOutlet=\"inlineRepliesTemplate\" />\r\n </div>\r\n }\r\n }\r\n }\r\n\r\n <div class=\"banta-top-sticky\">\r\n @if (!newestLast) {\r\n <button\r\n mat-button\r\n class=\"banta-nav\"\r\n [class.visible]=\"shouldShowNewMessageIndicator\"\r\n href=\"javascript:;\"\r\n (click)=\"showNewest($event)\"\r\n >\r\n <mat-icon>file_upload</mat-icon>\r\n Newest\r\n @if (heldMessages.length > 0) {\r\n <span class=\"count\">{{ heldMessages.length | number }}</span>\r\n }\r\n </button>\r\n }\r\n </div>\r\n\r\n <button mat-button class=\"pager\" (click)=\"showPrevious()\" [class.visible]=\"shouldShowPrevious\" [disabled]=\"isLoadingMore\">\r\n <mat-icon>expand_less</mat-icon>\r\n {{ previousLabel }}\r\n </button>\r\n\r\n @for (message of messages; track message.id) {\r\n @if (!message.hidden) {\r\n <banta-comment\r\n class=\"abbreviated\"\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 [readonly]=\"source?.readonly\"\r\n (click)=\"messageClicked = 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 (pinned)=\"pinMessage(message, $event.options)\"\r\n (unpinned)=\"unpinMessage(message)\"\r\n (unliked)=\"unlikeMessage(message)\"\r\n (reported)=\"reportMessage(message)\"\r\n (selected)=\"selectMessage(message)\"\r\n (shared)=\"sharedMessage($event)\"\r\n />\r\n <div class=\"banta-inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n <ng-container *ngTemplateOutlet=\"inlineRepliesTemplate\" />\r\n </div>\r\n }\r\n } @empty {\r\n <div class=\"banta-empty-state\" *ngIf=\"showEmptyState\">\r\n {{ emptyStateMessage }}\r\n </div>\r\n }\r\n\r\n <button mat-button class=\"pager\" (click)=\"showNext()\" [class.visible]=\"shouldShowNext\" [disabled]=\"isLoadingMore\">\r\n <mat-icon>expand_more</mat-icon>\r\n {{ nextLabel }}\r\n </button>\r\n\r\n <div class=\"banta-nav-point banta-bottom-sticky\">\r\n @if (newestLast) {\r\n <button\r\n [matBadge]=\"10\" matBadgeOverlap=\"false\"\r\n matBadgePosition=\"after\" matBadgeSize=\"large\"\r\n mat-button\r\n class=\"banta-nav\"\r\n [class.visible]=\"shouldShowNewMessageIndicator\"\r\n href=\"javascript:;\"\r\n (click)=\"showNewest($event)\"\r\n >\r\n <mat-icon>file_download</mat-icon>\r\n Newest\r\n @if (heldMessages.length > 0) {\r\n <span class=\"count\">{{ heldMessages.length | number }}</span>\r\n }\r\n </button>\r\n }\r\n </div>\r\n\r\n <div class=\"banta-loading-more\" *ngIf=\"isLoadingMore\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n\r\n @if (showDebug) {\r\n <div style=\"color: #666\">\r\n ({{ previousMessages.length }} .. {{ messages.length }} .. {{ nextMessages.length }})\r\n\r\n dir={{newestLast ? '-1' : '1'}}\r\n v={{maxVisibleMessages}}, M={{maxMessages}}\r\n </div>\r\n }\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}.banta-message-container{flex-grow:1;color:#111;background:#fff;padding:.5em 1em 3em .5em;opacity:1;transition:.5s opacity ease-in-out;position:relative}.banta-message-container.no-scroll{height:auto;overflow-y:visible}.banta-message-container.faded{opacity:.25}.banta-message-container .overlay{position:absolute;inset:0;z-index:10}:host.fixed-height .banta-message-container{overflow-y:auto}:host-context(.mat-dark-theme) .banta-message-container{color:#fff;background:#111}::ng-deep .banta-empty-state{text-align:center;margin:3em;color:#666}:host-context(.mat-dark-theme) .empty-state{color:#666}button.banta-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-color:#ddd}:host-context(.mat-dark-theme) button.banta-nav{background-color:#222;color:#fff}button.banta-nav span.count{background-color:#a93535;color:#fff;padding:4px 10px;border-radius:.5em;margin-left:.25em;font-size:90%}button.banta-nav.visible{opacity:1;pointer-events:initial}button.pager{appearance:none;border:none;width:100%;opacity:0;pointer-events:none;transition:.4s opacity ease-in-out}button.pager.visible{opacity:1;pointer-events:initial}.banta-top-sticky{position:sticky;top:.5em;z-index:10}.banta-bottom-sticky{position:sticky;bottom:3em;z-index:10}.banta-loading-more{padding:2em;text-align:center;margin:0 auto;width:fit-content}@media (max-width: 400px){.banta-message-container{padding:0 0 3em}}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i5.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: i6.CommentComponent, selector: "banta-comment", inputs: ["message", "customMenuItems", "showReplyAction", "maxLength", "permissions", "mine", "editing", "genericAvatarUrl", "readonly"], outputs: ["liked", "unliked", "selected", "edited", "deleted", "editStarted", "editEnded", "shared", "userSelected", "usernameSelected", "avatarSelected", "reported", "loaded", "pinned", "unpinned"] }, { kind: "pipe", type: i2.DecimalPipe, name: "number" }] }); }
|
|
705
|
-
}
|
|
706
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImport: i0, type: CommentViewComponent, decorators: [{
|
|
707
|
-
type: Component,
|
|
708
|
-
args: [{ selector: 'banta-comment-view', template: "<div class=\"banta-message-container\" #messageContainer>\r\n <ng-content select=\"[data-before]\"></ng-content>\r\n\r\n @if (!collapsePins) {\r\n @for (message of pinnedMessages; track message.id) {\r\n @if (!message.hidden) {\r\n <banta-comment\r\n class=\"abbreviated\"\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 [readonly]=\"source?.readonly\"\r\n (click)=\"messageClicked = 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 (pinned)=\"pinMessage(message, $event.options)\"\r\n (unpinned)=\"unpinMessage(message)\"\r\n (unliked)=\"unlikeMessage(message)\"\r\n (reported)=\"reportMessage(message)\"\r\n (selected)=\"selectMessage(message)\"\r\n (shared)=\"sharedMessage($event)\"\r\n />\r\n <div class=\"banta-inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n <ng-container *ngTemplateOutlet=\"inlineRepliesTemplate\" />\r\n </div>\r\n }\r\n }\r\n }\r\n\r\n <div class=\"banta-top-sticky\">\r\n @if (!newestLast) {\r\n <button\r\n mat-button\r\n class=\"banta-nav\"\r\n [class.visible]=\"shouldShowNewMessageIndicator\"\r\n href=\"javascript:;\"\r\n (click)=\"showNewest($event)\"\r\n >\r\n <mat-icon>file_upload</mat-icon>\r\n Newest\r\n @if (heldMessages.length > 0) {\r\n <span class=\"count\">{{ heldMessages.length | number }}</span>\r\n }\r\n </button>\r\n }\r\n </div>\r\n\r\n <button mat-button class=\"pager\" (click)=\"showPrevious()\" [class.visible]=\"shouldShowPrevious\" [disabled]=\"isLoadingMore\">\r\n <mat-icon>expand_less</mat-icon>\r\n {{ previousLabel }}\r\n </button>\r\n\r\n @for (message of messages; track message.id) {\r\n @if (!message.hidden) {\r\n <banta-comment\r\n class=\"abbreviated\"\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 [readonly]=\"source?.readonly\"\r\n (click)=\"messageClicked = 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 (pinned)=\"pinMessage(message, $event.options)\"\r\n (unpinned)=\"unpinMessage(message)\"\r\n (unliked)=\"unlikeMessage(message)\"\r\n (reported)=\"reportMessage(message)\"\r\n (selected)=\"selectMessage(message)\"\r\n (shared)=\"sharedMessage($event)\"\r\n />\r\n <div class=\"banta-inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n <ng-container *ngTemplateOutlet=\"inlineRepliesTemplate\" />\r\n </div>\r\n }\r\n } @empty {\r\n <div class=\"banta-empty-state\" *ngIf=\"showEmptyState\">\r\n {{ emptyStateMessage }}\r\n </div>\r\n }\r\n\r\n <button mat-button class=\"pager\" (click)=\"showNext()\" [class.visible]=\"shouldShowNext\" [disabled]=\"isLoadingMore\">\r\n <mat-icon>expand_more</mat-icon>\r\n {{ nextLabel }}\r\n </button>\r\n\r\n <div class=\"banta-nav-point banta-bottom-sticky\">\r\n @if (newestLast) {\r\n <button\r\n [matBadge]=\"10\" matBadgeOverlap=\"false\"\r\n matBadgePosition=\"after\" matBadgeSize=\"large\"\r\n mat-button\r\n class=\"banta-nav\"\r\n [class.visible]=\"shouldShowNewMessageIndicator\"\r\n href=\"javascript:;\"\r\n (click)=\"showNewest($event)\"\r\n >\r\n <mat-icon>file_download</mat-icon>\r\n Newest\r\n @if (heldMessages.length > 0) {\r\n <span class=\"count\">{{ heldMessages.length | number }}</span>\r\n }\r\n </button>\r\n }\r\n </div>\r\n\r\n <div class=\"banta-loading-more\" *ngIf=\"isLoadingMore\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n\r\n @if (showDebug) {\r\n <div style=\"color: #666\">\r\n ({{ previousMessages.length }} .. {{ messages.length }} .. {{ nextMessages.length }})\r\n\r\n dir={{newestLast ? '-1' : '1'}}\r\n v={{maxVisibleMessages}}, M={{maxMessages}}\r\n </div>\r\n }\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}.banta-message-container{flex-grow:1;color:#111;background:#fff;padding:.5em 1em 3em .5em;opacity:1;transition:.5s opacity ease-in-out;position:relative}.banta-message-container.no-scroll{height:auto;overflow-y:visible}.banta-message-container.faded{opacity:.25}.banta-message-container .overlay{position:absolute;inset:0;z-index:10}:host.fixed-height .banta-message-container{overflow-y:auto}:host-context(.mat-dark-theme) .banta-message-container{color:#fff;background:#111}::ng-deep .banta-empty-state{text-align:center;margin:3em;color:#666}:host-context(.mat-dark-theme) .empty-state{color:#666}button.banta-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-color:#ddd}:host-context(.mat-dark-theme) button.banta-nav{background-color:#222;color:#fff}button.banta-nav span.count{background-color:#a93535;color:#fff;padding:4px 10px;border-radius:.5em;margin-left:.25em;font-size:90%}button.banta-nav.visible{opacity:1;pointer-events:initial}button.pager{appearance:none;border:none;width:100%;opacity:0;pointer-events:none;transition:.4s opacity ease-in-out}button.pager.visible{opacity:1;pointer-events:initial}.banta-top-sticky{position:sticky;top:.5em;z-index:10}.banta-bottom-sticky{position:sticky;bottom:3em;z-index:10}.banta-loading-more{padding:2em;text-align:center;margin:0 auto;width:fit-content}@media (max-width: 400px){.banta-message-container{padding:0 0 3em}}\n"] }]
|
|
709
|
-
}], ctorParameters: () => [{ type: i1.ChatBackendBase }, { type: i0.ElementRef }], propDecorators: { source: [{
|
|
710
|
-
type: Input
|
|
711
|
-
}], maxMessages: [{
|
|
712
|
-
type: Input
|
|
713
|
-
}], maxVisibleMessages: [{
|
|
714
|
-
type: Input
|
|
715
|
-
}], collapsePins: [{
|
|
716
|
-
type: Input
|
|
717
|
-
}], newestLast: [{
|
|
718
|
-
type: Input
|
|
719
|
-
}], holdNewMessages: [{
|
|
720
|
-
type: Input
|
|
721
|
-
}], showEmptyState: [{
|
|
722
|
-
type: Input
|
|
723
|
-
}], emptyStateMessage: [{
|
|
724
|
-
type: Input
|
|
725
|
-
}], allowReplies: [{
|
|
726
|
-
type: Input
|
|
727
|
-
}], enableHoldOnClick: [{
|
|
728
|
-
type: Input
|
|
729
|
-
}], enableHoldOnScroll: [{
|
|
730
|
-
type: Input
|
|
731
|
-
}], customMenuItems: [{
|
|
732
|
-
type: Input
|
|
733
|
-
}], fixedHeight: [{
|
|
734
|
-
type: Input
|
|
735
|
-
}, {
|
|
736
|
-
type: HostBinding,
|
|
737
|
-
args: ['class.fixed-height']
|
|
738
|
-
}], selectedMessage: [{
|
|
739
|
-
type: Input
|
|
740
|
-
}], genericAvatarUrl: [{
|
|
741
|
-
type: Input
|
|
742
|
-
}], inlineRepliesTemplate: [{
|
|
743
|
-
type: ContentChild,
|
|
744
|
-
args: [BantaInlineRepliesDirective, { read: TemplateRef }]
|
|
745
|
-
}], userSelected: [{
|
|
746
|
-
type: Output
|
|
747
|
-
}], reported: [{
|
|
748
|
-
type: Output
|
|
749
|
-
}], liked: [{
|
|
750
|
-
type: Output
|
|
751
|
-
}], unliked: [{
|
|
752
|
-
type: Output
|
|
753
|
-
}], pinned: [{
|
|
754
|
-
type: Output
|
|
755
|
-
}], unpinned: [{
|
|
756
|
-
type: Output
|
|
757
|
-
}], usernameSelected: [{
|
|
758
|
-
type: Output
|
|
759
|
-
}], avatarSelected: [{
|
|
760
|
-
type: Output
|
|
761
|
-
}], shared: [{
|
|
762
|
-
type: Output
|
|
763
|
-
}], deleted: [{
|
|
764
|
-
type: Output
|
|
765
|
-
}], selected: [{
|
|
766
|
-
type: Output
|
|
767
|
-
}], messageEdited: [{
|
|
768
|
-
type: Output
|
|
769
|
-
}], sortOrderChanged: [{
|
|
770
|
-
type: Output
|
|
771
|
-
}], filterModeChanged: [{
|
|
772
|
-
type: Output
|
|
773
|
-
}], commentsQuery: [{
|
|
774
|
-
type: ViewChildren,
|
|
775
|
-
args: [CommentComponent]
|
|
776
|
-
}], messageContainer: [{
|
|
777
|
-
type: ViewChild,
|
|
778
|
-
args: ['messageContainer']
|
|
779
|
-
}] } });
|
|
780
|
-
//# 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, TemplateRef, ContentChild } from \"@angular/core\";\r\nimport { User, ChatMessage, CommentsOrder, FilterMode } 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\nimport { PinOptions } from \"../../chat-source\";\r\nimport { BantaInlineRepliesDirective } from \"../inline-replies.directive\";\r\n\r\nexport interface EditEvent {\r\n    message: ChatMessage;\r\n    newMessage: string;\r\n}\r\n\r\nconst DEFAULT_MAX_MESSAGES = 2000;\r\nconst DEFAULT_MAX_VISIBLE_MESSAGES = 200;\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        private elementRef: ElementRef<HTMLElement>\r\n    ) {\r\n    }\r\n\r\n    //#region Source\r\n\r\n    private _source: ChatSourceBase;\r\n    @Input() get source() { return this._source; }\r\n    set source(value) { this.setSource(value); }\r\n\r\n    //#endregion\r\n    //#region Fields\r\n\r\n    private _sourceSubs = new Subscription();\r\n    menuMessage: ChatMessage = null;\r\n    pinnedMessages: ChatMessage[] = [];\r\n    messages: ChatMessage[] = [];\r\n    currentUser: User;\r\n    customSortEnabled = false;\r\n    markSourceLoaded: () => void;\r\n    sourceLoaded = new Promise<void>(r => this.markSourceLoaded = r);\r\n    isViewingMore = false;\r\n    isLoadingMore = false;\r\n    hasMore = false;\r\n    messageClicked = false;\r\n\r\n    get previousMessages() { return this.newestLast ? this.olderMessages : this.newMessages; }\r\n    get nextMessages() { return this.newestLast ? this.newMessages : this.olderMessages; }\r\n\r\n    /**\r\n     * While this is called \"new\" messages, it really represents the messages that would be visible *at the beginning\r\n     * of the sort order*, which can be flipped by the newestLast feature (used for replies mode).\r\n     *\r\n     * So, when newestLast is false (top-level comments), regardless of the current sortOrder, newMessages are conceptually\r\n     * *above* the visible set of messages.\r\n     *\r\n     * When newestLast is true (as in replies mode), regardless of the current sortOrder, newMessages are conceptually\r\n     * *below* the visible set of messages.\r\n     */\r\n    newMessages: ChatMessage[] = [];\r\n\r\n    /**\r\n     * While this is called \"older\" messages, it really represents the messages that would be visible *at the end of the\r\n     * sort order*, which can be flipped by the newestLast feature (useds for replies mode).\r\n     *\r\n     * So, when newestLast is false, regardless of the current sortOrder, olderMessages are conceptually *below*\r\n     * the visible set of messages.\r\n     *\r\n     * When newestLast is true (as in replies mode), regardless of the current sortOrder, olderMessages are conceptually\r\n     * *above* the visible set of messages.\r\n     */\r\n    olderMessages: ChatMessage[] = [];\r\n\r\n    //#endregion\r\n    //#region Inputs\r\n\r\n    private _maxMessages: number;\r\n    @Input()\r\n    set maxMessages(value) { this._maxMessages = value; }\r\n    get maxMessages() { return this._maxMessages ?? DEFAULT_MAX_MESSAGES; }\r\n\r\n    private _maxVisibleMessages: number;\r\n    @Input()\r\n    set maxVisibleMessages(value) { this._maxVisibleMessages = value; }\r\n    get maxVisibleMessages() { return this._maxVisibleMessages ?? DEFAULT_MAX_VISIBLE_MESSAGES; }\r\n\r\n    @Input() collapsePins = false;\r\n    @Input() newestLast = false;\r\n    @Input() holdNewMessages = false;\r\n    @Input() showEmptyState = true;\r\n    @Input() emptyStateMessage = 'Be the first to comment!';\r\n    @Input() allowReplies = true;\r\n    @Input() enableHoldOnClick = false;\r\n    @Input() enableHoldOnScroll = true;\r\n    @Input() customMenuItems: MessageMenuItem[] = [];\r\n    @Input() @HostBinding('class.fixed-height') fixedHeight: boolean;\r\n    @Input() selectedMessage: ChatMessage;\r\n    @Input() genericAvatarUrl: string;\r\n\r\n    @ContentChild(BantaInlineRepliesDirective, { read: TemplateRef })\r\n    inlineRepliesTemplate: TemplateRef<any>;\r\n\r\n    //#endregion\r\n    //#region Outputs\r\n\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _liked = new Subject<ChatMessage>();\r\n    private _unliked = new Subject<ChatMessage>();\r\n    private _pinned = new Subject<{ message: ChatMessage, options: PinOptions }>();\r\n    private _unpinned = 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    private _sortOrderChanged = new Subject<CommentsOrder>();\r\n    private _filterModeChanged = new Subject<FilterMode>();\r\n\r\n    @Output() readonly userSelected = this._userSelected.asObservable();\r\n    @Output() readonly reported = this._reported.asObservable();\r\n    @Output() readonly liked = this._liked.asObservable();\r\n    @Output() readonly unliked = this._unliked.asObservable();\r\n    @Output() readonly pinned = this._pinned.asObservable();\r\n    @Output() readonly unpinned = this._unpinned.asObservable();\r\n    @Output() readonly usernameSelected = this._usernameSelected.asObservable();\r\n    @Output() readonly avatarSelected = this._avatarSelected.asObservable();\r\n    @Output() readonly shared = this._shared.asObservable();\r\n    @Output() readonly deleted = this._deleted.asObservable();\r\n    @Output() readonly selected = this._selected.asObservable();\r\n    @Output() readonly messageEdited = this._messageEdited.asObservable();\r\n    @Output() readonly sortOrderChanged = this._sortOrderChanged.asObservable();\r\n    @Output() readonly filterModeChanged = this._filterModeChanged.asObservable();\r\n\r\n    //#endregion\r\n    //#region UI Bindings\r\n\r\n    @ViewChildren(CommentComponent) commentsQuery: QueryList<CommentComponent>;\r\n    @ViewChild('messageContainer') messageContainer: ElementRef<HTMLElement>;\r\n    get comments() { return Array.from(this.commentsQuery); }\r\n\r\n    //#endregion\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 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 - this.maxVisibleMessages / 2);\r\n        this.newMessages = items.splice(0, startIndex);\r\n        this.messages = items.splice(0, this.maxVisibleMessages);\r\n        this.olderMessages = items;\r\n        this.isViewingMore = true;\r\n    }\r\n\r\n    get shouldShowNewMessageIndicator() {\r\n        return this.isViewingMore\r\n            || this.customSortEnabled\r\n            || this.sourceFilterMode !== FilterMode.ALL\r\n            || this.heldMessages.length > 0;\r\n    }\r\n\r\n    get shouldHoldNewMessages() {\r\n        if (this.customSortEnabled)\r\n            return true;\r\n        if (this.holdNewMessages || this.isViewingMore) {\r\n            console.log(`holding due to settings`);\r\n            return true;\r\n        }\r\n\r\n        if (this.enableHoldOnClick && this.messageClicked)\r\n            return true;\r\n\r\n        if (this.enableHoldOnScroll) {\r\n            let keyMessage: ChatMessage;\r\n\r\n            if (this.newestLast)\r\n                keyMessage = this.messages[this.messages.length - 1];\r\n            else\r\n                keyMessage = this.messages[0];\r\n\r\n            if (keyMessage) {\r\n                const messageElement = this.getElementForComment(keyMessage.id);\r\n                if (messageElement) {\r\n                    if (!this.isElementVisible(messageElement)) {\r\n                        console.log(`key element is not visible`);\r\n                        return true;\r\n                    } else {\r\n                        console.log(`key element is visible`);\r\n                    }\r\n                } else {\r\n                    console.log(`could not find key element`);\r\n                }\r\n            } else {\r\n                console.log(`could not find key message`);\r\n            }\r\n        }\r\n\r\n        return false;\r\n    }\r\n\r\n    private isElementVisible(element: Element) {\r\n        const elementRect = element.getBoundingClientRect();\r\n        return !!elementRect\r\n            && elementRect.bottom >= 0\r\n            && elementRect.right >= 0\r\n            && elementRect.left <= document.documentElement.clientWidth\r\n            && elementRect.top <= document.documentElement.clientHeight;\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    saveEdit(message: ChatMessage, newMessage: string) {\r\n        this._messageEdited.next({ message, newMessage });\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    pinMessage(message: ChatMessage, options: PinOptions) {\r\n        this._pinned.next({ message, options });\r\n    }\r\n\r\n    unpinMessage(message: ChatMessage) {\r\n        this._unpinned.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    private setSource(value: ChatSourceBase) {\r\n        this.customSortEnabled = (value?.sortOrder ?? CommentsOrder.NEWEST) !== CommentsOrder.NEWEST;\r\n        this.newMessages = [];\r\n        this.olderMessages = [];\r\n        this.heldMessages = [];\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            this._sourceSubs.add(this._source.messageUpdated.subscribe(msg => this.messageUpdated(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\r\n        // Get the pinned messages\r\n        let pinnedMessages = await this._source.getPinnedMessages();\r\n        pinnedMessages.forEach(m => m.transientState ??= {});\r\n        this.pinnedMessages = pinnedMessages;\r\n\r\n\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\r\n        if (this.messages.length > this.maxVisibleMessages) {\r\n            if (this.newestLast) {\r\n                this.previousMessages.push(...this.messages.splice(0, this.messages.length - this.maxVisibleMessages));\r\n            } else {\r\n                this.nextMessages.unshift(...this.messages.splice(this.maxVisibleMessages, this.messages.length));\r\n            }\r\n        }\r\n\r\n        this.debugMessages();\r\n        this.sortMessages();\r\n        if (this.markSourceLoaded)\r\n            this.markSourceLoaded();\r\n    }\r\n\r\n    debugMessages() {\r\n        if (!this.showDebug)\r\n            return;\r\n\r\n        // console.log([\r\n        //     ...this.previousMessages.map(x => x.message),\r\n        //     '[[',\r\n        //     ...this.messages.map(x => x.message),\r\n        //     ']]',\r\n        //     ...this.nextMessages.map(x => x.message)\r\n        // ].map(x => /\\d+/.test(x) ? this.zeroPad(x, 2) : x).join(\" \"));\r\n    }\r\n\r\n    zeroPad(number: number | string, count: number = 2) {\r\n        let str: string;\r\n\r\n        if (typeof number === 'number')\r\n            str = String(number);\r\n        else\r\n            str = number;\r\n\r\n        while (str.length < count)\r\n            str = '0' + str;\r\n\r\n        return str;\r\n    }\r\n    leftPad(str: string, count: number = 2) {\r\n        while (str.length < count)\r\n            str = ' ' + str;\r\n\r\n        return str;\r\n    }\r\n\r\n    messageIdentity(index: number, chatMessage: ChatMessage) {\r\n        return chatMessage.id;\r\n    }\r\n\r\n    get sourceSortOrder() {\r\n        return this.source?.sortOrder ?? CommentsOrder.NEWEST;\r\n    }\r\n\r\n    get sourceFilterMode() {\r\n        return this.source?.filterMode ?? FilterMode.ALL;\r\n    }\r\n\r\n    /**\r\n     * Show the newest content.\r\n     * - If an unnatural sort order is active (ie Oldest or Likes), it will be changed to Newest.\r\n     * - The new content will be placed where new content goes based on newestLast (replies mode), so if it is true, the content is\r\n     *   placed at the end, otherwise it is placed at the beginning.\r\n     *\r\n     * @param event\r\n     * @returns\r\n     */\r\n    async showNewest(event: MouseEvent) {\r\n\r\n        // Regardless of how we handle this, clear out our held messages\r\n\r\n        this.heldMessages = [];\r\n        this.messageClicked = false;\r\n\r\n        // If the sort order is not already Newest, switch to Newest and stop.\r\n        // The act of changing the sort order will cause the newest content to be loaded.\r\n\r\n        let naturalOrder = CommentsOrder.NEWEST;\r\n        if (this.sourceSortOrder !== naturalOrder || this.sourceFilterMode !== FilterMode.ALL) {\r\n            if (this.sourceSortOrder !== naturalOrder)\r\n                this._sortOrderChanged.next(naturalOrder);\r\n            if (this.sourceFilterMode !== FilterMode.ALL)\r\n                this._filterModeChanged.next(FilterMode.ALL);\r\n            return;\r\n        }\r\n\r\n        // On this path, we are already on Newest, but there is newer content available (such as when new content is\r\n        // being buffered due to user engagement on a comment)\r\n\r\n        this.isViewingMore = false;\r\n\r\n        // Move all newerMessages into messages, respecting the newestLast direction (normal or replies mode)\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\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        // Scroll to the newest comment.\r\n\r\n        if (this.messages.length > 0) {\r\n            if (this.newestLast) {\r\n                this.scrollToComment(this.messages[this.messages.length - 1].id);\r\n            } else {\r\n                this.scrollToComment(this.messages[0].id);\r\n            }\r\n        }\r\n    }\r\n\r\n    get showDebug() {\r\n        if (typeof window === 'undefined')\r\n            return false;\r\n\r\n        return localStorage['banta:debug'] === '1';\r\n    }\r\n\r\n    get shouldShowNext() {\r\n        if (!this.newestLast) {\r\n            return this.hasMore || this.nextMessages.length > 0;\r\n        }\r\n\r\n        return this.nextMessages.length > 0;\r\n    }\r\n\r\n    get shouldShowPrevious() {\r\n        if (this.newestLast) {\r\n            return this.hasMore || this.previousMessages.length > 0;\r\n        }\r\n\r\n        return this.previousMessages.length > 0;\r\n    }\r\n\r\n    get pageSize() {\r\n        return Math.min(20, this.maxVisibleMessages || 20);\r\n    }\r\n\r\n    async showPrevious() {\r\n        this.isViewingMore = true;\r\n        let nextPageSize = this.pageSize;\r\n        this.isLoadingMore = false;\r\n\r\n        if (isNaN(nextPageSize))\r\n            throw new Error(`Not safe to load more with NaN page size`);\r\n\r\n        if (this.previousMessages.length > 0) {\r\n            const storedMessages = this.previousMessages.splice(Math.max(0, this.previousMessages.length - nextPageSize), nextPageSize);\r\n            this.messages = [...storedMessages, ...this.messages];\r\n            nextPageSize -= storedMessages.length;\r\n        }\r\n\r\n\r\n        // Load more from backend if needed\r\n        // Note: Backend only supports fetching more content in one direction.\r\n\r\n        let lastMessage = this.previousMessages[0] ?? this.messages[0];\r\n\r\n        if (nextPageSize > 0 && this.newestLast && lastMessage) {\r\n            this.isLoadingMore = true;\r\n\r\n            let messages = await this.source.loadAfter(lastMessage, nextPageSize);\r\n\r\n            messages = messages.slice().reverse(); // because newestLast === true\r\n            messages.forEach(m => m.transientState ??= {});\r\n\r\n            // In replies mode (newestLast), we want to put these new messages onto the *top* of the set of visible messages.\r\n            // Otherwise we want to put them on the *bottom*.\r\n\r\n            this.messages = [...messages, ...this.messages];\r\n\r\n            // If we didn't receive any messages at all, there's no more to fetch.\r\n\r\n            if (messages.length === 0)\r\n                this.hasMore = false;\r\n\r\n            this.isLoadingMore = false;\r\n        }\r\n\r\n        // Extract the messages that do not fit in the maxVisibleMessages buffer.\r\n\r\n        if (this.messages.length > this.maxVisibleMessages) {\r\n            let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);\r\n            this.nextMessages.unshift(...overflow);\r\n            if (this.nextMessages.length > this.maxMessages)\r\n                this.nextMessages.splice(this.maxMessages, this.nextMessages.length);\r\n        }\r\n\r\n        this.debugMessages();\r\n    }\r\n\r\n    get sortNextLabel() {\r\n        if (this.sourceSortOrder === 'newest') {\r\n            return 'Older';\r\n        } else if (this.sourceSortOrder === 'oldest') {\r\n            return 'Newer';\r\n        } else if (this.sourceSortOrder === 'likes') {\r\n            return 'Less Likes';\r\n        }\r\n\r\n        return 'More';\r\n    }\r\n\r\n    get sortPreviousLabel() {\r\n        if (this.sourceSortOrder === 'newest') {\r\n            return 'Newer';\r\n        } else if (this.sourceSortOrder === 'oldest') {\r\n            return 'Older';\r\n        } else if (this.sourceSortOrder === 'likes') {\r\n            return 'More Likes';\r\n        }\r\n\r\n        return 'More';\r\n    }\r\n\r\n    get nextLabel() { return this.newestLast ? this.sortPreviousLabel : this.sortNextLabel; }\r\n    get previousLabel() { return this.newestLast ? this.sortNextLabel : this.sortPreviousLabel; }\r\n\r\n    /**\r\n     * Show more content\r\n     * - When in replies mode (newestLast), the content is added at the top\r\n     * - When in normal mode, the content is added at the bottom\r\n     * - The current sort order does *not* factor in here, which is why it is showMore() not showEarlier().\r\n     *\r\n     * @returns\r\n     */\r\n    async showNext() {\r\n        this.isViewingMore = true;\r\n\r\n        let nextPageSize = this.pageSize;\r\n\r\n        if (isNaN(nextPageSize))\r\n            throw new Error(`Not safe to load more with NaN page size`);\r\n\r\n        this.isLoadingMore = false;\r\n\r\n        if (this.nextMessages.length > 0) {\r\n            const storedMessages = this.nextMessages.splice(0, nextPageSize);\r\n            this.messages = [ ...this.messages, ...storedMessages ];\r\n            nextPageSize -= storedMessages.length;\r\n        }\r\n\r\n        const lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];\r\n\r\n        if (nextPageSize > 0 && !this.newestLast && lastMessage) {\r\n            // Load more from backend\r\n\r\n            this.isLoadingMore = true;\r\n\r\n            let messages = await this.source.loadAfter(lastMessage, nextPageSize);\r\n            messages.forEach(m => m.transientState ??= {});\r\n            this.messages = [ ...this.messages, ...messages ];\r\n\r\n            // If we didn't receive any messages at all, there's no more to fetch.\r\n\r\n            if (messages.length === 0)\r\n                this.hasMore = false;\r\n\r\n            this.isLoadingMore = false;\r\n        }\r\n\r\n        // Extract the messages that do not fit in the maxVisibleMessages buffer.\r\n\r\n\r\n        if (this.messages.length > this.maxVisibleMessages) {\r\n            let overflow = this.messages.splice(0, this.messages.length - this.maxVisibleMessages);\r\n\r\n            // Regardless of the order (newestLast), newMessages represents the direction that is being pushed, since it's definition\r\n            // depends on that order. Move overflowing messages into newMessages.\r\n\r\n            this.previousMessages.push(...overflow);\r\n            if (this.previousMessages.length > this.maxMessages)\r\n                this.previousMessages.splice(0, this.previousMessages.length - this.maxMessages);\r\n        }\r\n\r\n        this.debugMessages();\r\n    }\r\n\r\n    /**\r\n     * Show more content\r\n     * - When in replies mode (newestLast), the content is added at the top\r\n     * - When in normal mode, the content is added at the bottom\r\n     * - The current sort order does *not* factor in here, which is why it is showMore() not showEarlier().\r\n     *\r\n     * @returns\r\n     */\r\n    async showMore() {\r\n        this.isViewingMore = true;\r\n\r\n        let nextPageSize = this.pageSize;\r\n\r\n        if (isNaN(nextPageSize))\r\n            throw new Error(`Not safe to load more with NaN page size`);\r\n\r\n        this.isLoadingMore = false;\r\n\r\n        if (this.olderMessages.length > 0) {\r\n            const storedMessages = this.olderMessages.splice(0, nextPageSize);\r\n            this.messages = this.messages.concat(storedMessages);\r\n            nextPageSize -= storedMessages.length;\r\n            this.hasMore = this.olderMessages.length > 0;\r\n        }\r\n\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 (nextPageSize > 0 && lastMessage) {\r\n            // Load more from backend\r\n\r\n            this.isLoadingMore = true;\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            // In replies mode (newestLast), we want to put these new messages onto the *top* of the set of visible messages.\r\n            // Otherwise we want to put them on the *bottom*.\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            }\r\n\r\n            // If we didn't receive any messages at all, there's no more to fetch.\r\n\r\n            if (messages.length === 0) {\r\n                this.hasMore = false;\r\n            }\r\n\r\n            this.isLoadingMore = false;\r\n        }\r\n\r\n        // Extract the messages that do not fit in the maxVisibleMessages buffer.\r\n\r\n        if (this.messages.length > this.maxVisibleMessages) {\r\n            let overflow: ChatMessage[] = [];\r\n\r\n            // Move overflowing messages into newMessages.\r\n            // Regardless of the order (newestLast), newMessages represents the direction that is being loaded,\r\n            // since it's definition depends on that order.\r\n\r\n            if (this.newestLast) {\r\n                overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);\r\n                this.newMessages = overflow.concat(this.newMessages);\r\n                this.newMessages.splice(this.maxMessages - this.maxVisibleMessages, this.newMessages.length);\r\n            } else {\r\n                overflow = this.messages.splice(0, this.messages.length - this.maxVisibleMessages);\r\n                this.newMessages.push(...overflow);\r\n                if (this.newMessages.length > this.maxMessages - this.maxVisibleMessages)\r\n                    this.newMessages.splice(0, this.newMessages.length - (this.maxMessages - this.maxVisibleMessages));\r\n            }\r\n\r\n        }\r\n    }\r\n\r\n    private heldMessages: ChatMessage[] = [];\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.shouldHoldNewMessages) {\r\n            this.heldMessages.push(message);\r\n            destination = this.newMessages;\r\n            bucket = null;\r\n        }\r\n\r\n        // If we aren't on the newest sort order, new messages shouldn't be added at all\r\n        if (this.sourceSortOrder !== CommentsOrder.NEWEST)\r\n            return;\r\n\r\n\r\n        if (newestLast) {\r\n            destination.push(message);\r\n\r\n            if (this.maxVisibleMessages > 0) {\r\n                let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n                bucket?.push(...overflow);\r\n            }\r\n        } else {\r\n            destination.unshift(message);\r\n\r\n            if (this.maxVisibleMessages > 0) {\r\n                let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n                bucket?.unshift(...overflow);\r\n            }\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\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    private isPinned(message: ChatMessage) {\r\n        return message.pinned && (!message.pinnedUntil || message.pinnedUntil > Date.now());\r\n    }\r\n\r\n    private messageUpdated(message: ChatMessage) {\r\n        let pinned = this.isPinned(message);\r\n        let inPins = this.pinnedMessages.some(x => x.id === message.id);\r\n\r\n        if (pinned && !inPins) {\r\n            this.pinnedMessages.unshift(message);\r\n            let index = this.messages.indexOf(message);\r\n            if (index >= 0) {\r\n                this.messages.splice(index, 1);\r\n            }\r\n        } else if (!pinned && inPins) {\r\n            this.messages.push(message);\r\n            let index = this.pinnedMessages.indexOf(message);\r\n            if (index >= 0) {\r\n                this.pinnedMessages.splice(index, 1);\r\n            }\r\n\r\n            this.sortMessages();\r\n        }\r\n    }\r\n\r\n    scrollToLatest() {\r\n        if (!this.messageContainer) {\r\n            return;\r\n        }\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n\r\n        el.scrollIntoView({ block: 'start' });\r\n        //el.scrollTop = el.scrollHeight;\r\n    }\r\n\r\n    get element() {\r\n        return this.elementRef.nativeElement;\r\n    }\r\n\r\n    async scrollToComment(commentId: ChatMessage['id']) {\r\n        if (typeof window === 'undefined')\r\n            return;\r\n\r\n        await this.waitForAllCommentsToLoad();\r\n\r\n        const comment = this.getElementForComment(commentId);\r\n        if (comment) {\r\n            comment.scrollIntoView({\r\n                inline: 'center',\r\n                block: 'center'\r\n            });\r\n        }\r\n    }\r\n\r\n    getElementForComment(commentId: string) {\r\n        return this.element.querySelector(`[data-comment-id=\"${commentId}\"]`);\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=\"banta-message-container\" #messageContainer>\r\n    <ng-content select=\"[data-before]\"></ng-content>\r\n\r\n    @if (!collapsePins) {\r\n        @for (message of pinnedMessages; track message.id) {\r\n            @if (!message.hidden) {\r\n                <banta-comment\r\n                    class=\"abbreviated\"\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                    [readonly]=\"source?.readonly\"\r\n                    (click)=\"messageClicked = 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                    (pinned)=\"pinMessage(message, $event.options)\"\r\n                    (unpinned)=\"unpinMessage(message)\"\r\n                    (unliked)=\"unlikeMessage(message)\"\r\n                    (reported)=\"reportMessage(message)\"\r\n                    (selected)=\"selectMessage(message)\"\r\n                    (shared)=\"sharedMessage($event)\"\r\n                    />\r\n                <div class=\"banta-inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n                    <ng-container *ngTemplateOutlet=\"inlineRepliesTemplate\" />\r\n                </div>\r\n            }\r\n        }\r\n    }\r\n\r\n    <div class=\"banta-top-sticky\">\r\n        @if (!newestLast) {\r\n            <button\r\n                mat-button\r\n                class=\"banta-nav\"\r\n                [class.visible]=\"shouldShowNewMessageIndicator\"\r\n                href=\"javascript:;\"\r\n                (click)=\"showNewest($event)\"\r\n                >\r\n                <mat-icon>file_upload</mat-icon>\r\n                Newest\r\n                @if (heldMessages.length > 0) {\r\n                    <span class=\"count\">{{ heldMessages.length | number }}</span>\r\n                }\r\n            </button>\r\n        }\r\n    </div>\r\n\r\n    <button mat-button class=\"pager\" (click)=\"showPrevious()\" [class.visible]=\"shouldShowPrevious\" [disabled]=\"isLoadingMore\">\r\n        <mat-icon>expand_less</mat-icon>\r\n        {{ previousLabel }}\r\n    </button>\r\n\r\n    @for (message of messages; track message.id) {\r\n        @if (!message.hidden) {\r\n            <banta-comment\r\n                class=\"abbreviated\"\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                [readonly]=\"source?.readonly\"\r\n                (click)=\"messageClicked = 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                (pinned)=\"pinMessage(message, $event.options)\"\r\n                (unpinned)=\"unpinMessage(message)\"\r\n                (unliked)=\"unlikeMessage(message)\"\r\n                (reported)=\"reportMessage(message)\"\r\n                (selected)=\"selectMessage(message)\"\r\n                (shared)=\"sharedMessage($event)\"\r\n                />\r\n            <div class=\"banta-inline-replies-container\" *ngIf=\"selectedMessage === message\">\r\n                <ng-container *ngTemplateOutlet=\"inlineRepliesTemplate\" />\r\n            </div>\r\n        }\r\n    } @empty {\r\n        <div class=\"banta-empty-state\" *ngIf=\"showEmptyState\">\r\n            {{ emptyStateMessage }}\r\n        </div>\r\n    }\r\n\r\n    <button mat-button class=\"pager\" (click)=\"showNext()\" [class.visible]=\"shouldShowNext\" [disabled]=\"isLoadingMore\">\r\n        <mat-icon>expand_more</mat-icon>\r\n        {{ nextLabel }}\r\n    </button>\r\n\r\n    <div class=\"banta-nav-point banta-bottom-sticky\">\r\n        @if (newestLast) {\r\n            <button\r\n                [matBadge]=\"10\" matBadgeOverlap=\"false\"\r\n                matBadgePosition=\"after\" matBadgeSize=\"large\"\r\n                mat-button\r\n                class=\"banta-nav\"\r\n                [class.visible]=\"shouldShowNewMessageIndicator\"\r\n                href=\"javascript:;\"\r\n                (click)=\"showNewest($event)\"\r\n                >\r\n                <mat-icon>file_download</mat-icon>\r\n                Newest\r\n                @if (heldMessages.length > 0) {\r\n                    <span class=\"count\">{{ heldMessages.length | number }}</span>\r\n                }\r\n            </button>\r\n        }\r\n    </div>\r\n\r\n    <div class=\"banta-loading-more\" *ngIf=\"isLoadingMore\">\r\n        <mat-spinner></mat-spinner>\r\n    </div>\r\n\r\n    @if (showDebug) {\r\n        <div style=\"color: #666\">\r\n            ({{ previousMessages.length }} .. {{ messages.length }} .. {{ nextMessages.length }})\r\n\r\n            dir={{newestLast ? '-1' : '1'}}\r\n            v={{maxVisibleMessages}}, M={{maxMessages}}\r\n        </div>\r\n    }\r\n\r\n    <ng-content select=\":not([data-before]):not(.inline-replies)\"></ng-content>\r\n</div>\r\n"]}
|