@banta/sdk 4.5.4 → 4.6.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,8 +1,9 @@
1
1
  import { __awaiter } from "tslib";
2
- import { Component, Input, ViewChild, ElementRef, Output, HostBinding } from "@angular/core";
2
+ import { Component, Input, ViewChild, ElementRef, Output, HostBinding, ViewChildren, QueryList } from "@angular/core";
3
3
  import { CommentsOrder } from '@banta/common';
4
4
  import { Subject, Subscription } from 'rxjs';
5
5
  import { ChatBackendBase } from "../../chat-backend-base";
6
+ import { CommentComponent } from "../comment/comment.component";
6
7
  export class CommentViewComponent {
7
8
  constructor(backend) {
8
9
  this.backend = backend;
@@ -33,6 +34,21 @@ export class CommentViewComponent {
33
34
  this.olderMessages = [];
34
35
  this.sortOrderChanged = new Subject();
35
36
  }
37
+ get comments() {
38
+ return Array.from(this.commentsQuery);
39
+ }
40
+ /**
41
+ * Get the CommentComponent instantiated for the given ChatMessage,
42
+ * if it exists in the current view. Note that messages which are not
43
+ * currently shown to the user will not return a CommentComponent.
44
+ * @param message
45
+ * @returns
46
+ */
47
+ getCommentComponentForMessage(message) {
48
+ if (!message)
49
+ throw new Error(`You must pass a valid ChatMessage`);
50
+ return this.comments.find(x => { var _a; return ((_a = x.message) === null || _a === void 0 ? void 0 : _a.id) === message.id; });
51
+ }
36
52
  get selected() {
37
53
  return this._selected;
38
54
  }
@@ -213,9 +229,18 @@ export class CommentViewComponent {
213
229
  return maxPagingCursor;
214
230
  }
215
231
  sortMessages() {
216
- this.messages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));
217
- this.olderMessages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));
218
- this.newMessages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));
232
+ if (!this.source)
233
+ return;
234
+ let sorter;
235
+ if (this.source.sortOrder === CommentsOrder.LIKES)
236
+ sorter = (a, b) => b.likes - a.likes;
237
+ else if (this.source.sortOrder === CommentsOrder.NEWEST)
238
+ sorter = (a, b) => (b.sentAt - a.sentAt) * (this.newestLast ? -1 : 1);
239
+ else if (this.source.sortOrder === CommentsOrder.OLDEST)
240
+ sorter = (a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? -1 : 1);
241
+ this.messages.sort(sorter);
242
+ this.olderMessages.sort(sorter);
243
+ this.newMessages.sort(sorter);
219
244
  }
