@banta/sdk 5.8.3 → 5.8.4
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/esm2022/lib/comments/comment-view/comment-view.component.mjs +29 -16
- package/fesm2022/banta-sdk.mjs +28 -15
- package/fesm2022/banta-sdk.mjs.map +1 -1
- package/lib/banta/banta.component.d.ts +1 -1
- package/lib/chat-source.d.ts +3 -3
- package/lib/comments/banta-comments/banta-comments.component.d.ts +1 -1
- package/lib/comments/comment-view/comment-view.component.d.ts +7 -3
- package/lib/static-chat-source.d.ts +1 -1
- package/package.json +1 -1
|
@@ -9,6 +9,8 @@ import * as i3 from "@angular/material/icon";
|
|
|
9
9
|
import * as i4 from "@angular/material/button";
|
|
10
10
|
import * as i5 from "@angular/material/progress-spinner";
|
|
11
11
|
import * as i6 from "../comment/comment.component";
|
|
12
|
+
const DEFAULT_MAX_MESSAGES = 2000;
|
|
13
|
+
const DEFAULT_MAX_VISIBLE_MESSAGES = 200;
|
|
12
14
|
export class CommentViewComponent {
|
|
13
15
|
constructor(backend, elementRef) {
|
|
14
16
|
this.backend = backend;
|
|
@@ -28,7 +30,7 @@ export class CommentViewComponent {
|
|
|
28
30
|
* While this is called "new" messages, it really represents the messages that would be visible *at the beginning
|
|
29
31
|
* of the sort order*, which can be flipped by the newestLast feature (used for replies mode).
|
|
30
32
|
*
|
|
31
|
-
* So, when newestLast is false, regardless of the current sortOrder, newMessages are conceptually
|
|
33
|
+
* So, when newestLast is false (top-level comments), regardless of the current sortOrder, newMessages are conceptually
|
|
32
34
|
* *above* the visible set of messages.
|
|
33
35
|
*
|
|
34
36
|
* When newestLast is true (as in replies mode), regardless of the current sortOrder, newMessages are conceptually
|
|
@@ -46,10 +48,6 @@ export class CommentViewComponent {
|
|
|
46
48
|
* *above* the visible set of messages.
|
|
47
49
|
*/
|
|
48
50
|
this.olderMessages = [];
|
|
49
|
-
//#endregion
|
|
50
|
-
//#region Inputs
|
|
51
|
-
this.maxMessages = 2000;
|
|
52
|
-
this.maxVisibleMessages = 200;
|
|
53
51
|
this.newestLast = false;
|
|
54
52
|
this.holdNewMessages = false;
|
|
55
53
|
this.showEmptyState = true;
|
|
@@ -89,6 +87,10 @@ export class CommentViewComponent {
|
|
|
89
87
|
set source(value) { this.setSource(value); }
|
|
90
88
|
get previousMessages() { return this.newestLast ? this.olderMessages : this.newMessages; }
|
|
91
89
|
get nextMessages() { return this.newestLast ? this.newMessages : this.olderMessages; }
|
|
90
|
+
set maxMessages(value) { this._maxMessages = value; }
|
|
91
|
+
get maxMessages() { return this._maxMessages ?? DEFAULT_MAX_MESSAGES; }
|
|
92
|
+
set maxVisibleMessages(value) { this._maxVisibleMessages = value; }
|
|
93
|
+
get maxVisibleMessages() { return this._maxVisibleMessages ?? DEFAULT_MAX_VISIBLE_MESSAGES; }
|
|
92
94
|
get comments() { return Array.from(this.commentsQuery); }
|
|
93
95
|
//#endregion
|
|
94
96
|
/**
|
|
@@ -111,16 +113,15 @@ export class CommentViewComponent {
|
|
|
111
113
|
console.error(`Error while loading message in context: Failed to find message ${message.id}, maybe it was deleted!`);
|
|
112
114
|
return false;
|
|
113
115
|
}
|
|
114
|
-
let pageSize = this.maxVisibleMessages;
|
|
115
116
|
let items = [].concat(this.olderMessages, this.messages, this.newMessages);
|
|
116
117
|
let index = items.findIndex(x => x.id === message.id);
|
|
117
118
|
if (index < 0) {
|
|
118
119
|
console.error(`Error while loading message in context: Message was not present in message list!`);
|
|
119
120
|
return false;
|
|
120
121
|
}
|
|
121
|
-
let startIndex = Math.max(0, index -
|
|
122
|
+
let startIndex = Math.max(0, index - this.maxVisibleMessages / 2);
|
|
122
123
|
this.newMessages = items.splice(0, startIndex);
|
|
123
|
-
this.messages = items.splice(0,
|
|
124
|
+
this.messages = items.splice(0, this.maxVisibleMessages);
|
|
124
125
|
this.olderMessages = items;
|
|
125
126
|
this.isViewingMore = true;
|
|
126
127
|
}
|
|
@@ -358,12 +359,14 @@ export class CommentViewComponent {
|
|
|
358
359
|
return this.previousMessages.length > 0;
|
|
359
360
|
}
|
|
360
361
|
get pageSize() {
|
|
361
|
-
return Math.min(20, this.maxVisibleMessages);
|
|
362
|
+
return Math.min(20, this.maxVisibleMessages || 20);
|
|
362
363
|
}
|
|
363
364
|
async showPrevious() {
|
|
364
365
|
this.isViewingMore = true;
|
|
365
366
|
let nextPageSize = this.pageSize;
|
|
366
367
|
this.isLoadingMore = false;
|
|
368
|
+
if (isNaN(nextPageSize))
|
|
369
|
+
throw new Error(`Not safe to load more with NaN page size`);
|
|
367
370
|
if (this.previousMessages.length > 0) {
|
|
368
371
|
const storedMessages = this.previousMessages.splice(Math.max(0, this.previousMessages.length - nextPageSize), nextPageSize);
|
|
369
372
|
this.messages = [...storedMessages, ...this.messages];
|
|
@@ -431,6 +434,8 @@ export class CommentViewComponent {
|
|
|
431
434
|
async showNext() {
|
|
432
435
|
this.isViewingMore = true;
|
|
433
436
|
let nextPageSize = this.pageSize;
|
|
437
|
+
if (isNaN(nextPageSize))
|
|
438
|
+
throw new Error(`Not safe to load more with NaN page size`);
|
|
434
439
|
this.isLoadingMore = false;
|
|
435
440
|
if (this.nextMessages.length > 0) {
|
|
436
441
|
const storedMessages = this.nextMessages.splice(0, nextPageSize);
|
|
@@ -471,6 +476,8 @@ export class CommentViewComponent {
|
|
|
471
476
|
async showMore() {
|
|
472
477
|
this.isViewingMore = true;
|
|
473
478
|
let nextPageSize = this.pageSize;
|
|
479
|
+
if (isNaN(nextPageSize))
|
|
480
|
+
throw new Error(`Not safe to load more with NaN page size`);
|
|
474
481
|
this.isLoadingMore = false;
|
|
475
482
|
if (this.olderMessages.length > 0) {
|
|
476
483
|
const storedMessages = this.olderMessages.splice(0, nextPageSize);
|
|
@@ -514,14 +521,20 @@ export class CommentViewComponent {
|
|
|
514
521
|
// Extract the messages that do not fit in the maxVisibleMessages buffer.
|
|
515
522
|
if (this.messages.length > this.maxVisibleMessages) {
|
|
516
523
|
let overflow = [];
|
|
517
|
-
|
|
524
|
+
// Move overflowing messages into newMessages.
|
|
525
|
+
// Regardless of the order (newestLast), newMessages represents the direction that is being loaded,
|
|
526
|
+
// since it's definition depends on that order.
|
|
527
|
+
if (this.newestLast) {
|
|
518
528
|
overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);
|
|
519
|
-
|
|
529
|
+
this.newMessages = overflow.concat(this.newMessages);
|
|
530
|
+
this.newMessages.splice(this.maxMessages - this.maxVisibleMessages, this.newMessages.length);
|
|
531
|
+
}
|
|
532
|
+
else {
|
|
520
533
|
overflow = this.messages.splice(0, this.messages.length - this.maxVisibleMessages);
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
534
|
+
this.newMessages.push(...overflow);
|
|
535
|
+
if (this.newMessages.length > this.maxMessages - this.maxVisibleMessages)
|
|
536
|
+
this.newMessages.splice(0, this.newMessages.length - (this.maxMessages - this.maxVisibleMessages));
|
|
537
|
+
}
|
|
525
538
|
}
|
|
526
539
|
}
|
|
527
540
|
addMessage(message) {
|
|
@@ -712,4 +725,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImpor
|
|
|
712
725
|
type: ViewChild,
|
|
713
726
|
args: ['messageContainer']
|
|
714
727
|
}] } });
|
|
715
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment-view.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/comments/comment-view/comment-view.component.ts","../../../../../../projects/sdk/src/lib/comments/comment-view/comment-view.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output, HostBinding, ViewChildren, QueryList } from \"@angular/core\";\r\nimport { User, ChatMessage, CommentsOrder, 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\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        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    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, 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    @Input() maxMessages = 2000;\r\n    @Input() maxVisibleMessages: number = 200;\r\n    @Input() newestLast = false;\r\n    @Input() holdNewMessages = false;\r\n    @Input() showEmptyState = true;\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    //#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 _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 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 pageSize = this.maxVisibleMessages;\r\n        let items = [].concat(this.olderMessages, this.messages, this.newMessages);\r\n        let index = items.findIndex(x => x.id === message.id);\r\n\r\n        if (index < 0) {\r\n            console.error(`Error while loading message in context: Message was not present in message list!`);\r\n            return false;\r\n        }\r\n\r\n        let startIndex = Math.max(0, index - pageSize / 2);        \r\n        this.newMessages = items.splice(0, startIndex);\r\n        this.messages = items.splice(0, pageSize);\r\n        this.olderMessages = items;\r\n        this.isViewingMore = true;\r\n    }\r\n\r\n    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    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\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\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);\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 (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        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        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            if (this.newestLast)\r\n                overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);\r\n            else\r\n                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.newMessages = overflow.concat(this.newMessages);\r\n            this.newMessages.splice(this.maxMessages - this.maxVisibleMessages, this.newMessages.length);\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    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    <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                \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                (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-content select=\".inline-replies\"></ng-content>\r\n            </div>\r\n        }\r\n    } @empty {\r\n        <div class=\"banta-empty-state\" *ngIf=\"showEmptyState\">\r\n            Be the first to comment!\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"]}
|
|
728
|
+
//# 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":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAc,MAAM,EAAE,WAAW,EAAE,YAAY,EAAa,MAAM,eAAe,CAAC;AACtH,OAAO,EAAqB,aAAa,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC7E,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAI7C,OAAO,EAAE,gBAAgB,EAAE,MAAM,8BAA8B,CAAC;;;;;;;;AAOhE,MAAM,oBAAoB,GAAG,IAAI,CAAC;AAClC,MAAM,4BAA4B,GAAG,GAAG,CAAC;AAOzC,MAAM,OAAO,oBAAoB;IAC7B,YACY,OAAwB,EACxB,UAAmC;QADnC,YAAO,GAAP,OAAO,CAAiB;QACxB,eAAU,GAAV,UAAU,CAAyB;QAU/C,YAAY;QACZ,gBAAgB;QAER,gBAAW,GAAG,IAAI,YAAY,EAAE,CAAC;QACzC,gBAAW,GAAgB,IAAI,CAAC;QAChC,aAAQ,GAAkB,EAAE,CAAC;QAE7B,sBAAiB,GAAG,KAAK,CAAC;QAE1B,iBAAY,GAAG,IAAI,OAAO,CAAO,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC,CAAC;QACjE,kBAAa,GAAG,KAAK,CAAC;QACtB,kBAAa,GAAG,KAAK,CAAC;QACtB,YAAO,GAAG,KAAK,CAAC;QAChB,mBAAc,GAAG,KAAK,CAAC;QAKvB;;;;;;;;;WASG;QACH,gBAAW,GAAkB,EAAE,CAAC;QAEhC;;;;;;;;;WASG;QACH,kBAAa,GAAkB,EAAE,CAAC;QAezB,eAAU,GAAG,KAAK,CAAC;QACnB,oBAAe,GAAG,KAAK,CAAC;QACxB,mBAAc,GAAG,IAAI,CAAC;QACtB,iBAAY,GAAG,IAAI,CAAC;QACpB,sBAAiB,GAAG,KAAK,CAAC;QAC1B,uBAAkB,GAAG,IAAI,CAAC;QAC1B,oBAAe,GAAsB,EAAE,CAAC;QAKjD,YAAY;QACZ,iBAAiB;QAET,cAAS,GAAG,IAAI,OAAO,EAAe,CAAC;QACvC,WAAM,GAAG,IAAI,OAAO,EAAe,CAAC;QACpC,aAAQ,GAAG,IAAI,OAAO,EAAe,CAAC;QACtC,cAAS,GAAG,IAAI,OAAO,EAAe,CAAC;QACvC,kBAAa,GAAG,IAAI,OAAO,EAAe,CAAC;QAC3C,sBAAiB,GAAG,IAAI,OAAO,EAAQ,CAAC;QACxC,oBAAe,GAAG,IAAI,OAAO,EAAQ,CAAC;QACtC,YAAO,GAAG,IAAI,OAAO,EAAe,CAAC;QACrC,aAAQ,GAAG,IAAI,OAAO,EAAe,CAAC;QACtC,mBAAc,GAAG,IAAI,OAAO,EAAa,CAAC;QAC1C,sBAAiB,GAAG,IAAI,OAAO,EAAiB,CAAC;QACjD,uBAAkB,GAAG,IAAI,OAAO,EAAc,CAAC;QAEpC,iBAAY,GAAG,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;QACjD,aAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;QACzC,UAAK,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC;QACnC,YAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAE,CAAC;QACvC,qBAAgB,GAAG,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;QACzD,mBAAc,GAAG,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QACrD,WAAM,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;QACrC,YAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAE,CAAC;QACvC,aAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;QACzC,kBAAa,GAAG,IAAI,CAAC,cAAc,CAAC,YAAY,EAAE,CAAC;QACnD,qBAAgB,GAAG,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;QACzD,sBAAiB,GAAG,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QA+jBtE,iBAAY,GAAkB,EAAE,CAAC;IApqBzC,CAAC;IAKD,IAAa,MAAM,KAAK,OAAO,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;IAC9C,IAAI,MAAM,CAAC,KAAK,IAAI,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAiB5C,IAAI,gBAAgB,KAAK,OAAO,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC;IAC1F,IAAI,YAAY,KAAK,OAAO,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;IA8BtF,IACI,WAAW,CAAC,KAAK,IAAI,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,CAAC,CAAC;IACrD,IAAI,WAAW,KAAK,OAAO,IAAI,CAAC,YAAY,IAAI,oBAAoB,CAAC,CAAC,CAAC;IAGvE,IACI,kBAAkB,CAAC,KAAK,IAAI,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC,CAAC,CAAC;IACnE,IAAI,kBAAkB,KAAK,OAAO,IAAI,CAAC,mBAAmB,IAAI,4BAA4B,CAAC,CAAC,CAAC;IA+C7F,IAAI,QAAQ,KAAK,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;IAEzD,YAAY;IAEZ;;;OAGG;IACH,wBAAwB,CAAC,OAAoB;QACzC,OAAO,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC;eACjD,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC;eAC5C,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC,CAAC;IAC3D,CAAC;IAED,KAAK,CAAC,oBAAoB,CAAC,OAAoB;QAC3C,MAAM,IAAI,CAAC,YAAY,CAAC;QAExB,OAAO,CAAC,GAAG,CAAC,mBAAmB,OAAO,CAAC,EAAE,gBAAgB,CAAC,CAAC;QAC3D,OAAO,IAAI,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,wBAAwB,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7D,OAAO,CAAC,GAAG,CAAC,yCAAyC,OAAO,CAAC,EAAE,EAAE,CAAC,CAAC;YACnE,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;QAC1B,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,wBAAwB,CAAC,OAAO,CAAC,EAAE,CAAC;YAC1C,OAAO,CAAC,KAAK,CAAC,kEAAkE,OAAO,CAAC,EAAE,yBAAyB,CAAC,CAAC;YACrH,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,IAAI,KAAK,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;QAC3E,IAAI,KAAK,GAAG,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC,CAAC;QAEtD,IAAI,KAAK,GAAG,CAAC,EAAE,CAAC;YACZ,OAAO,CAAC,KAAK,CAAC,kFAAkF,CAAC,CAAC;YAClG,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,IAAI,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,kBAAkB,GAAG,CAAC,CAAC,CAAC;QAClE,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC;QAC/C,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;QACzD,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;IAC9B,CAAC;IAED,IAAI,6BAA6B;QAC7B,OAAO,IAAI,CAAC,aAAa;eAClB,IAAI,CAAC,iBAAiB;eACtB,IAAI,CAAC,gBAAgB,KAAK,UAAU,CAAC,GAAG;eACxC,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC;IACxC,CAAC;IAED,IAAI,qBAAqB;QACrB,IAAI,IAAI,CAAC,iBAAiB;YACtB,OAAO,IAAI,CAAC;QAChB,IAAI,IAAI,CAAC,eAAe,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YAC7C,OAAO,CAAC,GAAG,CAAC,yBAAyB,CAAC,CAAC;YACvC,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,IAAI,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,cAAc;YAC7C,OAAO,IAAI,CAAC;QAEhB,IAAI,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC1B,IAAI,UAAuB,CAAC;YAE5B,IAAI,IAAI,CAAC,UAAU;gBACf,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;;gBAErD,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;YAElC,IAAI,UAAU,EAAE,CAAC;gBACb,MAAM,cAAc,GAAG,IAAI,CAAC,oBAAoB,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;gBAChE,IAAI,cAAc,EAAE,CAAC;oBACjB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,EAAE,CAAC;wBACzC,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;wBAC1C,OAAO,IAAI,CAAC;oBAChB,CAAC;yBAAM,CAAC;wBACJ,OAAO,CAAC,GAAG,CAAC,wBAAwB,CAAC,CAAC;oBAC1C,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACJ,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;gBAC9C,CAAC;YACL,CAAC;iBAAM,CAAC;gBACJ,OAAO,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;YAC9C,CAAC;QACL,CAAC;QAED,OAAO,KAAK,CAAC;IACjB,CAAC;IAEO,gBAAgB,CAAC,OAAgB;QACrC,MAAM,WAAW,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC;QACpD,OAAO,CAAC,CAAC,WAAW;eACb,WAAW,CAAC,MAAM,IAAI,CAAC;eACvB,WAAW,CAAC,KAAK,IAAI,CAAC;eACtB,WAAW,CAAC,IAAI,IAAI,QAAQ,CAAC,eAAe,CAAC,WAAW;eACxD,WAAW,CAAC,GAAG,IAAI,QAAQ,CAAC,eAAe,CAAC,YAAY,CAAC;IACpE,CAAC;IAED;;;;;;OAMG;IACH,6BAA6B,CAAC,OAAoB;QAC9C,IAAI,CAAC,OAAO;YACR,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACzD,OAAO,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,EAAE,KAAK,OAAO,CAAC,EAAE,CAAC,CAAC;IACjE,CAAC;IAED,QAAQ,CAAC,OAAoB,EAAE,UAAkB;QAC7C,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,UAAU,EAAE,CAAC,CAAC;IACtD,CAAC;IAED,WAAW,CAAC,OAAoB;QAC5B,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC9B,CAAC;IAED,aAAa,CAAC,OAAoB;QAC9B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAED,aAAa,CAAC,OAAoB;QAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;IAED,aAAa,CAAC,OAAoB;QAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;IAED,iBAAiB,CAAC,OAAoB;QAClC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACrC,CAAC;IAED,cAAc,CAAC,IAAU;QACrB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACtC,CAAC;IAED,YAAY,CAAC,IAAU;QACnB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAED,aAAa,CAAC,OAAoB;QAC9B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC/B,CAAC;IAED,YAAY,CAAC,OAAoB;QAC7B,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,CAAC,OAAO,GAAG,KAAK,CAAC,CAAC;QAC7D,OAAO,CAAC,cAAc,CAAC,OAAO,GAAG,IAAI,CAAC;IAC1C,CAAC;IAED,aAAa,CAAC,OAAoB;QAC9B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAEO,SAAS,CAAC,KAAqB;QACnC,IAAI,CAAC,iBAAiB,GAAG,CAAC,KAAK,EAAE,SAAS,IAAI,aAAa,CAAC,MAAM,CAAC,KAAK,aAAa,CAAC,MAAM,CAAC;QAC7F,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QAEtB,MAAc,CAAC,gBAAgB,GAAG,KAAK,CAAC;QAEzC,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE,CAAC;YAC/B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QAC5B,CAAC;QACD,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QAErB,IAAI,KAAK,EAAE,CAAC;YACR,MAAM,QAAQ,GAAG,CAAC,KAAK,CAAC,QAAQ,IAAI,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC;YAChD,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,QAAQ,CAAC,MAAM,CAAC,CAAC;YAC/E,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,gCAAgC;YAErD,IAAI,CAAC,WAAW,GAAG,IAAI,YAAY,EAAE,CAAC;YACtC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,eAAe,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAC/F,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAEvF,IAAI,CAAC,WAAW,CAAC,GAAG,CAChB,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,CACtE,CAAC;YAEF,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC9B,CAAC;IACL,CAAC;IAEO,KAAK,CAAC,kBAAkB;QAC5B,IAAI,QAAQ,GAAG,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,mBAAmB,EAAE,CAAC,CAAC;QAC1D,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,KAAK,EAAE,CAAC,CAAC;QAC/C,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC;QAExE,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACjD,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBAClB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC;YAC3G,CAAC;iBAAM,CAAC;gBACJ,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;YACtG,CAAC;QACL,CAAC;QAED,IAAI,CAAC,aAAa,EAAE,CAAC;QACrB,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,IAAI,CAAC,gBAAgB;YACrB,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAChC,CAAC;IAED,aAAa;QACT,IAAI,CAAC,IAAI,CAAC,SAAS;YACf,OAAO;QAEX,iBAAiB;QACjB,qDAAqD;QACrD,YAAY;QACZ,4CAA4C;QAC5C,YAAY;QACZ,gDAAgD;QAChD,iEAAiE;IACrE,CAAC;IAED,OAAO,CAAC,MAAuB,EAAE,QAAgB,CAAC;QAC9C,IAAI,GAAW,CAAC;QAEhB,IAAI,OAAO,MAAM,KAAK,QAAQ;YAC1B,GAAG,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;;YAErB,GAAG,GAAG,MAAM,CAAC;QAEjB,OAAO,GAAG,CAAC,MAAM,GAAG,KAAK;YACrB,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;QAEpB,OAAO,GAAG,CAAC;IACf,CAAC;IACD,OAAO,CAAC,GAAW,EAAE,QAAgB,CAAC;QAClC,OAAO,GAAG,CAAC,MAAM,GAAG,KAAK;YACrB,GAAG,GAAG,GAAG,GAAG,GAAG,CAAC;QAEpB,OAAO,GAAG,CAAC;IACf,CAAC;IAED,eAAe,CAAC,KAAa,EAAE,WAAwB;QACnD,OAAO,WAAW,CAAC,EAAE,CAAC;IAC1B,CAAC;IAED,IAAI,eAAe;QACf,OAAO,IAAI,CAAC,MAAM,EAAE,SAAS,IAAI,aAAa,CAAC,MAAM,CAAC;IAC1D,CAAC;IAED,IAAI,gBAAgB;QAChB,OAAO,IAAI,CAAC,MAAM,EAAE,UAAU,IAAI,UAAU,CAAC,GAAG,CAAC;IACrD,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,UAAU,CAAC,KAAiB;QAE9B,gEAAgE;QAEhE,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;QAE5B,sEAAsE;QACtE,iFAAiF;QAEjF,IAAI,YAAY,GAAG,aAAa,CAAC,MAAM,CAAC;QACxC,IAAI,IAAI,CAAC,eAAe,KAAK,YAAY,IAAI,IAAI,CAAC,gBAAgB,KAAK,UAAU,CAAC,GAAG,EAAE,CAAC;YACpF,IAAI,IAAI,CAAC,eAAe,KAAK,YAAY;gBACrC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC9C,IAAI,IAAI,CAAC,gBAAgB,KAAK,UAAU,CAAC,GAAG;gBACxC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;YACjD,OAAO;QACX,CAAC;QAED,6GAA6G;QAC7G,sDAAsD;QAEtD,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAE3B,qGAAqG;QAErG,IAAI,IAAI,CAAC,UAAU;YACf,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC;;YAE1F,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAE9F,IAAI,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QACnF,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACzD,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QACjG,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC;QAE7C,gCAAgC;QAEhC,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC3B,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBAClB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACrE,CAAC;iBAAM,CAAC;gBACJ,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAC9C,CAAC;QACL,CAAC;IACL,CAAC;IAED,IAAI,SAAS;QACT,IAAI,OAAO,MAAM,KAAK,WAAW;YAC7B,OAAO,KAAK,CAAC;QAEjB,OAAO,YAAY,CAAC,aAAa,CAAC,KAAK,GAAG,CAAC;IAC/C,CAAC;IAED,IAAI,cAAc;QACd,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACnB,OAAO,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC;QACxD,CAAC;QAED,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC;IACxC,CAAC;IAED,IAAI,kBAAkB;QAClB,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YAClB,OAAO,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC;QAC5D,CAAC;QAED,OAAO,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC;IAC5C,CAAC;IAED,IAAI,QAAQ;QACR,OAAO,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,kBAAkB,IAAI,EAAE,CAAC,CAAC;IACvD,CAAC;IAED,KAAK,CAAC,YAAY;QACd,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC1B,IAAI,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC;QACjC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAE3B,IAAI,KAAK,CAAC,YAAY,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;QAEhE,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACnC,MAAM,cAAc,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,YAAY,CAAC,EAAE,YAAY,CAAC,CAAC;YAC5H,IAAI,CAAC,QAAQ,GAAG,CAAC,GAAG,cAAc,EAAE,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;YACtD,YAAY,IAAI,cAAc,CAAC,MAAM,CAAC;QAC1C,CAAC;QAGD,mCAAmC;QACnC,sEAAsE;QAEtE,IAAI,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;QAE/D,IAAI,YAAY,GAAG,CAAC,IAAI,IAAI,CAAC,UAAU,IAAI,WAAW,EAAE,CAAC;YACrD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;YAE1B,IAAI,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;YAEtE,QAAQ,GAAG,QAAQ,CAAC,KAAK,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,8BAA8B;YACrE,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,KAAK,EAAE,CAAC,CAAC;YAE/C,iHAAiH;YACjH,iDAAiD;YAEjD,IAAI,CAAC,QAAQ,GAAG,CAAC,GAAG,QAAQ,EAAE,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;YAEhD,sEAAsE;YAEtE,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC;gBACrB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YAEzB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC/B,CAAC;QAED,yEAAyE;QAEzE,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACjD,IAAI,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YACnF,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,GAAG,QAAQ,CAAC,CAAC;YACvC,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW;gBAC3C,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAC7E,CAAC;QAED,IAAI,CAAC,aAAa,EAAE,CAAC;IACzB,CAAC;IAED,IAAI,aAAa;QACb,IAAI,IAAI,CAAC,eAAe,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,OAAO,CAAC;QACnB,CAAC;aAAM,IAAI,IAAI,CAAC,eAAe,KAAK,QAAQ,EAAE,CAAC;YAC3C,OAAO,OAAO,CAAC;QACnB,CAAC;aAAM,IAAI,IAAI,CAAC,eAAe,KAAK,OAAO,EAAE,CAAC;YAC1C,OAAO,YAAY,CAAC;QACxB,CAAC;QAED,OAAO,MAAM,CAAC;IAClB,CAAC;IAED,IAAI,iBAAiB;QACjB,IAAI,IAAI,CAAC,eAAe,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,OAAO,CAAC;QACnB,CAAC;aAAM,IAAI,IAAI,CAAC,eAAe,KAAK,QAAQ,EAAE,CAAC;YAC3C,OAAO,OAAO,CAAC;QACnB,CAAC;aAAM,IAAI,IAAI,CAAC,eAAe,KAAK,OAAO,EAAE,CAAC;YAC1C,OAAO,YAAY,CAAC;QACxB,CAAC;QAED,OAAO,MAAM,CAAC;IAClB,CAAC;IAED,IAAI,SAAS,KAAK,OAAO,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;IACzF,IAAI,aAAa,KAAK,OAAO,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC;IAE7F;;;;;;;OAOG;IACH,KAAK,CAAC,QAAQ;QACV,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAE1B,IAAI,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC;QAEjC,IAAI,KAAK,CAAC,YAAY,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;QAEhE,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAE3B,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC/B,MAAM,cAAc,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;YACjE,IAAI,CAAC,QAAQ,GAAG,CAAE,GAAG,IAAI,CAAC,QAAQ,EAAE,GAAG,cAAc,CAAE,CAAC;YACxD,YAAY,IAAI,cAAc,CAAC,MAAM,CAAC;QAC1C,CAAC;QAED,MAAM,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAEjH,IAAI,YAAY,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,WAAW,EAAE,CAAC;YACtD,yBAAyB;YAEzB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;YAE1B,IAAI,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;YACtE,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,KAAK,EAAE,CAAC,CAAC;YAC/C,IAAI,CAAC,QAAQ,GAAG,CAAE,GAAG,IAAI,CAAC,QAAQ,EAAE,GAAG,QAAQ,CAAE,CAAC;YAElD,sEAAsE;YAEtE,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC;gBACrB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YAEzB,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC/B,CAAC;QAED,yEAAyE;QAGzE,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACjD,IAAI,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC;YAEvF,yHAAyH;YACzH,qEAAqE;YAErE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC;YACxC,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW;gBAC/C,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC;QACzF,CAAC;QAED,IAAI,CAAC,aAAa,EAAE,CAAC;IACzB,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,QAAQ;QACV,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAE1B,IAAI,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC;QAEjC,IAAI,KAAK,CAAC,YAAY,CAAC;YACnB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;QAEhE,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAE3B,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAChC,MAAM,cAAc,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,EAAE,YAAY,CAAC,CAAC;YAClE,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,cAAc,CAAC,CAAC;YACrD,YAAY,IAAI,cAAc,CAAC,MAAM,CAAC;YACtC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC;QACjD,CAAC;QAED,IAAI,WAAwB,CAAC;QAE7B,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YAClB,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;QAC5D,CAAC;aAAM,CAAC;YACJ,WAAW,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAC/G,CAAC;QAED,IAAI,YAAY,GAAG,CAAC,IAAI,WAAW,EAAE,CAAC;YAClC,yBAAyB;YAEzB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;YAG1B,IAAI,CAAC,WAAW,EAAE,CAAC;gBACf,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;gBAC3B,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;gBACrB,OAAO;YACX,CAAC;YAED,IAAI,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;YAEtE,IAAI,IAAI,CAAC,UAAU;gBACf,QAAQ,GAAG,QAAQ,CAAC,KAAK,EAAE,CAAC,OAAO,EAAE,CAAC;YAE1C,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,cAAc,KAAK,EAAE,CAAC,CAAC;YAE/C,iHAAiH;YACjH,iDAAiD;YAEjD,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBAClB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YACnD,CAAC;iBAAM,CAAC;gBACJ,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;YACnD,CAAC;YAED,sEAAsE;YAEtE,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACxB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;YACzB,CAAC;YAED,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC/B,CAAC;QAED,yEAAyE;QAEzE,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACjD,IAAI,QAAQ,GAAkB,EAAE,CAAC;YAEjC,8CAA8C;YAC9C,oGAAoG;YACpG,gDAAgD;YAEhD,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBAClB,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBAC/E,IAAI,CAAC,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACrD,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YACjG,CAAC;iBAAM,CAAC;gBACJ,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC;gBACnF,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC;gBACnC,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,kBAAkB;oBACpE,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC;YAC3G,CAAC;QAEL,CAAC;IACL,CAAC;IAIO,UAAU,CAAC,OAAoB;QAEnC,IAAI,CAAC,OAAO,CAAC,cAAc;YACvB,OAAO,CAAC,cAAc,KAAK,EAAE,CAAC;QAElC,IAAI,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC;QAChC,IAAI,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC;QAChC,IAAI,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC;QAEjC,IAAI,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAC7B,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAChC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;YAC/B,MAAM,GAAG,IAAI,CAAC;QAClB,CAAC;QAED,gFAAgF;QAChF,IAAI,IAAI,CAAC,eAAe,KAAK,aAAa,CAAC,MAAM;YAC7C,OAAO;QAGX,IAAI,UAAU,EAAE,CAAC;YACb,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAE1B,IAAI,IAAI,CAAC,kBAAkB,GAAG,CAAC,EAAE,CAAC;gBAC9B,IAAI,QAAQ,GAAG,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC;gBAC/E,MAAM,EAAE,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC;YAC9B,CAAC;QACL,CAAC;aAAM,CAAC;YACJ,WAAW,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;YAE7B,IAAI,IAAI,CAAC,kBAAkB,GAAG,CAAC,EAAE,CAAC;gBAC9B,IAAI,QAAQ,GAAG,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC;gBAC/E,MAAM,EAAE,OAAO,CAAC,GAAG,QAAQ,CAAC,CAAC;YACjC,CAAC;QACL,CAAC;QAED,IAAI,MAAM,EAAE,MAAM,GAAG,CAAC;YAClB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QAExB,OAAO,CAAC,YAAY,GAAG,MAAM,CAAC,IAAI,CAAC,sBAAsB,EAAE,CAAC,CAAC;QAC7D,IAAI,CAAC,YAAY,EAAE,CAAC;IACxB,CAAC;IAEO,sBAAsB;QAC1B,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,KAAK,aAAa,CAAC,MAAM;YAC9C,OAAO;QAEX,IAAI,eAAe,GAAG,CAAC,CAAC;QACxB,KAAK,IAAI,KAAK,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;YACtE,KAAK,IAAI,OAAO,IAAI,KAAK,EAAE,CAAC;gBACxB,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC;oBACvB,IAAI,YAAY,GAAG,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;oBACpD,IAAI,YAAY,GAAG,eAAe;wBAC9B,eAAe,GAAG,YAAY,CAAC;oBACnC,OAAO,CAAC,YAAY,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;gBAChD,CAAC;YACL,CAAC;QACL,CAAC;QAED,OAAO,eAAe,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,wBAAwB;QAC1B,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,UAAU,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC;QAC3C,MAAM,IAAI,CAAC,YAAY,CAAC;QACxB,MAAM,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;IAC/D,CAAC;IAEO,YAAY;QAChB,IAAI,CAAC,IAAI,CAAC,MAAM;YACZ,OAAO;QAEX,IAAI,MAAkD,CAAC;QAEvD,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,KAAK,aAAa,CAAC,KAAK;YAC7C,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC;aACpC,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,KAAK,aAAa,CAAC,MAAM;YACnD,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;aACrE,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,KAAK,aAAa,CAAC,MAAM;YACnD,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAE1E,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC3B,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAChC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAClC,CAAC;IAEO,eAAe,CAAC,OAAoB;QACxC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;IAC7B,CAAC;IAED,kBAAkB;QACd,IAAI,CAAC,IAAI,CAAC,gBAAgB;YACtB,OAAO,KAAK,CAAC;QAEjB,MAAM,EAAE,GAAG,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC;QAC/C,MAAM,aAAa,GAAG,EAAE,CAAC,SAAS,CAAC;QACnC,MAAM,YAAY,GAAG,EAAE,CAAC,YAAY,GAAG,EAAE,CAAC,YAAY,CAAC;QAEvD,OAAO,aAAa,GAAG,YAAY,GAAG,EAAE,CAAC;IAC7C,CAAC;IAEO,WAAW,CAAC,OAAoB;QACpC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAEzB,IAAI,CAAC,IAAI,CAAC,gBAAgB;YACtB,OAAO;QAEX,IAAI,CAAC,cAAc,EAAE,CAAC;IAC1B,CAAC;IAED,cAAc;QACV,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACzB,OAAO;QACX,CAAC;QAED,MAAM,EAAE,GAAG,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC;QAE/C,EAAE,CAAC,cAAc,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;QACtC,iCAAiC;IACrC,CAAC;IAED,IAAI,OAAO;QACP,OAAO,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC;IACzC,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,SAA4B;QAC9C,IAAI,OAAO,MAAM,KAAK,WAAW;YAC7B,OAAO;QAEX,MAAM,IAAI,CAAC,wBAAwB,EAAE,CAAC;QAEtC,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,SAAS,CAAC,CAAC;QACrD,IAAI,OAAO,EAAE,CAAC;YACV,OAAO,CAAC,cAAc,CAAC;gBACnB,MAAM,EAAE,QAAQ;gBAChB,KAAK,EAAE,QAAQ;aAClB,CAAC,CAAC;QACP,CAAC;IACL,CAAC;IAED,oBAAoB,CAAC,SAAiB;QAClC,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,qBAAqB,SAAS,IAAI,CAAC,CAAC;IAC1E,CAAC;IAED,UAAU,CAAC,OAAoB;QAC3B,IAAI,CAAC,IAAI,CAAC,WAAW;YACjB,OAAO,KAAK,CAAC;QAEjB,IAAI,OAAO,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;YACzD,OAAO,IAAI,CAAC;QAEhB,OAAO,KAAK,CAAC;IACjB,CAAC;8GAv0BQ,oBAAoB;kGAApB,oBAAoB,4hCA+Gf,gBAAgB,gDCpIlC,uuIAuGA;;2FDlFa,oBAAoB;kBALhC,SAAS;+BACI,oBAAoB;6GAcjB,MAAM;sBAAlB,KAAK;gBAkDF,WAAW;sBADd,KAAK;gBAMF,kBAAkB;sBADrB,KAAK;gBAIG,UAAU;sBAAlB,KAAK;gBACG,eAAe;sBAAvB,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,YAAY;sBAApB,KAAK;gBACG,iBAAiB;sBAAzB,KAAK;gBACG,kBAAkB;sBAA1B,KAAK;gBACG,eAAe;sBAAvB,KAAK;gBACsC,WAAW;sBAAtD,KAAK;;sBAAI,WAAW;uBAAC,oBAAoB;gBACjC,eAAe;sBAAvB,KAAK;gBACG,gBAAgB;sBAAxB,KAAK;gBAkBa,YAAY;sBAA9B,MAAM;gBACY,QAAQ;sBAA1B,MAAM;gBACY,KAAK;sBAAvB,MAAM;gBACY,OAAO;sBAAzB,MAAM;gBACY,gBAAgB;sBAAlC,MAAM;gBACY,cAAc;sBAAhC,MAAM;gBACY,MAAM;sBAAxB,MAAM;gBACY,OAAO;sBAAzB,MAAM;gBACY,QAAQ;sBAA1B,MAAM;gBACY,aAAa;sBAA/B,MAAM;gBACY,gBAAgB;sBAAlC,MAAM;gBACY,iBAAiB;sBAAnC,MAAM;gBAKyB,aAAa;sBAA5C,YAAY;uBAAC,gBAAgB;gBACC,gBAAgB;sBAA9C,SAAS;uBAAC,kBAAkB","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output, HostBinding, ViewChildren, QueryList } 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\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    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() newestLast = false;\r\n    @Input() holdNewMessages = false;\r\n    @Input() showEmptyState = true;\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    //#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 _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 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    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\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\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    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    <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                \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                (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-content select=\".inline-replies\"></ng-content>\r\n            </div>\r\n        }\r\n    } @empty {\r\n        <div class=\"banta-empty-state\" *ngIf=\"showEmptyState\">\r\n            Be the first to comment!\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"]}
|
package/fesm2022/banta-sdk.mjs
CHANGED
|
@@ -7587,6 +7587,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.9", ngImpor
|
|
|
7587
7587
|
args: ['class.visible']
|
|
7588
7588
|
}] } });
|
|
7589
7589
|
|
|
7590
|
+
const DEFAULT_MAX_MESSAGES = 2000;
|
|
7591
|
+
const DEFAULT_MAX_VISIBLE_MESSAGES = 200;
|
|
7590
7592
|
class CommentViewComponent {
|
|
7591
7593
|
constructor(backend, elementRef) {
|
|
7592
7594
|
this.backend = backend;
|
|
@@ -7606,7 +7608,7 @@ class CommentViewComponent {
|
|
|
7606
7608
|
* While this is called "new" messages, it really represents the messages that would be visible *at the beginning
|
|
7607
7609
|
* of the sort order*, which can be flipped by the newestLast feature (used for replies mode).
|
|
7608
7610
|
*
|
|
7609
|
-
* So, when newestLast is false, regardless of the current sortOrder, newMessages are conceptually
|
|
7611
|
+
* So, when newestLast is false (top-level comments), regardless of the current sortOrder, newMessages are conceptually
|
|
7610
7612
|
* *above* the visible set of messages.
|
|
7611
7613
|
*
|
|
7612
7614
|
* When newestLast is true (as in replies mode), regardless of the current sortOrder, newMessages are conceptually
|
|
@@ -7624,10 +7626,6 @@ class CommentViewComponent {
|
|
|
7624
7626
|
* *above* the visible set of messages.
|
|
7625
7627
|
*/
|
|
7626
7628
|
this.olderMessages = [];
|
|
7627
|
-
//#endregion
|
|
7628
|
-
//#region Inputs
|
|
7629
|
-
this.maxMessages = 2000;
|
|
7630
|
-
this.maxVisibleMessages = 200;
|
|
7631
7629
|
this.newestLast = false;
|
|
7632
7630
|
this.holdNewMessages = false;
|
|
7633
7631
|
this.showEmptyState = true;
|
|
@@ -7667,6 +7665,10 @@ class CommentViewComponent {
|
|
|
7667
7665
|
set source(value) { this.setSource(value); }
|
|
7668
7666
|
get previousMessages() { return this.newestLast ? this.olderMessages : this.newMessages; }
|
|
7669
7667
|
get nextMessages() { return this.newestLast ? this.newMessages : this.olderMessages; }
|
|
7668
|
+
set maxMessages(value) { this._maxMessages = value; }
|
|
7669
|
+
get maxMessages() { return this._maxMessages ?? DEFAULT_MAX_MESSAGES; }
|
|
7670
|
+
set maxVisibleMessages(value) { this._maxVisibleMessages = value; }
|
|
7671
|
+
get maxVisibleMessages() { return this._maxVisibleMessages ?? DEFAULT_MAX_VISIBLE_MESSAGES; }
|
|
7670
7672
|
get comments() { return Array.from(this.commentsQuery); }
|
|
7671
7673
|
//#endregion
|
|
7672
7674
|
/**
|
|
@@ -7689,16 +7691,15 @@ class CommentViewComponent {
|
|
|
7689
7691
|
console.error(`Error while loading message in context: Failed to find message ${message.id}, maybe it was deleted!`);
|
|
7690
7692
|
return false;
|
|
7691
7693
|
}
|
|
7692
|
-
let pageSize = this.maxVisibleMessages;
|
|
7693
7694
|
let items = [].concat(this.olderMessages, this.messages, this.newMessages);
|
|
7694
7695
|
let index = items.findIndex(x => x.id === message.id);
|
|
7695
7696
|
if (index < 0) {
|
|
7696
7697
|
console.error(`Error while loading message in context: Message was not present in message list!`);
|
|
7697
7698
|
return false;
|
|
7698
7699
|
}
|
|
7699
|
-
let startIndex = Math.max(0, index -
|
|
7700
|
+
let startIndex = Math.max(0, index - this.maxVisibleMessages / 2);
|
|
7700
7701
|
this.newMessages = items.splice(0, startIndex);
|
|
7701
|
-
this.messages = items.splice(0,
|
|
7702
|
+
this.messages = items.splice(0, this.maxVisibleMessages);
|
|
7702
7703
|
this.olderMessages = items;
|
|
7703
7704
|
this.isViewingMore = true;
|
|
7704
7705
|
}
|
|
@@ -7936,12 +7937,14 @@ class CommentViewComponent {
|
|
|
7936
7937
|
return this.previousMessages.length > 0;
|
|
7937
7938
|
}
|
|
7938
7939
|
get pageSize() {
|
|
7939
|
-
return Math.min(20, this.maxVisibleMessages);
|
|
7940
|
+
return Math.min(20, this.maxVisibleMessages || 20);
|
|
7940
7941
|
}
|
|
7941
7942
|
async showPrevious() {
|
|
7942
7943
|
this.isViewingMore = true;
|
|
7943
7944
|
let nextPageSize = this.pageSize;
|
|
7944
7945
|
this.isLoadingMore = false;
|
|
7946
|
+
if (isNaN(nextPageSize))
|
|
7947
|
+
throw new Error(`Not safe to load more with NaN page size`);
|
|
7945
7948
|
if (this.previousMessages.length > 0) {
|
|
7946
7949
|
const storedMessages = this.previousMessages.splice(Math.max(0, this.previousMessages.length - nextPageSize), nextPageSize);
|
|
7947
7950
|
this.messages = [...storedMessages, ...this.messages];
|
|
@@ -8009,6 +8012,8 @@ class CommentViewComponent {
|
|
|
8009
8012
|
async showNext() {
|
|
8010
8013
|
this.isViewingMore = true;
|
|
8011
8014
|
let nextPageSize = this.pageSize;
|
|
8015
|
+
if (isNaN(nextPageSize))
|
|
8016
|
+
throw new Error(`Not safe to load more with NaN page size`);
|
|
8012
8017
|
this.isLoadingMore = false;
|
|
8013
8018
|
if (this.nextMessages.length > 0) {
|
|
8014
8019
|
const storedMessages = this.nextMessages.splice(0, nextPageSize);
|
|
@@ -8049,6 +8054,8 @@ class CommentViewComponent {
|
|
|
8049
8054
|
async showMore() {
|
|
8050
8055
|
this.isViewingMore = true;
|
|
8051
8056
|
let nextPageSize = this.pageSize;
|
|
8057
|
+
if (isNaN(nextPageSize))
|
|
8058
|
+
throw new Error(`Not safe to load more with NaN page size`);
|
|
8052
8059
|
this.isLoadingMore = false;
|
|
8053
8060
|
if (this.olderMessages.length > 0) {
|
|
8054
8061
|
const storedMessages = this.olderMessages.splice(0, nextPageSize);
|
|
@@ -8092,14 +8099,20 @@ class CommentViewComponent {
|
|
|
8092
8099
|
// Extract the messages that do not fit in the maxVisibleMessages buffer.
|
|
8093
8100
|
if (this.messages.length > this.maxVisibleMessages) {
|
|
8094
8101
|
let overflow = [];
|
|
8095
|
-
|
|
8102
|
+
// Move overflowing messages into newMessages.
|
|
8103
|
+
// Regardless of the order (newestLast), newMessages represents the direction that is being loaded,
|
|
8104
|
+
// since it's definition depends on that order.
|
|
8105
|
+
if (this.newestLast) {
|
|
8096
8106
|
overflow = this.messages.splice(this.maxVisibleMessages, this.messages.length);
|
|
8097
|
-
|
|
8107
|
+
this.newMessages = overflow.concat(this.newMessages);
|
|
8108
|
+
this.newMessages.splice(this.maxMessages - this.maxVisibleMessages, this.newMessages.length);
|
|
8109
|
+
}
|
|
8110
|
+
else {
|
|
8098
8111
|
overflow = this.messages.splice(0, this.messages.length - this.maxVisibleMessages);
|
|
8099
|
-
|
|
8100
|
-
|
|
8101
|
-
|
|
8102
|
-
|
|
8112
|
+
this.newMessages.push(...overflow);
|
|
8113
|
+
if (this.newMessages.length > this.maxMessages - this.maxVisibleMessages)
|
|
8114
|
+
this.newMessages.splice(0, this.newMessages.length - (this.maxMessages - this.maxVisibleMessages));
|
|
8115
|
+
}
|
|
8103
8116
|
}
|
|
8104
8117
|
}
|
|
8105
8118
|
addMessage(message) {
|