@banta/sdk 4.6.17 → 4.6.18

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.
@@ -82,7 +82,10 @@ export class BantaChatComponent {
82
82
  this._reported.next(message);
83
83
  }
84
84
  upvote(message) {
85
- this._upvoted.next(message);
85
+ return __awaiter(this, void 0, void 0, function* () {
86
+ yield this.source.likeMessage(message.id);
87
+ this._upvoted.next(message);
88
+ });
86
89
  }
87
90
  get selected() {
88
91
  return this._selected;
@@ -161,4 +164,4 @@ BantaChatComponent.propDecorators = {
161
164
  upvoted: [{ type: Output }],
162
165
  userSelected: [{ type: Output }]
163
166
  };
164
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"banta-chat.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/chat/banta-chat/banta-chat.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, ViewChild } from \"@angular/core\";\r\nimport { Subject, Observable, Subscription } from 'rxjs';\r\n\r\nimport { User, ChatMessage, NewMessageForm } from '@banta/common';\r\nimport { ChatViewComponent } from '../chat-view/chat-view.component';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\n\r\n/**\r\n * Chat component\r\n */\r\n@Component({\r\n    selector: 'banta-chat',\r\n    templateUrl: './banta-chat.component.html',\r\n    styleUrls: ['./banta-chat.component.scss']\r\n})\r\nexport class BantaChatComponent {\r\n    constructor(\r\n        private backend : ChatBackendBase\r\n    ) {\r\n    }\r\n\r\n    private _source : ChatSourceBase;\r\n    private _subs = new Subscription();\r\n    user : User = null;\r\n\r\n    @Input() shouldInterceptMessageSend?: (message: ChatMessage, source: ChatSourceBase) => boolean | Promise<boolean>;\r\n\r\n\r\n    ngOnInit() {\r\n        this._subs.add(this.backend.userChanged.subscribe(user => this.user = user));\r\n    }\r\n\r\n    ngOnDestroy() {\r\n        this._subs.unsubscribe();\r\n    }\r\n\r\n    @Input()\r\n    get source() : ChatSourceBase {\r\n        return this._source;\r\n    }\r\n\r\n    set source(value) {\r\n        this._source = value;\r\n    }\r\n\r\n    @Input()\r\n    get topicID() : string {\r\n        return this._source.identifier;\r\n    }\r\n\r\n    set topicID(value) {\r\n        this.setSourceFromTopicID(value);\r\n    }\r\n\r\n    private async setSourceFromTopicID(topicID : string) {\r\n        if (this._source && this._source.close)\r\n            this._source.close();\r\n        this._source = null;\r\n        this._source = await this.backend.getSourceForTopic(topicID);\r\n    }\r\n\r\n    @Input() signInLabel = 'Sign In';\r\n    @Input() sendLabel = 'Send';\r\n    @Input() permissionDeniedLabel = 'Send';\r\n\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _upvoted = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _signInSelected = new Subject<void>();\r\n    private _permissionDeniedError = new Subject<string>();\r\n\r\n    @Output()\r\n    get signInSelected(): Observable<void> {\r\n        return this._signInSelected;\r\n    }\r\n\r\n    @Output()\r\n    get permissionDeniedError(): Observable<string> {\r\n        return this._permissionDeniedError;\r\n    }\r\n\r\n    showEmojiPanel = false;\r\n\r\n    showSignIn() {\r\n        this._signInSelected.next();\r\n    }\r\n\r\n    sendPermissionError(message: string) {\r\n        this._permissionDeniedError.next(message);\r\n    }\r\n\r\n    insertEmoji(emoji) {\r\n        let message = this.newMessage.message || '';\r\n\r\n        this.newMessage.message = message + emoji;\r\n    }\r\n\r\n    onKeyDown(event : KeyboardEvent) {\r\n        // TODO\r\n    }\r\n\r\n    @ViewChild('chatView', { static: true })\r\n    chatView : ChatViewComponent;\r\n\r\n    jumpToMessage(message : ChatMessage) {\r\n        if (this.chatView)\r\n            this.chatView.jumpTo(message);\r\n    }\r\n\r\n    select(message : ChatMessage) {\r\n        this._selected.next(message);\r\n    }\r\n\r\n    selectUser(message : ChatMessage) {\r\n        this._userSelected.next(message);\r\n    }\r\n\r\n    report(message : ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    upvote(message : ChatMessage) {\r\n        this._upvoted.next(message);\r\n    }\r\n\r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output()\r\n    get reported() {\r\n        return this._reported;\r\n    }\r\n\r\n    @Output()\r\n    get upvoted() {\r\n        return this._upvoted;\r\n    }\r\n\r\n    @Output()\r\n    get userSelected() {\r\n        return this._userSelected;\r\n    }\r\n\r\n    get canChat() {\r\n        if (!this.user)\r\n            return false;\r\n\r\n        // TODO\r\n        // if (!this.user.permissions)\r\n        //     return true;\r\n\r\n        // if (!this.user.permissions.canChat)\r\n        //     return true;\r\n\r\n        // return this.user.permissions?.canChat(this.source);\r\n        \r\n        return true;\r\n    }\r\n\r\n    newMessage : NewMessageForm = {};\r\n\r\n    async sendMessage() {\r\n        if (!this.source)\r\n            return;\r\n\r\n        let text = (this.newMessage.message || '').trim();\r\n        this.newMessage.message = '';\r\n\r\n        if (text === '')\r\n            return;\r\n\r\n        let message : ChatMessage = {\r\n            user: null,\r\n            sentAt: Date.now(),\r\n            likes: 0,\r\n            url: typeof window !== 'undefined' ? location.href : undefined,\r\n            message: text\r\n        };\r\n\r\n        try {\r\n            const intercept = await this.shouldInterceptMessageSend?.(message, this.source);\r\n            if (!intercept) {\r\n                await this.source.send(message);\r\n            }\r\n        } catch (e) {\r\n            console.error(`Failed to send message: `, message);\r\n            console.error(e);\r\n        }\r\n    }\r\n}\r\n"]}
167
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"banta-chat.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/chat/banta-chat/banta-chat.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, ViewChild } from \"@angular/core\";\r\nimport { Subject, Observable, Subscription } from 'rxjs';\r\n\r\nimport { User, ChatMessage, NewMessageForm } from '@banta/common';\r\nimport { ChatViewComponent } from '../chat-view/chat-view.component';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\n\r\n/**\r\n * Chat component\r\n */\r\n@Component({\r\n    selector: 'banta-chat',\r\n    templateUrl: './banta-chat.component.html',\r\n    styleUrls: ['./banta-chat.component.scss']\r\n})\r\nexport class BantaChatComponent {\r\n    constructor(\r\n        private backend : ChatBackendBase\r\n    ) {\r\n    }\r\n\r\n    private _source : ChatSourceBase;\r\n    private _subs = new Subscription();\r\n    user : User = null;\r\n\r\n    @Input() shouldInterceptMessageSend?: (message: ChatMessage, source: ChatSourceBase) => boolean | Promise<boolean>;\r\n\r\n\r\n    ngOnInit() {\r\n        this._subs.add(this.backend.userChanged.subscribe(user => this.user = user));\r\n    }\r\n\r\n    ngOnDestroy() {\r\n        this._subs.unsubscribe();\r\n    }\r\n\r\n    @Input()\r\n    get source() : ChatSourceBase {\r\n        return this._source;\r\n    }\r\n\r\n    set source(value) {\r\n        this._source = value;\r\n    }\r\n\r\n    @Input()\r\n    get topicID() : string {\r\n        return this._source.identifier;\r\n    }\r\n\r\n    set topicID(value) {\r\n        this.setSourceFromTopicID(value);\r\n    }\r\n\r\n    private async setSourceFromTopicID(topicID : string) {\r\n        if (this._source && this._source.close)\r\n            this._source.close();\r\n        this._source = null;\r\n        this._source = await this.backend.getSourceForTopic(topicID);\r\n    }\r\n\r\n    @Input() signInLabel = 'Sign In';\r\n    @Input() sendLabel = 'Send';\r\n    @Input() permissionDeniedLabel = 'Send';\r\n\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _upvoted = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _signInSelected = new Subject<void>();\r\n    private _permissionDeniedError = new Subject<string>();\r\n\r\n    @Output()\r\n    get signInSelected(): Observable<void> {\r\n        return this._signInSelected;\r\n    }\r\n\r\n    @Output()\r\n    get permissionDeniedError(): Observable<string> {\r\n        return this._permissionDeniedError;\r\n    }\r\n\r\n    showEmojiPanel = false;\r\n\r\n    showSignIn() {\r\n        this._signInSelected.next();\r\n    }\r\n\r\n    sendPermissionError(message: string) {\r\n        this._permissionDeniedError.next(message);\r\n    }\r\n\r\n    insertEmoji(emoji) {\r\n        let message = this.newMessage.message || '';\r\n\r\n        this.newMessage.message = message + emoji;\r\n    }\r\n\r\n    onKeyDown(event : KeyboardEvent) {\r\n        // TODO\r\n    }\r\n\r\n    @ViewChild('chatView', { static: true })\r\n    chatView : ChatViewComponent;\r\n\r\n    jumpToMessage(message : ChatMessage) {\r\n        if (this.chatView)\r\n            this.chatView.jumpTo(message);\r\n    }\r\n\r\n    select(message : ChatMessage) {\r\n        this._selected.next(message);\r\n    }\r\n\r\n    selectUser(message : ChatMessage) {\r\n        this._userSelected.next(message);\r\n    }\r\n\r\n    report(message : ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    async upvote(message : ChatMessage) {\r\n        await this.source.likeMessage(message.id);\r\n        this._upvoted.next(message);\r\n    }\r\n\r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output()\r\n    get reported() {\r\n        return this._reported;\r\n    }\r\n\r\n    @Output()\r\n    get upvoted() {\r\n        return this._upvoted;\r\n    }\r\n\r\n    @Output()\r\n    get userSelected() {\r\n        return this._userSelected;\r\n    }\r\n\r\n    get canChat() {\r\n        if (!this.user)\r\n            return false;\r\n\r\n        // TODO\r\n        // if (!this.user.permissions)\r\n        //     return true;\r\n\r\n        // if (!this.user.permissions.canChat)\r\n        //     return true;\r\n\r\n        // return this.user.permissions?.canChat(this.source);\r\n        \r\n        return true;\r\n    }\r\n\r\n    newMessage : NewMessageForm = {};\r\n\r\n    async sendMessage() {\r\n        if (!this.source)\r\n            return;\r\n\r\n        let text = (this.newMessage.message || '').trim();\r\n        this.newMessage.message = '';\r\n\r\n        if (text === '')\r\n            return;\r\n\r\n        let message : ChatMessage = {\r\n            user: null,\r\n            sentAt: Date.now(),\r\n            likes: 0,\r\n            url: typeof window !== 'undefined' ? location.href : undefined,\r\n            message: text\r\n        };\r\n\r\n        try {\r\n            const intercept = await this.shouldInterceptMessageSend?.(message, this.source);\r\n            if (!intercept) {\r\n                await this.source.send(message);\r\n            }\r\n        } catch (e) {\r\n            console.error(`Failed to send message: `, message);\r\n            console.error(e);\r\n        }\r\n    }\r\n}\r\n"]}
@@ -1,3 +1,4 @@
1
+ import { __awaiter } from "tslib";
1
2
  import { Component, Input, ViewChild, ElementRef, Output } from "@angular/core";