220
245
  messageReceived(message) {
221
246
  this.addMessage(message);
@@ -264,6 +289,7 @@ CommentViewComponent.propDecorators = {
264
289
  showEmptyState: [{ type: Input }],
265
290
  allowReplies: [{ type: Input }],
266
291
  customMenuItems: [{ type: Input }],
292
+ commentsQuery: [{ type: ViewChildren, args: [CommentComponent,] }],
267
293
  fixedHeight: [{ type: Input }, { type: HostBinding, args: ['class.fixed-height',] }],
268
294
  selectedMessage: [{ type: Input }],
269
295
  selected: [{ type: Output }],
@@ -284,4 +310,4 @@ CommentViewComponent.propDecorators = {
284
310
  newestLast: [{ type: Input }],
285
311
  sortOrderChanged: [{ type: Output }]
286
312
  };
287
- //# 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"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output, HostBinding } from \"@angular/core\";\r\nimport { User, ChatMessage, CommentsOrder } from '@banta/common';\r\nimport { Subject, Subscription } from 'rxjs';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\nimport { MessageMenuItem } from \"../../message-menu-item\";\r\n\r\nexport interface EditEvent {\r\n    message: ChatMessage;\r\n    newMessage: string;\r\n}\r\n\r\n@Component({\r\n    selector: 'banta-comment-view',\r\n    templateUrl: './comment-view.component.html',\r\n    styleUrls: ['./comment-view.component.scss']\r\n})\r\nexport class CommentViewComponent {\r\n    constructor(\r\n        private backend: ChatBackendBase\r\n    ) {\r\n\r\n    }\r\n\r\n    private _sourceSubs = new Subscription();\r\n    private _source: ChatSourceBase;\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _liked = new Subject<ChatMessage>();\r\n    private _unliked = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _usernameSelected = new Subject<User>();\r\n    private _avatarSelected = new Subject<User>();\r\n    private _shared = new Subject<ChatMessage>();\r\n    private _deleted = new Subject<ChatMessage>();\r\n    private _messageEdited = new Subject<EditEvent>();\r\n\r\n    @Input()\r\n    showEmptyState = true;\r\n\r\n    @Input()\r\n    allowReplies = true;\r\n\r\n    @Input() customMenuItems: MessageMenuItem[] = [];\r\n\r\n    @Input()\r\n    @HostBinding('class.fixed-height')\r\n    fixedHeight: boolean;\r\n\r\n    @Input()\r\n    selectedMessage: ChatMessage;\r\n    \r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output() \r\n    get messageEdited() {\r\n        return this._messageEdited.asObservable();\r\n    }\r\n\r\n    saveEdit(message: ChatMessage, newMessage: string) {\r\n        this._messageEdited.next({ message, newMessage });\r\n    }\r\n\r\n    @Output() get userSelected() { return this._userSelected; }\r\n    @Output() get reported() { return this._reported; }\r\n    @Output() get liked() { return this._liked; }\r\n    @Output() get unliked() { return this._unliked; }\r\n    @Output() get usernameSelected() { return this._usernameSelected; }\r\n    @Output() get avatarSelected() { return this._avatarSelected; }\r\n    @Output() get shared() { return this._shared; }\r\n    @Output() get deleted() { return this._deleted; }\r\n\r\n    menuMessage: ChatMessage = null;\r\n    messages: ChatMessage[] = [];\r\n    currentUser: User;\r\n\r\n    @Input()\r\n    get source() {\r\n        return this._source;\r\n    }\r\n\r\n    likeMessage(message: ChatMessage) {\r\n        this._liked.next(message);\r\n    }\r\n\r\n    unlikeMessage(message: ChatMessage) {\r\n        this._unliked.next(message);\r\n    }\r\n\r\n    reportMessage(message: ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    selectMessage(message: ChatMessage) {\r\n        this._selected.next(message);\r\n    }\r\n\r\n    selectMessageUser(message: ChatMessage) {\r\n        this._userSelected.next(message);\r\n    }\r\n\r\n    selectUsername(user: User) {\r\n        this._usernameSelected.next(user);\r\n    }\r\n\r\n    selectAvatar(user: User) {\r\n        this._avatarSelected.next(user);\r\n    }\r\n\r\n    sharedMessage(message: ChatMessage) {\r\n        this._shared.next(message);\r\n    }\r\n\r\n    startEditing(message: ChatMessage) {\r\n        this.messages.forEach(m => m.transientState.editing = false);\r\n        message.transientState.editing = true;\r\n    }\r\n\r\n    deleteMessage(message: ChatMessage) {\r\n        this._deleted.next(message);\r\n    }\r\n\r\n    customSortEnabled = false;\r\n\r\n    set source(value) {\r\n        this.customSortEnabled = value?.sortOrder !== CommentsOrder.NEWEST;\r\n        this.newMessages = [];\r\n        this.olderMessages = [];\r\n\r\n        (window as any).bantaSourceDebug = value;\r\n\r\n        if (this._sourceSubs) {\r\n            this._sourceSubs.unsubscribe();\r\n            this._sourceSubs = null;\r\n        }\r\n        this._source = value;\r\n\r\n        if (value) {\r\n            const messages = (value.messages || []).slice();\r\n            this.messages = messages;\r\n            this.olderMessages = messages.splice(this.maxVisibleMessages, messages.length);\r\n            this.hasMore = true; //this.olderMessages.length > 0;\r\n\r\n            this._sourceSubs = new Subscription();\r\n            this._sourceSubs.add(this._source.messageReceived.subscribe(msg => this.messageReceived(msg)));\r\n            this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));\r\n\r\n            this._sourceSubs.add(\r\n                this.backend.userChanged.subscribe(user => this.currentUser = user)\r\n            );\r\n\r\n            this.getInitialMessages();\r\n        }\r\n    }\r\n\r\n    private async getInitialMessages() {\r\n        let messages = (await this._source.getExistingMessages());\r\n        messages.forEach(m => m.transientState ??= {});\r\n        this.messages = this.newestLast ? messages.slice().reverse() : messages;\r\n        this.sortMessages();\r\n    }\r\n\r\n    @Input()\r\n    genericAvatarUrl: string;\r\n\r\n    @ViewChild('messageContainer')\r\n    messageContainer: ElementRef<HTMLElement>;\r\n\r\n    @Input()\r\n    maxMessages = 2000;\r\n\r\n    @Input()\r\n    maxVisibleMessages: number = 200;\r\n\r\n    @Input()\r\n    newestLast = false;\r\n\r\n    isViewingMore = false;\r\n    isLoadingMore = false;\r\n    hasMore = false;\r\n\r\n    newMessages: ChatMessage[] = [];\r\n    olderMessages: ChatMessage[] = [];\r\n\r\n    messageIdentity(index: number, chatMessage: ChatMessage) {\r\n        return chatMessage.id;\r\n    }\r\n\r\n    @Output()\r\n    sortOrderChanged = new Subject<CommentsOrder>();\r\n\r\n    async showNew() {\r\n        let naturalOrder = CommentsOrder.NEWEST;\r\n        if (this.source && this.source.sortOrder !== naturalOrder) {\r\n            this.sortOrderChanged.next(naturalOrder);\r\n            return;\r\n        }\r\n\r\n        this.isViewingMore = false;\r\n\r\n        if (this.newestLast)\r\n            this.messages = this.messages.concat(this.newMessages.splice(0, this.newMessages.length));\r\n        else\r\n            this.messages = this.newMessages.splice(0, this.newMessages.length).concat(this.messages);\r\n        let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);\r\n        this.olderMessages = overflow.concat(this.olderMessages);\r\n        this.olderMessages.splice(this.maxMessages - this.maxVisibleMessages, this.olderMessages.length);\r\n    }\r\n\r\n    async showMore() {\r\n        this.isViewingMore = true;\r\n\r\n        if (this.olderMessages.length > 0) {\r\n            this.isLoadingMore = false;\r\n            this.messages = this.messages.concat(this.olderMessages.splice(0, 50));\r\n        }\r\n        this.isLoadingMore = true;\r\n\r\n        let nextPageSize = 20;\r\n        let lastMessage: ChatMessage;\r\n\r\n        if (this.newestLast) {\r\n            lastMessage = this.olderMessages[0] ?? this.messages[0];\r\n        } else {\r\n            lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];\r\n        }\r\n\r\n        console.dir(lastMessage);\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        messages.forEach(m => m.transientState ??= {});\r\n\r\n        if (this.newestLast)\r\n            this.messages = messages.concat(this.messages);\r\n        else\r\n            this.messages = this.messages.concat(messages);\r\n        this.isLoadingMore = false;\r\n        if (messages.length === 0) {\r\n            console.log(`Reached the end of the list.`);\r\n            this.hasMore = false;\r\n        }\r\n    }\r\n\r\n    private addMessage(message: ChatMessage) {\r\n\r\n        if (!message.transientState)\r\n            message.transientState ??= {};\r\n        \r\n        let destination = this.messages;\r\n        let bucket = this.olderMessages;\r\n        let newestLast = this.newestLast;\r\n\r\n        if (this.isViewingMore) {\r\n            destination = this.newMessages;\r\n            bucket = null;\r\n        }\r\n\r\n        \r\n        if (newestLast) {\r\n            destination.push(message);\r\n            let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n            bucket?.push(...overflow);\r\n        } else {\r\n            destination.unshift(message);\r\n            let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n            bucket?.unshift(...overflow);\r\n        }\r\n\r\n        if (bucket?.length > 0)\r\n            this.hasMore = true;\r\n\r\n        message.pagingCursor = String(this.incrementPagingCursors());\r\n        this.sortMessages();\r\n    }\r\n\r\n    private incrementPagingCursors() {\r\n        if (this.source.sortOrder !== CommentsOrder.NEWEST)\r\n            return;\r\n        \r\n        let maxPagingCursor = 0;\r\n        for (let group of [this.messages, this.olderMessages, this.newMessages]) {\r\n            for (let message of group) {\r\n                if (message.pagingCursor) {\r\n                    let pagingCursor = Number(message.pagingCursor) + 1;\r\n                    if (pagingCursor > maxPagingCursor)\r\n                        maxPagingCursor = pagingCursor;\r\n                    message.pagingCursor = String(pagingCursor);\r\n                }\r\n            }\r\n        }\r\n\r\n        return maxPagingCursor;\r\n    }\r\n\r\n    private sortMessages() {\r\n        this.messages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));\r\n        this.olderMessages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));\r\n        this.newMessages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));\r\n    }\r\n\r\n    private messageReceived(message: ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (this.isScrolledToLatest())\r\n            setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    isScrolledToLatest() {\r\n        if (!this.messageContainer)\r\n            return false;\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n        const currentScroll = el.scrollTop;\r\n        const currentTotal = el.scrollHeight - el.offsetHeight;\r\n\r\n        return currentScroll > currentTotal - 10;\r\n    }\r\n\r\n    private messageSent(message: ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (!this.messageContainer)\r\n            return;\r\n\r\n        this.scrollToLatest();\r\n    }\r\n\r\n    scrollToLatest() {\r\n        if (!this.messageContainer)\r\n            return;\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n        el.scrollTop = el.scrollHeight;\r\n    }\r\n\r\n    mentionsMe(message: ChatMessage) {\r\n        if (!this.currentUser)\r\n            return false;\r\n\r\n        if (message.message.includes(`@${this.currentUser.username}`))\r\n            return true;\r\n\r\n        return false;\r\n    }\r\n}\r\n"]}
313
+ //# 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"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output, HostBinding, ViewChildren, QueryList } from \"@angular/core\";\r\nimport { User, ChatMessage, CommentsOrder } from '@banta/common';\r\nimport { Subject, Subscription } from 'rxjs';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\nimport { MessageMenuItem } from \"../../message-menu-item\";\r\nimport { CommentComponent } from \"../comment/comment.component\";\r\n\r\nexport interface EditEvent {\r\n    message: ChatMessage;\r\n    newMessage: string;\r\n}\r\n\r\n@Component({\r\n    selector: 'banta-comment-view',\r\n    templateUrl: './comment-view.component.html',\r\n    styleUrls: ['./comment-view.component.scss']\r\n})\r\nexport class CommentViewComponent {\r\n    constructor(\r\n        private backend: ChatBackendBase\r\n    ) {\r\n\r\n    }\r\n\r\n    private _sourceSubs = new Subscription();\r\n    private _source: ChatSourceBase;\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _liked = new Subject<ChatMessage>();\r\n    private _unliked = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _usernameSelected = new Subject<User>();\r\n    private _avatarSelected = new Subject<User>();\r\n    private _shared = new Subject<ChatMessage>();\r\n    private _deleted = new Subject<ChatMessage>();\r\n    private _messageEdited = new Subject<EditEvent>();\r\n\r\n    @Input()\r\n    showEmptyState = true;\r\n\r\n    @Input()\r\n    allowReplies = true;\r\n\r\n    @Input() customMenuItems: MessageMenuItem[] = [];\r\n\r\n    @ViewChildren(CommentComponent)\r\n    commentsQuery: QueryList<CommentComponent>;\r\n\r\n    get comments() {\r\n        return Array.from(this.commentsQuery);\r\n    }\r\n\r\n    /**\r\n     * Get the CommentComponent instantiated for the given ChatMessage,\r\n     * if it exists in the current view. Note that messages which are not \r\n     * currently shown to the user will not return a CommentComponent.\r\n     * @param message \r\n     * @returns \r\n     */\r\n    getCommentComponentForMessage(message: ChatMessage) {\r\n        if (!message)\r\n            throw new Error(`You must pass a valid ChatMessage`);\r\n        return this.comments.find(x => x.message?.id === message.id);\r\n    }\r\n\r\n    @Input()\r\n    @HostBinding('class.fixed-height')\r\n    fixedHeight: boolean;\r\n\r\n    @Input()\r\n    selectedMessage: ChatMessage;\r\n    \r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output() \r\n    get messageEdited() {\r\n        return this._messageEdited.asObservable();\r\n    }\r\n\r\n    saveEdit(message: ChatMessage, newMessage: string) {\r\n        this._messageEdited.next({ message, newMessage });\r\n    }\r\n\r\n    @Output() get userSelected() { return this._userSelected; }\r\n    @Output() get reported() { return this._reported; }\r\n    @Output() get liked() { return this._liked; }\r\n    @Output() get unliked() { return this._unliked; }\r\n    @Output() get usernameSelected() { return this._usernameSelected; }\r\n    @Output() get avatarSelected() { return this._avatarSelected; }\r\n    @Output() get shared() { return this._shared; }\r\n    @Output() get deleted() { return this._deleted; }\r\n\r\n    menuMessage: ChatMessage = null;\r\n    messages: ChatMessage[] = [];\r\n    currentUser: User;\r\n\r\n    @Input()\r\n    get source() {\r\n        return this._source;\r\n    }\r\n\r\n    likeMessage(message: ChatMessage) {\r\n        this._liked.next(message);\r\n    }\r\n\r\n    unlikeMessage(message: ChatMessage) {\r\n        this._unliked.next(message);\r\n    }\r\n\r\n    reportMessage(message: ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    selectMessage(message: ChatMessage) {\r\n        this._selected.next(message);\r\n    }\r\n\r\n    selectMessageUser(message: ChatMessage) {\r\n        this._userSelected.next(message);\r\n    }\r\n\r\n    selectUsername(user: User) {\r\n        this._usernameSelected.next(user);\r\n    }\r\n\r\n    selectAvatar(user: User) {\r\n        this._avatarSelected.next(user);\r\n    }\r\n\r\n    sharedMessage(message: ChatMessage) {\r\n        this._shared.next(message);\r\n    }\r\n\r\n    startEditing(message: ChatMessage) {\r\n        this.messages.forEach(m => m.transientState.editing = false);\r\n        message.transientState.editing = true;\r\n    }\r\n\r\n    deleteMessage(message: ChatMessage) {\r\n        this._deleted.next(message);\r\n    }\r\n\r\n    customSortEnabled = false;\r\n\r\n    set source(value) {\r\n        this.customSortEnabled = value?.sortOrder !== CommentsOrder.NEWEST;\r\n        this.newMessages = [];\r\n        this.olderMessages = [];\r\n\r\n        (window as any).bantaSourceDebug = value;\r\n\r\n        if (this._sourceSubs) {\r\n            this._sourceSubs.unsubscribe();\r\n            this._sourceSubs = null;\r\n        }\r\n        this._source = value;\r\n\r\n        if (value) {\r\n            const messages = (value.messages || []).slice();\r\n            this.messages = messages;\r\n            this.olderMessages = messages.splice(this.maxVisibleMessages, messages.length);\r\n            this.hasMore = true; //this.olderMessages.length > 0;\r\n\r\n            this._sourceSubs = new Subscription();\r\n            this._sourceSubs.add(this._source.messageReceived.subscribe(msg => this.messageReceived(msg)));\r\n            this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));\r\n\r\n            this._sourceSubs.add(\r\n                this.backend.userChanged.subscribe(user => this.currentUser = user)\r\n            );\r\n\r\n            this.getInitialMessages();\r\n        }\r\n    }\r\n\r\n    private async getInitialMessages() {\r\n        let messages = (await this._source.getExistingMessages());\r\n        messages.forEach(m => m.transientState ??= {});\r\n        this.messages = this.newestLast ? messages.slice().reverse() : messages;\r\n        this.sortMessages();\r\n    }\r\n\r\n    @Input()\r\n    genericAvatarUrl: string;\r\n\r\n    @ViewChild('messageContainer')\r\n    messageContainer: ElementRef<HTMLElement>;\r\n\r\n    @Input()\r\n    maxMessages = 2000;\r\n\r\n    @Input()\r\n    maxVisibleMessages: number = 200;\r\n\r\n    @Input()\r\n    newestLast = false;\r\n\r\n    isViewingMore = false;\r\n    isLoadingMore = false;\r\n    hasMore = false;\r\n\r\n    newMessages: ChatMessage[] = [];\r\n    olderMessages: ChatMessage[] = [];\r\n\r\n    messageIdentity(index: number, chatMessage: ChatMessage) {\r\n        return chatMessage.id;\r\n    }\r\n\r\n    @Output()\r\n    sortOrderChanged = new Subject<CommentsOrder>();\r\n\r\n    async showNew() {\r\n        let naturalOrder = CommentsOrder.NEWEST;\r\n        if (this.source && this.source.sortOrder !== naturalOrder) {\r\n            this.sortOrderChanged.next(naturalOrder);\r\n            return;\r\n        }\r\n\r\n        this.isViewingMore = false;\r\n\r\n        if (this.newestLast)\r\n            this.messages = this.messages.concat(this.newMessages.splice(0, this.newMessages.length));\r\n        else\r\n            this.messages = this.newMessages.splice(0, this.newMessages.length).concat(this.messages);\r\n        let overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);\r\n        this.olderMessages = overflow.concat(this.olderMessages);\r\n        this.olderMessages.splice(this.maxMessages - this.maxVisibleMessages, this.olderMessages.length);\r\n    }\r\n\r\n    async showMore() {\r\n        this.isViewingMore = true;\r\n\r\n        if (this.olderMessages.length > 0) {\r\n            this.isLoadingMore = false;\r\n            this.messages = this.messages.concat(this.olderMessages.splice(0, 50));\r\n        }\r\n        this.isLoadingMore = true;\r\n\r\n        let nextPageSize = 20;\r\n        let lastMessage: ChatMessage;\r\n\r\n        if (this.newestLast) {\r\n            lastMessage = this.olderMessages[0] ?? this.messages[0];\r\n        } else {\r\n            lastMessage = this.olderMessages[this.olderMessages.length - 1] ?? this.messages[this.messages.length - 1];\r\n        }\r\n\r\n        console.dir(lastMessage);\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        messages.forEach(m => m.transientState ??= {});\r\n\r\n        if (this.newestLast)\r\n            this.messages = messages.concat(this.messages);\r\n        else\r\n            this.messages = this.messages.concat(messages);\r\n        this.isLoadingMore = false;\r\n        if (messages.length === 0) {\r\n            console.log(`Reached the end of the list.`);\r\n            this.hasMore = false;\r\n        }\r\n    }\r\n\r\n    private addMessage(message: ChatMessage) {\r\n\r\n        if (!message.transientState)\r\n            message.transientState ??= {};\r\n        \r\n        let destination = this.messages;\r\n        let bucket = this.olderMessages;\r\n        let newestLast = this.newestLast;\r\n\r\n        if (this.isViewingMore) {\r\n            destination = this.newMessages;\r\n            bucket = null;\r\n        }\r\n\r\n        \r\n        if (newestLast) {\r\n            destination.push(message);\r\n            let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n            bucket?.push(...overflow);\r\n        } else {\r\n            destination.unshift(message);\r\n            let overflow = destination.splice(this.maxVisibleMessages, destination.length);\r\n            bucket?.unshift(...overflow);\r\n        }\r\n\r\n        if (bucket?.length > 0)\r\n            this.hasMore = true;\r\n\r\n        message.pagingCursor = String(this.incrementPagingCursors());\r\n        this.sortMessages();\r\n    }\r\n\r\n    private incrementPagingCursors() {\r\n        if (this.source.sortOrder !== CommentsOrder.NEWEST)\r\n            return;\r\n        \r\n        let maxPagingCursor = 0;\r\n        for (let group of [this.messages, this.olderMessages, this.newMessages]) {\r\n            for (let message of group) {\r\n                if (message.pagingCursor) {\r\n                    let pagingCursor = Number(message.pagingCursor) + 1;\r\n                    if (pagingCursor > maxPagingCursor)\r\n                        maxPagingCursor = pagingCursor;\r\n                    message.pagingCursor = String(pagingCursor);\r\n                }\r\n            }\r\n        }\r\n\r\n        return maxPagingCursor;\r\n    }\r\n\r\n    private sortMessages() {\r\n        if (!this.source)\r\n            return;\r\n        \r\n        let sorter: (a: ChatMessage, b: ChatMessage) => number;\r\n\r\n        if (this.source.sortOrder === CommentsOrder.LIKES)\r\n            sorter = (a, b) => b.likes - a.likes;\r\n        else if (this.source.sortOrder === CommentsOrder.NEWEST)\r\n            sorter = (a, b) => (b.sentAt - a.sentAt) * (this.newestLast ? -1 : 1);\r\n        else if (this.source.sortOrder === CommentsOrder.OLDEST)\r\n            sorter = (a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? -1 : 1);\r\n\r\n        this.messages.sort(sorter);\r\n        this.olderMessages.sort(sorter);\r\n        this.newMessages.sort(sorter);\r\n    }\r\n\r\n    private messageReceived(message: ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (this.isScrolledToLatest())\r\n            setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    isScrolledToLatest() {\r\n        if (!this.messageContainer)\r\n            return false;\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n        const currentScroll = el.scrollTop;\r\n        const currentTotal = el.scrollHeight - el.offsetHeight;\r\n\r\n        return currentScroll > currentTotal - 10;\r\n    }\r\n\r\n    private messageSent(message: ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (!this.messageContainer)\r\n            return;\r\n\r\n        this.scrollToLatest();\r\n    }\r\n\r\n    scrollToLatest() {\r\n        if (!this.messageContainer)\r\n            return;\r\n\r\n        const el = this.messageContainer.nativeElement;\r\n        el.scrollTop = el.scrollHeight;\r\n    }\r\n\r\n    mentionsMe(message: ChatMessage) {\r\n        if (!this.currentUser)\r\n            return false;\r\n\r\n        if (message.message.includes(`@${this.currentUser.username}`))\r\n            return true;\r\n\r\n        return false;\r\n    }\r\n}\r\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Observable, Subject, BehaviorSubject, Subscription } from 'rxjs';
2
2
  import { publish } from 'rxjs/operators';
3
- import { Component, Input, ViewChild, Pipe, Output, HostBinding, NgModule, ElementRef, Directive, NgZone, ContentChild, TemplateRef, Injectable, Inject } from '@angular/core';
3
+ import { Component, Input, ViewChild, Pipe, Output, HostBinding, NgModule, ElementRef, ViewChildren, Directive, NgZone, ContentChild, TemplateRef, Injectable, Inject } from '@angular/core';
4
4
  import { marked, Renderer } from 'marked';
5
5
  import { sanitize } from 'dompurify';
6
6
  import { DomSanitizer } from '@angular/platform-browser';
@@ -7853,7 +7853,8 @@ LiveMessageComponent.propDecorators = {
7853
7853
  };
7854
7854
 
7855
7855
  class CommentComponent {
7856
- constructor() {
7856
+ constructor(elementRef) {
7857
+ this.elementRef = elementRef;
7857
7858
  this._reported = new Subject();
7858
7859
  this._selected = new Subject();
7859
7860
  this._liked = new Subject();
@@ -7873,6 +7874,9 @@ class CommentComponent {
7873
7874
  this._editEnded = new Subject();
7874
7875
  this._edited = new Subject();
7875
7876
  }
7877
+ get element() {
7878
+ return this.elementRef.nativeElement;
7879
+ }
7876
7880
  ngOnInit() {
7877
7881
  let maxTime = 100;
7878
7882
  let minTime = 0;
@@ -7972,6 +7976,9 @@ CommentComponent.decorators = [
7972
7976
  styles: ["@-webkit-keyframes comment-appear{0%{transform:translate(6em)}to{transform:translate(0)}}@keyframes comment-appear{0%{transform:translate(6em)}to{transform:translate(0)}}:host{display:flex;flex-direction:column;position:relative;padding:.5em;visibility:hidden}:host.new{visibility:visible;-webkit-animation-name:comment-appear;animation-name:comment-appear;-webkit-animation-duration:.25s;animation-duration:.25s;-webkit-animation-fill-mode:both;animation-fill-mode:both}:host.highlighted{background:#00223a;outline:2px solid #003277}:host.visible{visibility:visible}:host:hover{background:#eee}:host .message-content .content{margin-left:60px;margin-right:.5em}:host .message-content .attachments-row{margin-top:15px;display:flex;gap:10px}:host .message-content .attachments-row img{border-radius:10px;width:300px;max-width:100%;max-height:20em;-o-object-fit:cover;object-fit:cover}:host.abbreviated .message-content .content{text-overflow:ellipsis;overflow-y:hidden}:host .actions{display:flex;padding-right:10px;margin-left:60px;align-items:center}:host .actions button,banta-timestamp{color:#666;flex-shrink:0}banta-timestamp{font-size:10pt;margin-left:1em;text-align:right}.user{position:relative;margin:1em 0 0;display:flex;align-items:center;flex-wrap:wrap}.user .user-1,.user .user-2{display:flex;flex-wrap:nowrap;align-items:center;min-width:0}.user .user-2{margin:1em 0}.user .user-identity{display:flex;flex-direction:column;min-width:0}.user .display-name,.user .username{z-index:1;position:relative;padding:0 0 0 1em;font-size:10pt;color:#000;margin:0 auto 0 0;display:block;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;max-width:100%;white-space:nowrap;text-overflow:ellipsis;overflow:hidden;flex-shrink:1;flex-grow:0;min-width:0}.user .display-name.username.username.username,.user .username.username.username.username{color:#666}.avatar{height:48px;width:48px;background-position:50%;background-size:cover;background-color:#333;border-radius:100%;flex-shrink:0;flex-grow:0}.counted-action{display:flex;align-items:center}.counted-action.active .count-indicator,.counted-action.active button{color:#00a5ff}.counted-action button .count-indicator{margin-left:.5em}.count-indicator{font-size:9pt;padding:0 0 0 3px;color:#666}:host-context(.mat-dark-theme) .count-indicator{border-color:#333}:host-context(.mat-dark-theme):hover{background:#060606}.user-tag,:host-context(.mat-dark-theme) .user .display-name,:host-context(.mat-dark-theme) .user .username{color:#fff}.user-tag{text-transform:uppercase;font-size:12px;border:1px solid #b27373;background:#7a412b;padding:3px 5px;margin:0 .5em 0 1em;border-radius:3px}.spacer{flex-shrink:1;flex-grow:1}ul.message-facts{margin:0;padding:0;color:#666}ul.message-facts li{list-style-type:none;border-left:1px solid #666;font-size:10pt;padding-left:.5em;margin-left:.5em;margin-top:.5em}ul.message-facts li:first-child{border-left:1px solid transparent;margin-left:0;padding-left:0}@media (max-width:400px){.avatar{height:32px;width:32px}:host .actions{margin-left:0;margin-top:.5em}:host .message-content .content{margin-left:44px;margin-right:.5em}}:host-context(.banta-mobile) .avatar{height:32px;width:32px}:host-context(.banta-mobile) :host .actions{margin-left:0;margin-top:.5em}:host-context(.banta-mobile) :host .message-content .content{margin-left:44px;margin-right:.5em}.card-attachment a{display:flex;align-items:flex-start;gap:1em;width:100%;border:1px solid #666;border-radius:4px;padding:1em;box-sizing:border-box;background-color:#191919}.card-attachment a img{width:300px;aspect-ratio:16/9;-o-object-fit:cover;object-fit:cover;border-radius:10px}.card-attachment a h1{margin:0;font-size:30px}"]
7973
7977
  },] }
7974
7978
  ];
7979
+ CommentComponent.ctorParameters = () => [
7980
+ { type: ElementRef }
7981
+ ];
7975
7982
  CommentComponent.propDecorators = {
7976
7983
  maxLength: [{ type: Input }],
7977
7984
  isNew: [{ type: HostBinding, args: ['class.new',] }],
@@ -8029,6 +8036,21 @@ class CommentViewComponent {
8029
8036
  this.olderMessages = [];
8030
8037
  this.sortOrderChanged = new Subject();
8031
8038
  }
8039
+ get comments() {
8040
+ return Array.from(this.commentsQuery);
8041
+ }
8042
+ /**
8043
+ * Get the CommentComponent instantiated for the given ChatMessage,
8044
+ * if it exists in the current view. Note that messages which are not
8045
+ * currently shown to the user will not return a CommentComponent.
8046
+ * @param message
8047
+ * @returns
8048
+ */
8049
+ getCommentComponentForMessage(message) {
8050
+ if (!message)
8051
+ throw new Error(`You must pass a valid ChatMessage`);
8052
+ return this.comments.find(x => { var _a; return ((_a = x.message) === null || _a === void 0 ? void 0 : _a.id) === message.id; });
8053
+ }
8032
8054
  get selected() {
8033
8055
  return this._selected;
8034
8056
  }
@@ -8209,9 +8231,18 @@ class CommentViewComponent {
8209
8231
  return maxPagingCursor;
8210
8232
  }
8211
8233
  sortMessages() {
8212
- this.messages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));
8213
- this.olderMessages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));
8214
- this.newMessages.sort((a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? 1 : -1));
8234
+ if (!this.source)
8235
+ return;
8236
+ let sorter;
8237
+ if (this.source.sortOrder === CommentsOrder.LIKES)
8238
+ sorter = (a, b) => b.likes - a.likes;
8239
+ else if (this.source.sortOrder === CommentsOrder.NEWEST)
8240
+ sorter = (a, b) => (b.sentAt - a.sentAt) * (this.newestLast ? -1 : 1);
8241
+ else if (this.source.sortOrder === CommentsOrder.OLDEST)
8242
+ sorter = (a, b) => (a.sentAt - b.sentAt) * (this.newestLast ? -1 : 1);
8243
+ this.messages.sort(sorter);
8244
+ this.olderMessages.sort(sorter);
8245
+ this.newMessages.sort(sorter);
8215
8246
  }