2
3
  import { Subject, Subscription } from 'rxjs';
3
4
  import { ChatBackendBase } from "../../chat-backend-base";
@@ -46,14 +47,31 @@ export class ChatViewComponent {
46
47
  this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));
47
48
  this._sourceSubs.add(this.backend.userChanged
48
49
  .subscribe(user => this.currentUser = user));
50
+ this.getInitialMessages();
49
51
  }
50
52
  }
53
+ getInitialMessages() {
54
+ return __awaiter(this, void 0, void 0, function* () {
55
+ let messages = (yield this._source.getExistingMessages());
56
+ messages.forEach(m => { var _a; return (_a = m.transientState) !== null && _a !== void 0 ? _a : (m.transientState = {}); });
57
+ this.messages = messages.slice().reverse();
58
+ this.sortMessages();
59
+ });
60
+ }
61
+ sortMessages() {
62
+ if (!this.source)
63
+ return;
64
+ let sorter;
65
+ sorter = (a, b) => (a.sentAt - b.sentAt);
66
+ this.messages.sort(sorter);
67
+ }
51
68
  addMessage(message) {
52
69
  if (this.messages.length > this.maxMessages + 1) {
53
70
  while (this.messages.length > this.maxMessages)
54
71
  this.messages.shift();
55
72
  }
56
73
  this.messages.push(message);
74
+ this.sortMessages();
57
75
  }