8216
8247
  messageReceived(message) {
8217
8248
  this.addMessage(message);
@@ -8260,6 +8291,7 @@ CommentViewComponent.propDecorators = {
8260
8291
  showEmptyState: [{ type: Input }],
8261
8292
  allowReplies: [{ type: Input }],
8262
8293
  customMenuItems: [{ type: Input }],
8294
+ commentsQuery: [{ type: ViewChildren, args: [CommentComponent,] }],
8263
8295
  fixedHeight: [{ type: Input }, { type: HostBinding, args: ['class.fixed-height',] }],
8264
8296
  selectedMessage: [{ type: Input }],
8265
8297
  selected: [{ type: Output }],
@@ -8386,6 +8418,9 @@ class BantaCommentsComponent {
8386
8418
  }
8387
8419
  });
8388
8420
  }
8421
+ get element() {
8422
+ return this.elementRef.nativeElement;
8423
+ }
8389
8424
  handleBackendExceptionAsAlert(e, prefix = '') {
8390
8425
  try {
8391
8426
  this.handleBackendException(e, prefix);
@@ -8424,6 +8459,10 @@ class BantaCommentsComponent {
8424
8459
  }
8425
8460
  get isMobileSized() { return this.width < 500; }
8426
8461
  ngAfterViewInit() {
8462
+ this.threadViewQuery.changes.subscribe(changes => {
8463
+ console.log(`i see changes...`);
8464
+ console.dir(changes);
8465
+ });
8427
8466
  if (typeof window !== 'undefined') {
8428
8467
  let callback = () => {
8429
8468
  let size = this.elementRef.nativeElement.getBoundingClientRect();
@@ -8432,8 +8471,17 @@ class BantaCommentsComponent {
8432
8471
  this.height = size.height;
8433
8472
  });
8434
8473
  };
8435
- this.resizeObserver = new ResizeObserver(callback);
8436
- this.resizeObserver.observe(this.elementRef.nativeElement);
8474
+ if (typeof ResizeObserver !== 'undefined') {
8475
+ this.resizeObserver = new ResizeObserver(callback);
8476
+ this.resizeObserver.observe(this.elementRef.nativeElement);
8477
+ }
8478
+ else {
8479
+ // One check is better than nothing!
8480
+ setTimeout(() => {
8481
+ this.width = this.elementRef.nativeElement.clientWidth;
8482
+ this.height = this.elementRef.nativeElement.clientHeight;
8483
+ }, 1000);
8484
+ }
8437
8485
  callback();
8438
8486
  }
8439
8487
  }
@@ -8444,6 +8492,8 @@ class BantaCommentsComponent {
8444
8492
  }
8445
8493
  setSourceFromTopicID(topicID) {
8446
8494
  return __awaiter(this, void 0, void 0, function* () {
8495
+ if (typeof window === 'undefined')
8496
+ return;
8447
8497
  setTimeout(() => __awaiter(this, void 0, void 0, function* () {
8448
8498
  console.log(`[banta-comments] Subscribing to topic source '${topicID}'`);
8449
8499
  this.source = yield this.backend.getSourceForTopic(topicID, { sortOrder: this.sortOrder });
@@ -8484,6 +8534,30 @@ class BantaCommentsComponent {
8484
8534
  this._loadingTimer = setInterval(() => this.updateLoading(), 1000);
8485
8535
  });
8486
8536
  }
8537
+ /**
8538
+ * Attempts to find the CommentComponent that corresponds to the given ChatMessage.
8539
+ * The ChatMessage could be a top-level message in this conversation, or a reply,
8540
+ * but note that a CommentComponent will only exist for a reply if the user has
8541
+ * the relevant reply thread open.
8542
+ * @param message The message to look for
8543
+ */
8544
+ getCommentComponentForMessage(message) {
8545
+ var _a;
8546
+ if (message.parentMessageId)
8547
+ return (_a = this.threadView) === null || _a === void 0 ? void 0 : _a.getCommentComponentForMessage(message);
8548
+ else
8549
+ return this.commentView.getCommentComponentForMessage(message);
8550
+ }
8551
+ /**
8552
+ * Access the CommentViewComponent corresponding to the currently open reply thread.
8553
+ * This is not the top level comments, but instead the reply thread that the user has
8554
+ * opened (only one set of replies can be open at a time).
8555
+ *
8556
+ * For details about what CommentViewComponent affords you, see the commentView property.
8557
+ */
8558
+ get threadView() {
8559
+ return Array.from(this.threadViewQuery).filter(x => x !== this.commentView)[0];
8560
+ }
8487
8561
  updateLoading() {
8488
8562
  var _a, _b;
8489
8563
  if (((_a = this.source) === null || _a === void 0 ? void 0 : _a.state) && ((_b = this.source) === null || _b === void 0 ? void 0 : _b.state) !== 'connecting') {
@@ -8805,7 +8879,7 @@ class BantaCommentsComponent {
8805
8879
  BantaCommentsComponent.decorators = [
8806
8880
  { type: Component, args: [{
8807
8881
  selector: 'banta-comments',
8808
- template: "<ng-container *ngIf=\"loading\">\r\n <div class=\"loading-screen\" [class.visible]=\"showLoadingScreen\">\r\n <h1>Loading...</h1>\r\n <div>\r\n <mat-spinner [diameter]=\"300\" [strokeWidth]=\"2\"></mat-spinner>\r\n </div>\r\n\r\n <p class=\"loading-message\" [class.visible]=\"loadingMessageVisible\">{{loadingMessage}}</p>\r\n </div>\r\n</ng-container>\r\n<ng-container *ngIf=\"!loading\">\r\n <div class=\"focused\" [class.visible]=\"selectedMessageVisible\" *ngIf=\"selectedMessage && !useInlineReplies\">\r\n\r\n <div>\r\n <a mat-button href=\"javascript:;\" (click)=\"unselectMessage()\">\r\n <mat-icon>arrow_back</mat-icon>\r\n Latest Comments\r\n </a>\r\n </div>\r\n\r\n <banta-comment\r\n [message]=\"selectedMessage\"\r\n [liking]=\"selectedMessage.transientState.liking\"\r\n [mine]=\"user?.id === selectedMessage.user?.id\"\r\n [permissions]=\"source?.permissions\"\r\n [showReplyAction]=\"false\"\r\n [editing]=\"selectedMessage.transientState.editing\"\r\n [maxLength]=\"maxCommentLength\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (editStarted)=\"startEditing(selectedMessage)\"\r\n (editEnded)=\"selectedMessage.transientState.editing = false\"\r\n (edited)=\"saveEdit(selectedMessage, $event)\"\r\n (userSelected)=\"selectMessageUser(selectedMessage)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (liked)=\"likeMessage(source, selectedMessage)\"\r\n (unliked)=\"unlikeMessage(source, selectedMessage)\"\r\n (reported)=\"reportMessage(selectedMessage)\"\r\n (selected)=\"selectMessage(selectedMessage)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage(selectedMessage)\"\r\n ></banta-comment>\r\n\r\n <div class=\"replies\">\r\n\r\n <ng-container *ngIf=\"!selectedMessageThread\">\r\n <div class=\"loading\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n </ng-container>\r\n\r\n <ng-container *ngIf=\"selectedMessageThread\">\r\n <banta-comment-view\r\n [source]=\"selectedMessageThread\"\r\n [allowReplies]=\"false\"\r\n [fixedHeight]=\"false\"\r\n [showEmptyState]=\"false\"\r\n [newestLast]=\"true\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (liked)=\"likeMessage(selectedMessageThread, $event)\"\r\n (unliked)=\"unlikeMessage(selectedMessageThread, $event)\"\r\n (messageEdited)=\"editMessage(selectedMessageThread, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage($event)\"\r\n [customMenuItems]=\"customMenuItems\"\r\n ></banta-comment-view>\r\n\r\n <banta-comment-field\r\n [sendLabel]=\"replyLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n (signInSelected)=\"showSignIn()\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n [source]=\"selectedMessageThread\"\r\n [maxLength]=\"maxCommentLength\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [signInLabel]=\"signInLabel\"\r\n [permissionDeniedLabel]=\"source?.permissions?.canPostErrorMessage || permissionDeniedLabel\"\r\n (permissionDeniedError)=\"handlePermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [user]=\"user\"\r\n [label]=\"postReplyLabel\"\r\n [submit]=\"sendReply\"\r\n [allowAttachments]=\"allowAttachments\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n >\r\n <ng-container *ngTemplateOutlet=\"sendReplyOptionsTemplate\"></ng-container>\r\n </banta-comment-field>\r\n </ng-container>\r\n </div>\r\n </div>\r\n\r\n <div class=\"main\" [class.hidden]=\"selectedMessage && !useInlineReplies\">\r\n <banta-comment-field\r\n [source]=\"source\"\r\n [user]=\"user\"\r\n [sendLabel]=\"sendLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [signInLabel]=\"signInLabel\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n [label]=\"postCommentLabel\"\r\n [maxLength]=\"maxCommentLength\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n (signInSelected)=\"showSignIn()\"\r\n [permissionDeniedLabel]=\"source?.permissions?.canPostErrorMessage || permissionDeniedLabel\"\r\n (permissionDeniedError)=\"handlePermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [submit]=\"sendMessage\"\r\n [allowAttachments]=\"allowAttachments\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n >\r\n \r\n </banta-comment-field>\r\n\r\n <banta-comment-sort\r\n [(sort)]=\"sortOrder\"></banta-comment-sort>\r\n\r\n <banta-comment-view\r\n [class.faded]=\"selectedMessage && !useInlineReplies\"\r\n [source]=\"source\"\r\n [fixedHeight]=\"fixedHeight\"\r\n [maxMessages]=\"maxMessages\"\r\n [maxVisibleMessages]=\"maxVisibleMessages\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n [customMenuItems]=\"customMenuItems\"\r\n (userSelected)=\"selectMessageUser($event)\"\r\n (sortOrderChanged)=\"sortOrder = $event\"\r\n (selected)=\"selectMessage($event)\"\r\n (liked)=\"likeMessage(source, $event)\"\r\n (unliked)=\"unlikeMessage(source, $event)\"\r\n (messageEdited)=\"editMessage(source, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n [selectedMessage]=\"selectedMessage\"\r\n (deleted)=\"deleteMessage($event)\"\r\n >\r\n <div class=\"inline-replies\">\r\n <div class=\"focused\" [class.visible]=\"selectedMessageVisible\" *ngIf=\"selectedMessage\">\r\n <div class=\"replies\">\r\n \r\n <ng-container *ngIf=\"!selectedMessageThread\">\r\n <div class=\"loading\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n </ng-container>\r\n \r\n <ng-container *ngIf=\"selectedMessageThread\">\r\n <banta-comment-view\r\n [source]=\"selectedMessageThread\"\r\n [allowReplies]=\"false\"\r\n [fixedHeight]=\"false\"\r\n [showEmptyState]=\"false\"\r\n [newestLast]=\"true\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (liked)=\"likeMessage(selectedMessageThread, $event)\"\r\n (unliked)=\"unlikeMessage(selectedMessageThread, $event)\"\r\n (messageEdited)=\"editMessage(selectedMessageThread, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage($event)\"\r\n ></banta-comment-view>\r\n \r\n <banta-comment-field\r\n [sendLabel]=\"replyLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n (signInSelected)=\"showSignIn()\"\r\n [maxLength]=\"maxCommentLength\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n [source]=\"selectedMessageThread\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [signInLabel]=\"signInLabel\"\r\n [permissionDeniedLabel]=\"source?.permissions?.canPostErrorMessage || permissionDeniedLabel\"\r\n (permissionDeniedError)=\"handlePermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [user]=\"user\"\r\n [label]=\"postReplyLabel\"\r\n [submit]=\"sendReply\"\r\n [allowAttachments]=\"allowAttachments\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n >\r\n <ng-container *ngTemplateOutlet=\"sendReplyOptionsTemplate\"></ng-container>\r\n </banta-comment-field>\r\n </ng-container>\r\n </div>\r\n </div> \r\n </div>\r\n </banta-comment-view>\r\n </div>\r\n</ng-container>\r\n",
8882
+ template: "<ng-container *ngIf=\"loading\">\r\n <div class=\"loading-screen\" [class.visible]=\"showLoadingScreen\">\r\n <h1>Loading...</h1>\r\n <div>\r\n <mat-spinner [diameter]=\"300\" [strokeWidth]=\"2\"></mat-spinner>\r\n </div>\r\n\r\n <p class=\"loading-message\" [class.visible]=\"loadingMessageVisible\">{{loadingMessage}}</p>\r\n </div>\r\n</ng-container>\r\n<ng-container *ngIf=\"!loading\">\r\n <div class=\"focused\" [class.visible]=\"selectedMessageVisible\" *ngIf=\"selectedMessage && !useInlineReplies\">\r\n\r\n <div>\r\n <a mat-button href=\"javascript:;\" (click)=\"unselectMessage()\">\r\n <mat-icon>arrow_back</mat-icon>\r\n Latest Comments\r\n </a>\r\n </div>\r\n\r\n <banta-comment\r\n [message]=\"selectedMessage\"\r\n [liking]=\"selectedMessage.transientState.liking\"\r\n [mine]=\"user?.id === selectedMessage.user?.id\"\r\n [permissions]=\"source?.permissions\"\r\n [showReplyAction]=\"false\"\r\n [editing]=\"selectedMessage.transientState.editing\"\r\n [maxLength]=\"maxCommentLength\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (editStarted)=\"startEditing(selectedMessage)\"\r\n (editEnded)=\"selectedMessage.transientState.editing = false\"\r\n (edited)=\"saveEdit(selectedMessage, $event)\"\r\n (userSelected)=\"selectMessageUser(selectedMessage)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (liked)=\"likeMessage(source, selectedMessage)\"\r\n (unliked)=\"unlikeMessage(source, selectedMessage)\"\r\n (reported)=\"reportMessage(selectedMessage)\"\r\n (selected)=\"selectMessage(selectedMessage)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage(selectedMessage)\"\r\n ></banta-comment>\r\n\r\n <div class=\"replies\">\r\n\r\n <ng-container *ngIf=\"!selectedMessageThread\">\r\n <div class=\"loading\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n </ng-container>\r\n\r\n <ng-container *ngIf=\"selectedMessageThread\">\r\n <banta-comment-view\r\n class=\"replies\"\r\n #threadView\r\n [source]=\"selectedMessageThread\"\r\n [allowReplies]=\"false\"\r\n [fixedHeight]=\"false\"\r\n [showEmptyState]=\"false\"\r\n [newestLast]=\"true\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (liked)=\"likeMessage(selectedMessageThread, $event)\"\r\n (unliked)=\"unlikeMessage(selectedMessageThread, $event)\"\r\n (messageEdited)=\"editMessage(selectedMessageThread, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage($event)\"\r\n [customMenuItems]=\"customMenuItems\"\r\n ></banta-comment-view>\r\n\r\n <banta-comment-field\r\n [sendLabel]=\"replyLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n (signInSelected)=\"showSignIn()\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n [source]=\"selectedMessageThread\"\r\n [maxLength]=\"maxCommentLength\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [signInLabel]=\"signInLabel\"\r\n [permissionDeniedLabel]=\"source?.permissions?.canPostErrorMessage || permissionDeniedLabel\"\r\n (permissionDeniedError)=\"handlePermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [user]=\"user\"\r\n [label]=\"postReplyLabel\"\r\n [submit]=\"sendReply\"\r\n [allowAttachments]=\"allowAttachments\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n >\r\n <ng-container *ngTemplateOutlet=\"sendReplyOptionsTemplate\"></ng-container>\r\n </banta-comment-field>\r\n </ng-container>\r\n </div>\r\n </div>\r\n\r\n <div class=\"main\" [class.hidden]=\"selectedMessage && !useInlineReplies\">\r\n <banta-comment-field\r\n [source]=\"source\"\r\n [user]=\"user\"\r\n [sendLabel]=\"sendLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [signInLabel]=\"signInLabel\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n [label]=\"postCommentLabel\"\r\n [maxLength]=\"maxCommentLength\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n (signInSelected)=\"showSignIn()\"\r\n [permissionDeniedLabel]=\"source?.permissions?.canPostErrorMessage || permissionDeniedLabel\"\r\n (permissionDeniedError)=\"handlePermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [submit]=\"sendMessage\"\r\n [allowAttachments]=\"allowAttachments\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n >\r\n \r\n </banta-comment-field>\r\n\r\n <banta-comment-sort\r\n [(sort)]=\"sortOrder\"></banta-comment-sort>\r\n\r\n <banta-comment-view\r\n #commentView\r\n [class.faded]=\"selectedMessage && !useInlineReplies\"\r\n [source]=\"source\"\r\n [fixedHeight]=\"fixedHeight\"\r\n [maxMessages]=\"maxMessages\"\r\n [maxVisibleMessages]=\"maxVisibleMessages\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n [customMenuItems]=\"customMenuItems\"\r\n (userSelected)=\"selectMessageUser($event)\"\r\n (sortOrderChanged)=\"sortOrder = $event\"\r\n (selected)=\"selectMessage($event)\"\r\n (liked)=\"likeMessage(source, $event)\"\r\n (unliked)=\"unlikeMessage(source, $event)\"\r\n (messageEdited)=\"editMessage(source, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n [selectedMessage]=\"selectedMessage\"\r\n (deleted)=\"deleteMessage($event)\"\r\n >\r\n <div class=\"inline-replies\">\r\n <div class=\"focused\" [class.visible]=\"selectedMessageVisible\" *ngIf=\"selectedMessage\">\r\n <div class=\"replies\">\r\n \r\n <ng-container *ngIf=\"!selectedMessageThread\">\r\n <div class=\"loading\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n </ng-container>\r\n \r\n <ng-container *ngIf=\"selectedMessageThread\">\r\n <banta-comment-view\r\n [source]=\"selectedMessageThread\"\r\n [allowReplies]=\"false\"\r\n [fixedHeight]=\"false\"\r\n [showEmptyState]=\"false\"\r\n [newestLast]=\"true\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (liked)=\"likeMessage(selectedMessageThread, $event)\"\r\n (unliked)=\"unlikeMessage(selectedMessageThread, $event)\"\r\n (messageEdited)=\"editMessage(selectedMessageThread, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage($event)\"\r\n ></banta-comment-view>\r\n \r\n <banta-comment-field\r\n [sendLabel]=\"replyLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n (signInSelected)=\"showSignIn()\"\r\n [maxLength]=\"maxCommentLength\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n [source]=\"selectedMessageThread\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [signInLabel]=\"signInLabel\"\r\n [permissionDeniedLabel]=\"source?.permissions?.canPostErrorMessage || permissionDeniedLabel\"\r\n (permissionDeniedError)=\"handlePermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [user]=\"user\"\r\n [label]=\"postReplyLabel\"\r\n [submit]=\"sendReply\"\r\n [allowAttachments]=\"allowAttachments\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n >\r\n <ng-container *ngTemplateOutlet=\"sendReplyOptionsTemplate\"></ng-container>\r\n </banta-comment-field>\r\n </ng-container>\r\n </div>\r\n </div> \r\n </div>\r\n </banta-comment-view>\r\n </div>\r\n</ng-container>\r\n",
8809
8883
  styles: [":host{display:flex;flex-direction:column}@-webkit-keyframes select-comment{0%{transform:scale(1.15)}to{transform:scale(1)}}@keyframes select-comment{0%{transform:scale(1.15)}to{transform:scale(1)}}.focused{-webkit-animation-name:select-comment;animation-name:select-comment;-webkit-animation-duration:.4s;animation-duration:.4s;-webkit-animation-fill-mode:both;animation-fill-mode:both}.focused .replies{margin-top:1em;margin-left:2em;border-left:2px solid #333;padding-left:2em}banta-comment-view{opacity:1;transition:opacity .4s ease-in-out}banta-comment-view.faded{opacity:.25}.loading{display:block;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;margin:0 auto;min-height:16em}.main.hidden{display:none}.loading-screen{text-align:center;opacity:0;transition:opacity .25s ease-in-out}.loading-screen.visible{opacity:1}.loading-screen h1{font-weight:100}.loading-screen mat-spinner{margin:5em auto}.loading-screen .loading-message{opacity:0;transition:opacity .25s ease-in-out;width:500px;max-width:100%;margin:0 auto}.loading-screen .loading-message.visible{opacity:1}banta-comment-sort{margin:0 0 0 auto;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;display:block}.inline-replies{margin-left:4em}@media (max-width:500px){.focused .replies{margin-left:0}.inline-replies{margin-left:1em}.focused .replies{padding-left:.5em}banta-comment-sort{margin:0;width:100%}}:host-context(.banta-mobile) .focused .replies{margin-left:0}:host-context(.banta-mobile) .inline-replies{margin-left:1em}:host-context(.banta-mobile) .focused .replies{padding-left:.5em}:host-context(.banta-mobile) banta-comment-sort{margin:0;width:100%}"]
8810
8884
  },] }
8811
8885
  ];
@@ -8823,6 +8897,8 @@ BantaCommentsComponent.propDecorators = {
8823
8897
  maxCommentLength: [{ type: Input }],
8824
8898
  loadingMessages: [{ type: Input }],
8825
8899
  useInlineReplies: [{ type: Input }],
8900
+ commentView: [{ type: ViewChild, args: ['commentView',] }],
8901
+ threadViewQuery: [{ type: ViewChildren, args: [CommentViewComponent,] }],
8826
8902
  signInLabel: [{ type: Input }],
8827
8903
  sendLabel: [{ type: Input }],
8828
8904
  replyLabel: [{ type: Input }],