58
76
  messageReceived(message) {
59
77
  this.addMessage(message);
@@ -143,4 +161,4 @@ ChatViewComponent.propDecorators = {
143
161
  messageContainer: [{ type: ViewChild, args: ['messageContainer',] }],
144
162
  maxMessages: [{ type: Input }]
145
163
  };
146
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-view.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/chat/chat-view/chat-view.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output } from \"@angular/core\";\r\nimport { User, ChatMessage } from '@banta/common';\r\nimport { Subject, Subscription } from 'rxjs';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\n\r\n@Component({\r\n    selector: 'banta-chat-view',\r\n    templateUrl: './chat-view.component.html',\r\n    styleUrls: ['./chat-view.component.scss']\r\n})\r\nexport class ChatViewComponent {\r\n    constructor(\r\n        private backend: ChatBackendBase,\r\n        private elementRef : ElementRef<HTMLElement>\r\n    ) {\r\n\r\n    }\r\n\r\n    private _sourceSubs = new Subscription();\r\n    private _source : ChatSourceBase;\r\n\r\n    @Input()\r\n    get source() {\r\n        return this._source;\r\n    }\r\n\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _upvoted = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n\r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output()\r\n    get userSelected() {\r\n        return this._userSelected;\r\n    }\r\n\r\n    @Output()\r\n    get reported() {\r\n        return this._reported;\r\n    }\r\n\r\n    @Output()\r\n    get upvoted() {\r\n        return this._upvoted;\r\n    }\r\n\r\n    set source(value) {\r\n        if (this._sourceSubs) {\r\n            this._sourceSubs.unsubscribe();\r\n            this._sourceSubs = null;\r\n        }\r\n\r\n        this._source = value;\r\n\r\n        this.messages = [];\r\n\r\n        if (value) {\r\n            this._sourceSubs = new Subscription();\r\n            this.messages = value.messages.slice();\r\n\r\n            console.log(`Source set:`);\r\n            console.dir(value);\r\n\r\n            console.log(`Messages loaded:`);\r\n            console.dir(this.messages); \r\n            \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\r\n                    .subscribe(user => this.currentUser = user)\r\n            );\r\n        }\r\n    }\r\n\r\n    messages : ChatMessage[] = [];\r\n    currentUser : User;\r\n\r\n    @ViewChild('messageContainer')\r\n    messageContainer : ElementRef<HTMLElement>;\r\n\r\n    @Input()\r\n    maxMessages : number = 200;\r\n\r\n    private addMessage(message : ChatMessage) {\r\n        if (this.messages.length > this.maxMessages + 1) {\r\n            while (this.messages.length > this.maxMessages)\r\n                this.messages.shift();\r\n        }\r\n\r\n        this.messages.push(message);\r\n    }\r\n\r\n    private messageReceived(message : ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (this.isScrolledToLatest())\r\n            setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    isScrolledToLatest() {\r\n        if (!this.messageContainer)\r\n            return false;\r\n\r\n        let el = this.messageContainer.nativeElement;\r\n        let currentScroll = el.scrollTop;\r\n        let 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        setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    scrollToLatest() {\r\n        if (!this.messageContainer)\r\n            return;\r\n        \r\n        let el = this.messageContainer.nativeElement;\r\n        el.scrollTop = el.scrollHeight;\r\n    }\r\n\r\n    jumpTo(message : ChatMessage) {\r\n        let element = this.elementRef.nativeElement;\r\n        let messageElement = element.querySelector(`banta-chat-message[data-id=\"${message.id}\"]`)\r\n\r\n        if (!messageElement) {\r\n            alert(`could not find message ${message.id}`);\r\n            return;\r\n        }\r\n\r\n        messageElement.scrollIntoView({ behavior: 'smooth' });\r\n\r\n        this.flashMessage(message);\r\n    }\r\n\r\n    flashMessage(message : ChatMessage) {\r\n        if (!message)\r\n            return;\r\n        \r\n        this.flashedMessageId = message.id;\r\n        //setTimeout(() => this.flashedMessageId = null, 250);\r\n    }\r\n\r\n    flashedMessageId : string;\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    upvoteMessage(message : ChatMessage) {\r\n        this._upvoted.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    avatarForUser(user : User) {\r\n        if (user && user.avatarUrl) {\r\n            let url = user.avatarUrl;\r\n            return `url(${url})`;\r\n        }\r\n\r\n        return null;\r\n    }\r\n}"]}
164
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-view.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/chat/chat-view/chat-view.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ViewChild, ElementRef, Output } from \"@angular/core\";\r\nimport { User, ChatMessage } from '@banta/common';\r\nimport { Subject, Subscription } from 'rxjs';\r\nimport { ChatBackendBase } from \"../../chat-backend-base\";\r\nimport { ChatSourceBase } from \"../../chat-source-base\";\r\n\r\n@Component({\r\n    selector: 'banta-chat-view',\r\n    templateUrl: './chat-view.component.html',\r\n    styleUrls: ['./chat-view.component.scss']\r\n})\r\nexport class ChatViewComponent {\r\n    constructor(\r\n        private backend: ChatBackendBase,\r\n        private elementRef : ElementRef<HTMLElement>\r\n    ) {\r\n\r\n    }\r\n\r\n    private _sourceSubs = new Subscription();\r\n    private _source : ChatSourceBase;\r\n\r\n    @Input()\r\n    get source() {\r\n        return this._source;\r\n    }\r\n\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _upvoted = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n\r\n    @Output()\r\n    get selected() {\r\n        return this._selected;\r\n    }\r\n\r\n    @Output()\r\n    get userSelected() {\r\n        return this._userSelected;\r\n    }\r\n\r\n    @Output()\r\n    get reported() {\r\n        return this._reported;\r\n    }\r\n\r\n    @Output()\r\n    get upvoted() {\r\n        return this._upvoted;\r\n    }\r\n\r\n    set source(value) {\r\n        if (this._sourceSubs) {\r\n            this._sourceSubs.unsubscribe();\r\n            this._sourceSubs = null;\r\n        }\r\n\r\n        this._source = value;\r\n\r\n        this.messages = [];\r\n\r\n        if (value) {\r\n            this._sourceSubs = new Subscription();\r\n            this.messages = value.messages.slice();\r\n\r\n            console.log(`Source set:`);\r\n            console.dir(value);\r\n\r\n            console.log(`Messages loaded:`);\r\n            console.dir(this.messages); \r\n            \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\r\n                    .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 = messages.slice().reverse();\r\n        this.sortMessages();\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        sorter = (a, b) => (a.sentAt - b.sentAt);\r\n\r\n        this.messages.sort(sorter);\r\n    }\r\n\r\n    messages : ChatMessage[] = [];\r\n    currentUser : User;\r\n\r\n    @ViewChild('messageContainer')\r\n    messageContainer : ElementRef<HTMLElement>;\r\n\r\n    @Input()\r\n    maxMessages : number = 200;\r\n\r\n    private addMessage(message : ChatMessage) {\r\n        if (this.messages.length > this.maxMessages + 1) {\r\n            while (this.messages.length > this.maxMessages)\r\n                this.messages.shift();\r\n        }\r\n\r\n        this.messages.push(message);\r\n        this.sortMessages();\r\n    }\r\n\r\n    private messageReceived(message : ChatMessage) {\r\n        this.addMessage(message);\r\n\r\n        if (this.isScrolledToLatest())\r\n            setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    isScrolledToLatest() {\r\n        if (!this.messageContainer)\r\n            return false;\r\n\r\n        let el = this.messageContainer.nativeElement;\r\n        let currentScroll = el.scrollTop;\r\n        let 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        setTimeout(() => this.scrollToLatest());\r\n    }\r\n\r\n    scrollToLatest() {\r\n        if (!this.messageContainer)\r\n            return;\r\n        \r\n        let el = this.messageContainer.nativeElement;\r\n        el.scrollTop = el.scrollHeight;\r\n    }\r\n\r\n    jumpTo(message : ChatMessage) {\r\n        let element = this.elementRef.nativeElement;\r\n        let messageElement = element.querySelector(`banta-chat-message[data-id=\"${message.id}\"]`)\r\n\r\n        if (!messageElement) {\r\n            alert(`could not find message ${message.id}`);\r\n            return;\r\n        }\r\n\r\n        messageElement.scrollIntoView({ behavior: 'smooth' });\r\n\r\n        this.flashMessage(message);\r\n    }\r\n\r\n    flashMessage(message : ChatMessage) {\r\n        if (!message)\r\n            return;\r\n        \r\n        this.flashedMessageId = message.id;\r\n        //setTimeout(() => this.flashedMessageId = null, 250);\r\n    }\r\n\r\n    flashedMessageId : string;\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    upvoteMessage(message : ChatMessage) {\r\n        this._upvoted.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    avatarForUser(user : User) {\r\n        if (user && user.avatarUrl) {\r\n            let url = user.avatarUrl;\r\n            return `url(${url})`;\r\n        }\r\n\r\n        return null;\r\n    }\r\n}"]}
@@ -7389,14 +7389,31 @@ class ChatViewComponent {
7389
7389
  this._sourceSubs.add(this._source.messageSent.subscribe(msg => this.messageSent(msg)));
7390
7390
  this._sourceSubs.add(this.backend.userChanged
7391
7391
  .subscribe(user => this.currentUser = user));
7392
+ this.getInitialMessages();
7392
7393
  }
7393
7394
  }
7395
+ getInitialMessages() {
7396
+ return __awaiter(this, void 0, void 0, function* () {
7397
+ let messages = (yield this._source.getExistingMessages());
7398
+ messages.forEach(m => { var _a; return (_a = m.transientState) !== null && _a !== void 0 ? _a : (m.transientState = {}); });
7399
+ this.messages = messages.slice().reverse();
7400
+ this.sortMessages();
7401
+ });
7402
+ }
7403
+ sortMessages() {
7404
+ if (!this.source)
7405
+ return;
7406
+ let sorter;
7407
+ sorter = (a, b) => (a.sentAt - b.sentAt);
7408
+ this.messages.sort(sorter);
7409
+ }
7394
7410
  addMessage(message) {
7395
7411
  if (this.messages.length > this.maxMessages + 1) {
7396
7412
  while (this.messages.length > this.maxMessages)
7397
7413
  this.messages.shift();
7398
7414
  }
7399
7415
  this.messages.push(message);
7416
+ this.sortMessages();
7400
7417
  }
7401
7418
  messageReceived(message) {
7402
7419
  this.addMessage(message);
@@ -7566,7 +7583,10 @@ class BantaChatComponent {
7566
7583
  this._reported.next(message);
7567
7584
  }
7568
7585
  upvote(message) {
7569
- this._upvoted.next(message);
7586
+ return __awaiter(this, void 0, void 0, function* () {
7587
+ yield this.source.likeMessage(message.id);
7588
+ this._upvoted.next(message);
7589
+ });
7570
7590
  }
7571
7591
  get selected() {
7572
7592
  return this._selected;