@banta/sdk 3.3.11 → 4.0.2
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/banta-sdk.metadata.json +1 -1
- package/bundles/banta-sdk.umd.js +1157 -413
- package/bundles/banta-sdk.umd.js.map +1 -1
- package/bundles/banta-sdk.umd.min.js +1 -1
- package/bundles/banta-sdk.umd.min.js.map +1 -1
- package/esm2015/lib/banta/banta.component.js +9 -15
- package/esm2015/lib/banta-sdk.module.js +10 -4
- package/esm2015/lib/chat/banta-chat/banta-chat.component.js +16 -19
- package/esm2015/lib/chat/chat-message/chat-message.component.js +2 -2
- package/esm2015/lib/chat/chat-view/chat-view.component.js +7 -6
- package/esm2015/lib/chat/live-chat-message.component.js +3 -3
- package/esm2015/lib/chat-backend-base.js +17 -0
- package/esm2015/lib/chat-backend.js +74 -0
- package/esm2015/lib/chat-source-base.js +2 -0
- package/esm2015/lib/chat-source.js +151 -0
- package/esm2015/lib/comments/banta-comments/banta-comments.component.js +344 -174
- package/esm2015/lib/comments/comment/comment.component.js +56 -20
- package/esm2015/lib/comments/comment-field/comment-field.component.js +19 -17
- package/esm2015/lib/comments/comment-view/comment-view.component.js +77 -39
- package/esm2015/lib/comments/live-comment.component.js +3 -3
- package/esm2015/lib/common/index.js +1 -3
- package/esm2015/lib/index.js +6 -1
- package/esm2015/lib/sdk-options.js +2 -0
- package/fesm2015/banta-sdk.js +761 -310
- package/fesm2015/banta-sdk.js.map +1 -1
- package/lib/banta/banta.component.d.ts +7 -8
- package/lib/banta-sdk.module.d.ts +2 -1
- package/lib/chat/banta-chat/banta-chat.component.d.ts +10 -12
- package/lib/chat/chat-view/chat-view.component.d.ts +7 -4
- package/lib/chat/live-chat-message.component.d.ts +2 -2
- package/lib/chat-backend-base.d.ts +22 -0
- package/lib/chat-backend.d.ts +21 -0
- package/lib/chat-source-base.d.ts +31 -0
- package/lib/chat-source.d.ts +38 -0
- package/lib/comments/banta-comments/banta-comments.component.d.ts +68 -61
- package/lib/comments/comment/comment.component.d.ts +25 -6
- package/lib/comments/comment-field/comment-field.component.d.ts +9 -5
- package/lib/comments/comment-view/comment-view.component.d.ts +26 -8
- package/lib/comments/live-comment.component.d.ts +2 -2
- package/lib/common/index.d.ts +0 -2
- package/lib/index.d.ts +5 -0
- package/lib/sdk-options.d.ts +4 -0
- package/package.json +1 -1
- package/esm2015/lib/common/banta.service.js +0 -21
- package/esm2015/lib/common/chat-backend.service.js +0 -7
- package/lib/common/banta.service.d.ts +0 -9
- package/lib/common/chat-backend.service.d.ts +0 -14
|
@@ -2,18 +2,37 @@ import { __awaiter } from "tslib";
|
|
|
2
2
|
import { Component, ElementRef, Input, Output } from '@angular/core';
|
|
3
3
|
import { CommentsOrder } from '@banta/common';
|
|
4
4
|
import { Subject, Observable, Subscription } from 'rxjs';
|
|
5
|
-
import { ChatBackendService } from '../../common';
|
|
6
|
-
import { BantaService } from '../../common';
|
|
7
5
|
import { ActivatedRoute } from '@angular/router';
|
|
6
|
+
import { ChatBackendBase } from '../../chat-backend-base';
|
|
8
7
|
/**
|
|
9
8
|
* Comments component
|
|
10
9
|
*/
|
|
11
10
|
export class BantaCommentsComponent {
|
|
12
|
-
constructor(
|
|
13
|
-
this.banta = banta;
|
|
11
|
+
constructor(backend, elementRef, activatedRoute) {
|
|
14
12
|
this.backend = backend;
|
|
15
13
|
this.elementRef = elementRef;
|
|
16
14
|
this.activatedRoute = activatedRoute;
|
|
15
|
+
// Loading Screen
|
|
16
|
+
this._loadingMessage = '';
|
|
17
|
+
this.loadingMessageVisible = false;
|
|
18
|
+
this.loading = true;
|
|
19
|
+
this.showLoadingScreen = false;
|
|
20
|
+
this._loadingMessageIndex = 0;
|
|
21
|
+
this.loadingMessages = [
|
|
22
|
+
`Just a second...`,
|
|
23
|
+
`We're definitely working on it.`,
|
|
24
|
+
`There's no need to refresh.`,
|
|
25
|
+
`It's definitely worth the wait!`,
|
|
26
|
+
`This has never happened before.`,
|
|
27
|
+
`We'll keep trying, but it's not looking great.
|
|
28
|
+
Commenting & chat services may be down.
|
|
29
|
+
If you continue to experience issues, please contact support.
|
|
30
|
+
`
|
|
31
|
+
];
|
|
32
|
+
// Properties
|
|
33
|
+
this._signInSelected = new Subject();
|
|
34
|
+
this._permissionDeniedError = new Subject();
|
|
35
|
+
this._editAvatarSelected = new Subject();
|
|
17
36
|
this._upvoted = new Subject();
|
|
18
37
|
this._reported = new Subject();
|
|
19
38
|
this._selected = new Subject();
|
|
@@ -23,12 +42,8 @@ export class BantaCommentsComponent {
|
|
|
23
42
|
this._avatarSelected = new Subject();
|
|
24
43
|
this._subs = new Subscription();
|
|
25
44
|
this._sortOrder = CommentsOrder.NEWEST;
|
|
26
|
-
this.
|
|
27
|
-
|
|
28
|
-
{ hashtag: 'timeout', description: 'Cause a slow timeout error' },
|
|
29
|
-
{ hashtag: 'slow', description: 'Be slow when this message is posted' },
|
|
30
|
-
];
|
|
31
|
-
this.participants = [];
|
|
45
|
+
this.selectedMessageVisible = false;
|
|
46
|
+
// Inputs
|
|
32
47
|
this.signInLabel = 'Sign In';
|
|
33
48
|
this.sendLabel = 'Send';
|
|
34
49
|
this.replyLabel = 'Reply';
|
|
@@ -36,174 +51,289 @@ export class BantaCommentsComponent {
|
|
|
36
51
|
this.permissionDeniedLabel = 'Send';
|
|
37
52
|
this.postCommentLabel = 'Post a comment';
|
|
38
53
|
this.postReplyLabel = 'Post a reply';
|
|
39
|
-
this.
|
|
40
|
-
this.
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
54
|
+
this.participants = [];
|
|
55
|
+
this.hashtags = [
|
|
56
|
+
{ hashtag: 'error', description: 'Cause an error' },
|
|
57
|
+
{ hashtag: 'timeout', description: 'Cause a slow timeout error' },
|
|
58
|
+
{ hashtag: 'slow', description: 'Be slow when this message is posted' },
|
|
59
|
+
];
|
|
60
|
+
this.sendMessage = (message) => __awaiter(this, void 0, void 0, function* () {
|
|
61
|
+
var _a;
|
|
62
|
+
try {
|
|
63
|
+
const intercept = yield ((_a = this.shouldInterceptMessageSend) === null || _a === void 0 ? void 0 : _a.call(this, message, this.source));
|
|
64
|
+
if (!intercept) {
|
|
65
|
+
yield this.source.send(message);
|
|
66
|
+
}
|
|
67
|
+
if (this.source.sortOrder !== CommentsOrder.NEWEST) {
|
|
68
|
+
this.sortOrder = CommentsOrder.NEWEST;
|
|
69
|
+
}
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
catch (e) {
|
|
73
|
+
this.handleBackendException(e, 'Could not send: ');
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
this.sendReply = (message) => __awaiter(this, void 0, void 0, function* () {
|
|
77
|
+
var _b;
|
|
78
|
+
try {
|
|
79
|
+
const intercept = yield ((_b = this.shouldInterceptMessageSend) === null || _b === void 0 ? void 0 : _b.call(this, message, this.source));
|
|
80
|
+
if (!intercept) {
|
|
81
|
+
yield this.selectedMessageThread.send(message);
|
|
82
|
+
}
|
|
83
|
+
return true;
|
|
84
|
+
}
|
|
85
|
+
catch (e) {
|
|
86
|
+
this.handleBackendException(e, 'Could not send reply: ');
|
|
87
|
+
}
|
|
88
|
+
});
|
|
45
89
|
}
|
|
46
|
-
|
|
47
|
-
|
|
90
|
+
handleBackendExceptionAsAlert(e, prefix = '') {
|
|
91
|
+
try {
|
|
92
|
+
this.handleBackendException(e, prefix);
|
|
93
|
+
}
|
|
94
|
+
catch (e) {
|
|
95
|
+
alert(e.message);
|
|
96
|
+
}
|
|
48
97
|
}
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
98
|
+
handleBackendException(e, prefix = '') {
|
|
99
|
+
let errorMessage = e.message;
|
|
100
|
+
if (errorMessage.startsWith('permission-denied|')) {
|
|
101
|
+
errorMessage = errorMessage.replace(/^permission-denied\|/, '');
|
|
102
|
+
if (errorMessage.startsWith(`app-handle|`)) {
|
|
103
|
+
// If this is an error during authorizeAction on the backend, pass control to the user-provided
|
|
104
|
+
// permission-denied handler.
|
|
105
|
+
this.sendPermissionDenied(errorMessage.replace(/^app-handle\|/, ''));
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
55
108
|
}
|
|
109
|
+
throw new Error(`${prefix}${errorMessage}`);
|
|
56
110
|
}
|
|
111
|
+
// Lifecycle Events / Initialization
|
|
57
112
|
ngOnInit() {
|
|
58
|
-
this._subs.add(this.
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
if (comment.length > 0) {
|
|
69
|
-
// comment.item(0).scroll({behavior: 'smooth'});
|
|
70
|
-
comment.item(0).scrollIntoView();
|
|
113
|
+
this._subs.add(this.backend.userChanged.subscribe(user => this.user = user));
|
|
114
|
+
this.startLoading();
|
|
115
|
+
if (typeof window !== 'undefined') {
|
|
116
|
+
let queryString = window.location.search.substring(1);
|
|
117
|
+
let query = queryString.split('&')
|
|
118
|
+
.map(s => s.split('='))
|
|
119
|
+
.reduce((o, [k, v]) => (o[k] = v, o), {});
|
|
120
|
+
const commentID = query['comment'];
|
|
121
|
+
if (commentID) {
|
|
122
|
+
this.sharedCommentID = commentID;
|
|
71
123
|
}
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
checkForSharedComment() {
|
|
75
|
-
const commentID = this.activatedRoute.snapshot.queryParamMap.get('comment');
|
|
76
|
-
if (commentID)
|
|
77
|
-
this.scrollToComment(commentID);
|
|
124
|
+
}
|
|
78
125
|
}
|
|
79
126
|
ngOnDestroy() {
|
|
80
127
|
this._subs.unsubscribe();
|
|
81
128
|
}
|
|
82
|
-
get source() {
|
|
83
|
-
return this._source;
|
|
84
|
-
}
|
|
85
|
-
set source(value) {
|
|
86
|
-
this._source = value;
|
|
87
|
-
}
|
|
88
|
-
get topicID() {
|
|
89
|
-
return this._topicID;
|
|
90
|
-
}
|
|
91
|
-
set topicID(value) {
|
|
92
|
-
if (this._topicID !== value) {
|
|
93
|
-
this._topicID = value;
|
|
94
|
-
setTimeout(() => this.setSourceFromTopicID(value));
|
|
95
|
-
}
|
|
96
|
-
}
|
|
97
129
|
setSourceFromTopicID(topicID) {
|
|
98
|
-
var _a, _b;
|
|
99
130
|
return __awaiter(this, void 0, void 0, function* () {
|
|
100
|
-
|
|
101
|
-
|
|
131
|
+
if (this._source) {
|
|
132
|
+
this._source.close();
|
|
133
|
+
this._source = null;
|
|
134
|
+
}
|
|
102
135
|
setTimeout(() => __awaiter(this, void 0, void 0, function* () {
|
|
136
|
+
console.log(`[banta-comments] Subscribing to topic source '${topicID}'`);
|
|
103
137
|
this._source = yield this.backend.getSourceForTopic(topicID, { sortOrder: this.sortOrder });
|
|
104
|
-
|
|
138
|
+
if (this.sharedCommentID) {
|
|
139
|
+
this.navigateToSharedComment(this.sharedCommentID);
|
|
140
|
+
this.sharedCommentID = null;
|
|
141
|
+
}
|
|
105
142
|
this._source.messageReceived.subscribe(m => this.addParticipant(m));
|
|
106
143
|
this._source.messageSent.subscribe(m => this.addParticipant(m));
|
|
107
144
|
this._source.messages.forEach(m => this.addParticipant(m));
|
|
108
145
|
}));
|
|
109
146
|
});
|
|
110
147
|
}
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
return;
|
|
114
|
-
let existing = this.participants.find(x => x.id === message.user.id);
|
|
115
|
-
if (existing)
|
|
116
|
-
return;
|
|
117
|
-
this.participants.push(message.user);
|
|
118
|
-
}
|
|
119
|
-
showSignIn() {
|
|
120
|
-
this._signInSelected.next();
|
|
121
|
-
}
|
|
122
|
-
showEditAvatar() {
|
|
123
|
-
this._editAvatarSelected.next();
|
|
124
|
-
}
|
|
125
|
-
get newMessageText() {
|
|
126
|
-
return this._newMessageText;
|
|
127
|
-
}
|
|
128
|
-
set newMessageText(value) {
|
|
129
|
-
this._newMessageText = value;
|
|
130
|
-
if (this._newMessageText === '' && this.sendError)
|
|
131
|
-
setTimeout(() => this.sendError = null);
|
|
132
|
-
}
|
|
133
|
-
get signInSelected() {
|
|
134
|
-
return this._signInSelected;
|
|
135
|
-
}
|
|
136
|
-
get editAvatarSelected() {
|
|
137
|
-
return this._editAvatarSelected;
|
|
148
|
+
get loadingMessage() {
|
|
149
|
+
return this._loadingMessage;
|
|
138
150
|
}
|
|
139
|
-
|
|
140
|
-
|
|
151
|
+
set loadingMessage(value) {
|
|
152
|
+
this.loadingMessageVisible = false;
|
|
153
|
+
setTimeout(() => {
|
|
154
|
+
this._loadingMessage = value;
|
|
155
|
+
this._loadingMessage = value;
|
|
156
|
+
setTimeout(() => {
|
|
157
|
+
this.loadingMessageVisible = true;
|
|
158
|
+
});
|
|
159
|
+
}, 500);
|
|
141
160
|
}
|
|
142
|
-
|
|
143
|
-
this
|
|
161
|
+
startLoading() {
|
|
162
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
163
|
+
this.loadingStartedAt = this.messageChangedAt = Date.now();
|
|
164
|
+
if (this.updateLoading())
|
|
165
|
+
return;
|
|
166
|
+
yield new Promise(resolve => setTimeout(() => resolve(), 100));
|
|
167
|
+
if (this.updateLoading())
|
|
168
|
+
return;
|
|
169
|
+
yield new Promise(resolve => setTimeout(() => resolve(), 250));
|
|
170
|
+
if (this.updateLoading())
|
|
171
|
+
return;
|
|
172
|
+
yield new Promise(resolve => setTimeout(() => resolve(), 500));
|
|
173
|
+
if (this.updateLoading())
|
|
174
|
+
return;
|
|
175
|
+
console.log(`[Banta] Loading is taking a long time! Showing loading screen.`);
|
|
176
|
+
this.showLoadingScreen = true;
|
|
177
|
+
if (typeof window !== 'undefined')
|
|
178
|
+
this._loadingTimer = setInterval(() => this.updateLoading(), 1000);
|
|
179
|
+
});
|
|
144
180
|
}
|
|
145
|
-
|
|
146
|
-
var _a;
|
|
147
|
-
if (
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
181
|
+
updateLoading() {
|
|
182
|
+
var _a, _b;
|
|
183
|
+
if (((_a = this.source) === null || _a === void 0 ? void 0 : _a.state) && ((_b = this.source) === null || _b === void 0 ? void 0 : _b.state) !== 'connecting') {
|
|
184
|
+
clearInterval(this._loadingTimer);
|
|
185
|
+
this.loadingMessage = `Here we go!`;
|
|
186
|
+
setTimeout(() => {
|
|
187
|
+
this.loading = false;
|
|
188
|
+
}, 750);
|
|
152
189
|
return true;
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
return this._selected.asObservable();
|
|
190
|
+
}
|
|
191
|
+
let messageSwitchTime = 5 * 1000;
|
|
192
|
+
if (this.messageChangedAt + messageSwitchTime < Date.now()) {
|
|
193
|
+
if (this.loadingMessages[this._loadingMessageIndex]) {
|
|
194
|
+
this.loadingMessage = this.loadingMessages[this._loadingMessageIndex++];
|
|
195
|
+
this.messageChangedAt = Date.now();
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
return false;
|
|
163
199
|
}
|
|
164
|
-
get
|
|
165
|
-
|
|
200
|
+
get source() { return this._source; }
|
|
201
|
+
set source(value) {
|
|
202
|
+
this._source = value;
|
|
203
|
+
if (value && this.sharedCommentID) {
|
|
204
|
+
this.navigateToSharedComment(this.sharedCommentID);
|
|
205
|
+
this.sharedCommentID = null;
|
|
206
|
+
}
|
|
166
207
|
}
|
|
167
|
-
get
|
|
168
|
-
|
|
208
|
+
get topicID() { return this._topicID; }
|
|
209
|
+
set topicID(value) {
|
|
210
|
+
if (this._topicID !== value) {
|
|
211
|
+
this._topicID = value;
|
|
212
|
+
setTimeout(() => this.setSourceFromTopicID(value));
|
|
213
|
+
}
|
|
169
214
|
}
|
|
170
|
-
|
|
171
|
-
|
|
215
|
+
// Outputs
|
|
216
|
+
get signInSelected() { return this._signInSelected; }
|
|
217
|
+
get editAvatarSelected() { return this._editAvatarSelected; }
|
|
218
|
+
get permissionDeniedError() { return this._permissionDeniedError; }
|
|
219
|
+
get upvoted() { return this._upvoted.asObservable(); }
|
|
220
|
+
get reported() { return this._reported.asObservable(); }
|
|
221
|
+
get selected() { return this._selected.asObservable(); }
|
|
222
|
+
get userSelected() { return this._userSelected.asObservable(); }
|
|
223
|
+
get usernameSelected() { return this._usernameSelected.asObservable(); }
|
|
224
|
+
get avatarSelected() { return this._avatarSelected.asObservable(); }
|
|
225
|
+
get shared() { return this._shared.asObservable(); }
|
|
226
|
+
get sortOrder() { return this._sortOrder; }
|
|
227
|
+
set sortOrder(value) {
|
|
228
|
+
if (this._sortOrder !== value) {
|
|
229
|
+
this._sortOrder = value;
|
|
230
|
+
setTimeout(() => {
|
|
231
|
+
this.setSourceFromTopicID(this.topicID);
|
|
232
|
+
});
|
|
233
|
+
}
|
|
172
234
|
}
|
|
173
|
-
|
|
174
|
-
|
|
235
|
+
// UI Interactions
|
|
236
|
+
scrollToComment(commentId) {
|
|
237
|
+
setTimeout(() => {
|
|
238
|
+
const comment = document.querySelectorAll(`[data-comment-id="${commentId}"]`);
|
|
239
|
+
if (comment.length > 0) {
|
|
240
|
+
// comment.item(0).scroll({behavior: 'smooth'});
|
|
241
|
+
comment.item(0).scrollIntoView();
|
|
242
|
+
}
|
|
243
|
+
}, 1000);
|
|
175
244
|
}
|
|
176
|
-
|
|
245
|
+
navigateToSharedComment(id) {
|
|
246
|
+
var _a, _b, _c;
|
|
247
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
248
|
+
let source = this.source;
|
|
249
|
+
yield source.ready;
|
|
250
|
+
console.log(`Navigating to shared comment with ID '${id}'...`);
|
|
251
|
+
let message;
|
|
252
|
+
try {
|
|
253
|
+
message = yield this.source.get(id);
|
|
254
|
+
}
|
|
255
|
+
catch (e) {
|
|
256
|
+
console.error(`Failed to find comment from URL: ${e.message}`);
|
|
257
|
+
return;
|
|
258
|
+
}
|
|
259
|
+
(_a = message.transientState) !== null && _a !== void 0 ? _a : (message.transientState = {});
|
|
260
|
+
// If there is a parent message, we should instead focus that and let the
|
|
261
|
+
// scrollToComment and highlight do the work.
|
|
262
|
+
if (message.parentMessageId) {
|
|
263
|
+
let parentMessage = yield this.source.get(message.parentMessageId);
|
|
264
|
+
(_b = parentMessage.transientState) !== null && _b !== void 0 ? _b : (parentMessage.transientState = {});
|
|
265
|
+
let thread = yield this.selectMessage(parentMessage);
|
|
266
|
+
// Need to re-retrieve the message within the new chat source to affect its
|
|
267
|
+
// transient state.
|
|
268
|
+
yield thread.ready;
|
|
269
|
+
message = yield thread.get(message.id);
|
|
270
|
+
(_c = message.transientState) !== null && _c !== void 0 ? _c : (message.transientState = {});
|
|
271
|
+
message.transientState.highlighted = true;
|
|
272
|
+
console.dir(message);
|
|
273
|
+
yield new Promise(resolve => setTimeout(() => resolve(), 500));
|
|
274
|
+
}
|
|
275
|
+
else {
|
|
276
|
+
this.selectMessage(message);
|
|
277
|
+
}
|
|
278
|
+
this.scrollToComment(id);
|
|
279
|
+
});
|
|
177
280
|
}
|
|
178
|
-
|
|
179
|
-
this.
|
|
281
|
+
sendPermissionDenied(message) {
|
|
282
|
+
this._permissionDeniedError.next(message);
|
|
180
283
|
}
|
|
181
|
-
|
|
284
|
+
scrollToMessage(message) {
|
|
285
|
+
let el = this.elementRef.nativeElement.querySelector(`[data-comment-id="${message.id}"]`);
|
|
286
|
+
if (!el)
|
|
287
|
+
return;
|
|
288
|
+
el.scrollIntoView({ block: 'center', inline: 'start' });
|
|
182
289
|
}
|
|
183
|
-
|
|
184
|
-
|
|
290
|
+
addParticipant(message) {
|
|
291
|
+
if (!message || !message.user || !message.user.id)
|
|
292
|
+
return;
|
|
293
|
+
let existing = this.participants.find(x => x.id === message.user.id);
|
|
294
|
+
if (existing)
|
|
295
|
+
return;
|
|
296
|
+
this.participants.push(message.user);
|
|
185
297
|
}
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
298
|
+
// Actions
|
|
299
|
+
likeMessage(source, message) {
|
|
300
|
+
var _a;
|
|
301
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
302
|
+
this._upvoted.next(message);
|
|
303
|
+
message.transientState.liking = true;
|
|
304
|
+
if (!((_a = message.userState) === null || _a === void 0 ? void 0 : _a.liked))
|
|
305
|
+
message.likes = (message.likes || 0) + 1;
|
|
306
|
+
try {
|
|
307
|
+
yield source.likeMessage(message.id);
|
|
308
|
+
}
|
|
309
|
+
catch (e) {
|
|
310
|
+
this.handleBackendExceptionAsAlert(e, 'Could not like this message: ');
|
|
311
|
+
}
|
|
312
|
+
yield new Promise(resolve => setTimeout(() => resolve(), 250));
|
|
313
|
+
message.transientState.liking = false;
|
|
193
314
|
});
|
|
194
315
|
}
|
|
195
|
-
|
|
316
|
+
unlikeMessage(source, message) {
|
|
317
|
+
var _a;
|
|
196
318
|
return __awaiter(this, void 0, void 0, function* () {
|
|
197
319
|
this._upvoted.next(message);
|
|
198
|
-
message
|
|
199
|
-
|
|
200
|
-
|
|
320
|
+
message.transientState.liking = true;
|
|
321
|
+
if ((_a = message.userState) === null || _a === void 0 ? void 0 : _a.liked)
|
|
322
|
+
message.likes = (message.likes || 0) - 1;
|
|
323
|
+
try {
|
|
324
|
+
yield source.unlikeMessage(message.id);
|
|
325
|
+
}
|
|
326
|
+
catch (e) {
|
|
327
|
+
this.handleBackendExceptionAsAlert(e, 'Failed to unlike this message: ');
|
|
328
|
+
}
|
|
201
329
|
yield new Promise(resolve => setTimeout(() => resolve(), 250));
|
|
202
|
-
message
|
|
330
|
+
message.transientState.liking = false;
|
|
203
331
|
});
|
|
204
332
|
}
|
|
205
333
|
reportMessage(message) {
|
|
206
|
-
this
|
|
334
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
335
|
+
this._reported.next(message);
|
|
336
|
+
});
|
|
207
337
|
}
|
|
208
338
|
unselectMessage() {
|
|
209
339
|
return __awaiter(this, void 0, void 0, function* () {
|
|
@@ -223,70 +353,101 @@ export class BantaCommentsComponent {
|
|
|
223
353
|
return __awaiter(this, void 0, void 0, function* () {
|
|
224
354
|
this._selected.next(message);
|
|
225
355
|
this.selectedMessage = message;
|
|
356
|
+
let selectedMessageThread = yield this.backend.getSourceForThread(this.topicID, message.id);
|
|
226
357
|
setTimeout(() => this.selectedMessageVisible = true);
|
|
227
358
|
setTimeout(() => __awaiter(this, void 0, void 0, function* () {
|
|
228
|
-
this.selectedMessageThread =
|
|
359
|
+
this.selectedMessageThread = selectedMessageThread;
|
|
229
360
|
}), 250);
|
|
361
|
+
return selectedMessageThread;
|
|
362
|
+
});
|
|
363
|
+
}
|
|
364
|
+
showSignIn() {
|
|
365
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
366
|
+
this._signInSelected.next();
|
|
367
|
+
});
|
|
368
|
+
}
|
|
369
|
+
showEditAvatar() {
|
|
370
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
371
|
+
this._editAvatarSelected.next();
|
|
230
372
|
});
|
|
231
373
|
}
|
|
232
374
|
selectMessageUser(message) {
|
|
233
|
-
this
|
|
375
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
376
|
+
this._userSelected.next(message);
|
|
377
|
+
});
|
|
234
378
|
}
|
|
235
379
|
selectUsername(user) {
|
|
236
|
-
this
|
|
380
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
381
|
+
this._usernameSelected.next(user);
|
|
382
|
+
});
|
|
237
383
|
}
|
|
238
384
|
selectAvatar(user) {
|
|
239
|
-
this
|
|
385
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
386
|
+
this._avatarSelected.next(user);
|
|
387
|
+
});
|
|
240
388
|
}
|
|
241
389
|
shareMessage(message) {
|
|
242
|
-
this
|
|
390
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
391
|
+
this._shared.next(message);
|
|
392
|
+
});
|
|
243
393
|
}
|
|
244
|
-
|
|
394
|
+
deleteMessage(message) {
|
|
245
395
|
return __awaiter(this, void 0, void 0, function* () {
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
sentAt: Date.now(),
|
|
255
|
-
updatedAt: Date.now()
|
|
256
|
-
});
|
|
257
|
-
this.replyMessage = '';
|
|
396
|
+
if (!confirm("Are you sure you want to delete this comment? You cannot undo this action."))
|
|
397
|
+
return;
|
|
398
|
+
try {
|
|
399
|
+
yield this.source.deleteMessage(message.id);
|
|
400
|
+
}
|
|
401
|
+
catch (e) {
|
|
402
|
+
this.handleBackendExceptionAsAlert(e, `Could not delete message: `);
|
|
403
|
+
}
|
|
258
404
|
});
|
|
259
405
|
}
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
406
|
+
editMessage(source, message, newText) {
|
|
407
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
408
|
+
try {
|
|
409
|
+
yield source.editMessage(message.id, newText);
|
|
410
|
+
}
|
|
411
|
+
catch (e) {
|
|
412
|
+
this.handleBackendExceptionAsAlert(e, 'Could not edit this message: ');
|
|
413
|
+
return;
|
|
414
|
+
}
|
|
415
|
+
message.message = newText;
|
|
416
|
+
message.transientState.editing = false;
|
|
417
|
+
});
|
|
418
|
+
}
|
|
419
|
+
startEditing(message) {
|
|
420
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
421
|
+
this.selectedMessage.transientState.editing = false;
|
|
422
|
+
message.transientState.editing = true;
|
|
423
|
+
});
|
|
424
|
+
}
|
|
425
|
+
saveEdit(message, text) {
|
|
426
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
427
|
+
try {
|
|
428
|
+
yield this.source.editMessage(message.id, text);
|
|
429
|
+
message.transientState.editing = false;
|
|
430
|
+
}
|
|
431
|
+
catch (e) {
|
|
432
|
+
this.handleBackendExceptionAsAlert(e, `Could not edit message: `);
|
|
433
|
+
}
|
|
434
|
+
});
|
|
265
435
|
}
|
|
266
436
|
}
|
|
267
437
|
BantaCommentsComponent.decorators = [
|
|
268
438
|
{ type: Component, args: [{
|
|
269
439
|
selector: 'banta-comments',
|
|
270
|
-
template: "\r\n<div class=\"focused\" [class.visible]=\"selectedMessageVisible\" *ngIf=\"selectedMessage\">\r\n\r\n
|
|
271
|
-
styles: [":host{display:flex;flex-direction:column}@-webkit-keyframes select-comment{0%{transform:scale(1.15)}to{transform:scale(1)}}@keyframes select-comment{0%{transform:scale(1.15)}to{transform:scale(1)}}.focused{-webkit-animation-name:select-comment;animation-name:select-comment;-webkit-animation-duration:.4s;animation-duration:.4s;-webkit-animation-fill-mode:both;animation-fill-mode:both}.focused .replies{margin-top:1em;margin-left:4em}banta-comment-view{opacity:1;transition:opacity .4s ease-in-out}banta-comment-view.faded{opacity:.25}.loading{display:block;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;margin:0 auto;min-height:16em}.main.hidden{display:none}@media (max-width:500px){.focused .replies{margin-left:0}}"]
|
|
440
|
+
template: "<ng-container *ngIf=\"loading\">\r\n <div class=\"loading-screen\" [class.visible]=\"showLoadingScreen\">\r\n <h1>Loading Comments</h1>\r\n <div>\r\n <mat-spinner [diameter]=\"300\" [strokeWidth]=\"2\"></mat-spinner>\r\n </div>\r\n\r\n <p class=\"loading-message\" [class.visible]=\"loadingMessageVisible\">{{loadingMessage}}</p>\r\n </div>\r\n</ng-container>\r\n<ng-container *ngIf=\"!loading\">\r\n <div class=\"focused\" [class.visible]=\"selectedMessageVisible\" *ngIf=\"selectedMessage\">\r\n\r\n <div>\r\n <a mat-button href=\"javascript:;\" (click)=\"unselectMessage()\">\r\n <mat-icon>arrow_back</mat-icon>\r\n Latest Comments\r\n </a>\r\n </div>\r\n\r\n <banta-comment\r\n [message]=\"selectedMessage\"\r\n [liking]=\"selectedMessage.transientState.liking\"\r\n [mine]=\"user?.id === selectedMessage.user?.id\"\r\n [permissions]=\"source?.permissions\"\r\n [showReplyAction]=\"false\"\r\n [editing]=\"selectedMessage.transientState.editing\"\r\n (editStarted)=\"startEditing(selectedMessage)\"\r\n (editEnded)=\"selectedMessage.transientState.editing = false\"\r\n (edited)=\"saveEdit(selectedMessage, $event)\"\r\n (userSelected)=\"selectMessageUser(selectedMessage)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (liked)=\"likeMessage(source, selectedMessage)\"\r\n (unliked)=\"unlikeMessage(source, selectedMessage)\"\r\n (reported)=\"reportMessage(selectedMessage)\"\r\n (selected)=\"selectMessage(selectedMessage)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage(selectedMessage)\"\r\n ></banta-comment>\r\n\r\n <div class=\"replies\">\r\n\r\n <ng-container *ngIf=\"!selectedMessageThread\">\r\n <div class=\"loading\">\r\n <mat-spinner></mat-spinner>\r\n </div>\r\n </ng-container>\r\n\r\n <ng-container *ngIf=\"selectedMessageThread\">\r\n <banta-comment-view\r\n [source]=\"selectedMessageThread\"\r\n [allowReplies]=\"false\"\r\n [fixedHeight]=\"false\"\r\n [showEmptyState]=\"false\"\r\n [newestLast]=\"true\"\r\n (liked)=\"likeMessage(selectedMessageThread, $event)\"\r\n (unliked)=\"unlikeMessage(selectedMessageThread, $event)\"\r\n (messageEdited)=\"editMessage(selectedMessageThread, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage($event)\"\r\n ></banta-comment-view>\r\n\r\n <banta-comment-field\r\n [sendLabel]=\"replyLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n (signInSelected)=\"showSignIn()\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n [source]=\"selectedMessageThread\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [signInLabel]=\"signInLabel\"\r\n [permissionDeniedLabel]=\"permissionDeniedLabel\"\r\n (permissionDeniedError)=\"sendPermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [user]=\"user\"\r\n [label]=\"postReplyLabel\"\r\n [submit]=\"sendReply\"\r\n >\r\n <ng-content select=\".reply-send-options\"></ng-content>\r\n </banta-comment-field>\r\n </ng-container>\r\n </div>\r\n </div>\r\n\r\n <div class=\"main\" [class.hidden]=\"selectedMessage\">\r\n <banta-comment-field\r\n [source]=\"source\"\r\n [user]=\"user\"\r\n [sendLabel]=\"sendLabel\"\r\n [sendingLabel]=\"sendingLabel\"\r\n [signInLabel]=\"signInLabel\"\r\n [canComment]=\"source?.permissions?.canPost\"\r\n [hashtags]=\"hashtags\"\r\n [participants]=\"participants\"\r\n [label]=\"postCommentLabel\"\r\n (editAvatarSelected)=\"showEditAvatar()\"\r\n (signInSelected)=\"showSignIn()\"\r\n [permissionDeniedLabel]=\"permissionDeniedLabel\"\r\n (permissionDeniedError)=\"sendPermissionDenied($event)\"\r\n [shouldInterceptMessageSend]=\"shouldInterceptMessageSend\"\r\n [submit]=\"sendMessage\"\r\n >\r\n \r\n </banta-comment-field>\r\n\r\n <banta-comment-sort\r\n [(sort)]=\"sortOrder\"></banta-comment-sort>\r\n\r\n <banta-comment-view\r\n [class.faded]=\"selectedMessage\"\r\n [source]=\"source\"\r\n [fixedHeight]=\"fixedHeight\"\r\n [maxMessages]=\"maxMessages\"\r\n [maxVisibleMessages]=\"maxVisibleMessages\"\r\n [genericAvatarUrl]=\"genericAvatarUrl\"\r\n (userSelected)=\"selectMessageUser($event)\"\r\n (sortOrderChanged)=\"sortOrder = $event\"\r\n (selected)=\"selectMessage($event)\"\r\n (liked)=\"likeMessage(source, $event)\"\r\n (unliked)=\"unlikeMessage(source, $event)\"\r\n (messageEdited)=\"editMessage(source, $event.message, $event.newMessage)\"\r\n (reported)=\"reportMessage($event)\"\r\n (usernameSelected)=\"selectUsername($event)\"\r\n (avatarSelected)=\"selectAvatar($event)\"\r\n (shared)=\"shareMessage($event)\"\r\n (deleted)=\"deleteMessage($event)\"\r\n ></banta-comment-view>\r\n </div>\r\n</ng-container>",
|
|
441
|
+
styles: [":host{display:flex;flex-direction:column}@-webkit-keyframes select-comment{0%{transform:scale(1.15)}to{transform:scale(1)}}@keyframes select-comment{0%{transform:scale(1.15)}to{transform:scale(1)}}.focused{-webkit-animation-name:select-comment;animation-name:select-comment;-webkit-animation-duration:.4s;animation-duration:.4s;-webkit-animation-fill-mode:both;animation-fill-mode:both}.focused .replies{margin-top:1em;margin-left:4em}banta-comment-view{opacity:1;transition:opacity .4s ease-in-out}banta-comment-view.faded{opacity:.25}.loading{display:block;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;margin:0 auto;min-height:16em}.main.hidden{display:none}@media (max-width:500px){.focused .replies{margin-left:0}}.loading-screen{text-align:center;opacity:0;transition:opacity .25s ease-in-out}.loading-screen.visible{opacity:1}.loading-screen h1{font-weight:100}.loading-screen mat-spinner{margin:5em auto}.loading-screen .loading-message{opacity:0;transition:opacity .25s ease-in-out;width:500px;max-width:100%;margin:0 auto}.loading-screen .loading-message.visible{opacity:1}"]
|
|
272
442
|
},] }
|
|
273
443
|
];
|
|
274
444
|
BantaCommentsComponent.ctorParameters = () => [
|
|
275
|
-
{ type:
|
|
276
|
-
{ type: ChatBackendService },
|
|
445
|
+
{ type: ChatBackendBase },
|
|
277
446
|
{ type: ElementRef },
|
|
278
447
|
{ type: ActivatedRoute }
|
|
279
448
|
];
|
|
280
449
|
BantaCommentsComponent.propDecorators = {
|
|
281
|
-
|
|
282
|
-
participants: [{ type: Input }],
|
|
283
|
-
source: [{ type: Input }],
|
|
284
|
-
fixedHeight: [{ type: Input }],
|
|
285
|
-
maxMessages: [{ type: Input }],
|
|
286
|
-
maxVisibleMessages: [{ type: Input }],
|
|
287
|
-
genericAvatarUrl: [{ type: Input }],
|
|
288
|
-
shouldInterceptMessageSend: [{ type: Input }],
|
|
289
|
-
topicID: [{ type: Input }],
|
|
450
|
+
loadingMessages: [{ type: Input }],
|
|
290
451
|
signInLabel: [{ type: Input }],
|
|
291
452
|
sendLabel: [{ type: Input }],
|
|
292
453
|
replyLabel: [{ type: Input }],
|
|
@@ -294,6 +455,15 @@ BantaCommentsComponent.propDecorators = {
|
|
|
294
455
|
permissionDeniedLabel: [{ type: Input }],
|
|
295
456
|
postCommentLabel: [{ type: Input }],
|
|
296
457
|
postReplyLabel: [{ type: Input }],
|
|
458
|
+
fixedHeight: [{ type: Input }],
|
|
459
|
+
maxMessages: [{ type: Input }],
|
|
460
|
+
maxVisibleMessages: [{ type: Input }],
|
|
461
|
+
genericAvatarUrl: [{ type: Input }],
|
|
462
|
+
shouldInterceptMessageSend: [{ type: Input }],
|
|
463
|
+
participants: [{ type: Input }],
|
|
464
|
+
source: [{ type: Input }],
|
|
465
|
+
hashtags: [{ type: Input }],
|
|
466
|
+
topicID: [{ type: Input }],
|
|
297
467
|
signInSelected: [{ type: Output }],
|
|
298
468
|
editAvatarSelected: [{ type: Output }],
|
|
299
469
|
permissionDeniedError: [{ type: Output }],
|
|
@@ -305,4 +475,4 @@ BantaCommentsComponent.propDecorators = {
|
|
|
305
475
|
avatarSelected: [{ type: Output }],
|
|
306
476
|
shared: [{ type: Output }]
|
|
307
477
|
};
|
|
308
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"banta-comments.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/comments/banta-comments/banta-comments.component.ts"],"names":[],"mappings":";AAAA,OAAO,EAAiB,SAAS,EAAE,UAAU,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACpF,OAAO,EAA8C,aAAa,EAAE,MAAM,eAAe,CAAC;AAE1F,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AACzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAEjD;;GAEG;AAMH,MAAM,OAAO,sBAAsB;IAC/B,YACY,KAAmB,EACnB,OAA2B,EAC3B,UAAmC,EACnC,cAA8B;QAH9B,UAAK,GAAL,KAAK,CAAc;QACnB,YAAO,GAAP,OAAO,CAAoB;QAC3B,eAAU,GAAV,UAAU,CAAyB;QACnC,mBAAc,GAAd,cAAc,CAAgB;QAIlC,aAAQ,GAAG,IAAI,OAAO,EAAe,CAAC;QACtC,cAAS,GAAG,IAAI,OAAO,EAAe,CAAC;QACvC,cAAS,GAAG,IAAI,OAAO,EAAe,CAAC;QACvC,kBAAa,GAAG,IAAI,OAAO,EAAe,CAAC;QAC3C,YAAO,GAAG,IAAI,OAAO,EAAe,CAAC;QAErC,sBAAiB,GAAG,IAAI,OAAO,EAAQ,CAAC;QACxC,oBAAe,GAAG,IAAI,OAAO,EAAQ,CAAC;QAItC,UAAK,GAAG,IAAI,YAAY,EAAE,CAAC;QAEnC,eAAU,GAAkB,aAAa,CAAC,MAAM,CAAC;QAexC,aAAQ,GAAc;YAC3B,EAAC,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,gBAAgB,EAAC;YACjD,EAAC,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,4BAA4B,EAAC;YAC/D,EAAC,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,qCAAqC,EAAC;SACxE,CAAC;QAEO,iBAAY,GAAW,EAAE,CAAC;QA4G1B,gBAAW,GAAG,SAAS,CAAC;QACxB,cAAS,GAAG,MAAM,CAAC;QACnB,eAAU,GAAG,OAAO,CAAC;QACrB,iBAAY,GAAG,SAAS,CAAC;QACzB,0BAAqB,GAAG,MAAM,CAAC;QAC/B,qBAAgB,GAAG,gBAAgB,CAAC;QACpC,mBAAc,GAAG,cAAc,CAAC;QAEjC,oBAAe,GAAG,IAAI,OAAO,EAAQ,CAAC;QACtC,2BAAsB,GAAG,IAAI,OAAO,EAAQ,CAAC;QAC7C,wBAAmB,GAAG,IAAI,OAAO,EAAQ,CAAC;QAmFlD,YAAO,GAAG,KAAK,CAAC;QAEhB,gBAAW,GAAG,KAAK,CAAC;QA6CpB,2BAAsB,GAAG,KAAK,CAAC;IA5R/B,CAAC;IAiBD,IAAI,SAAS;QACT,OAAO,IAAI,CAAC,UAAU,CAAC;IAC3B,CAAC;IAED,IAAI,SAAS,CAAC,KAAK;QACf,IAAI,IAAI,CAAC,UAAU,KAAK,KAAK,EAAE;YAC3B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;YACxB,UAAU,CAAC,GAAG,EAAE;gBACZ,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC5C,CAAC,CAAC,CAAC;SACN;IACL,CAAC;IAUD,QAAQ;QACJ,IAAI,CAAC,KAAK,CAAC,GAAG,CACV,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,CAC7D,CAAC;IACN,CAAC;IAED,eAAe;QACX,IAAI,OAAO,MAAM,KAAK,WAAW;YAAE,IAAI,CAAC,qBAAqB,EAAE,CAAC;IACpE,CAAC;IAED,eAAe,CAAC,SAA4B;QACxC,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,OAAO,GAAG,QAAQ,CAAC,gBAAgB,CAAC,qBAAqB,SAAS,IAAI,CAAC,CAAC;YAC9E,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAA;YACpB,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE;gBACtB,gDAAgD;gBAChD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,CAAC;aAClC;QACH,CAAC,EAAE,IAAI,CAAC,CAAC;IACb,CAAC;IAED,qBAAqB;QACjB,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC5E,IAAI,SAAS;YAAE,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC;IACnD,CAAC;IAED,WAAW;QACP,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;IAC7B,CAAC;IAED,IACI,MAAM;QACN,OAAO,IAAI,CAAC,OAAO,CAAC;IACxB,CAAC;IAED,IAAI,MAAM,CAAC,KAAK;QACZ,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;IACzB,CAAC;IASD,IACI,OAAO;QACP,OAAO,IAAI,CAAC,QAAQ,CAAC;IACzB,CAAC;IAID,IAAI,OAAO,CAAC,KAAK;QACb,IAAI,IAAI,CAAC,QAAQ,KAAK,KAAK,EAAE;YACzB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC,CAAC;SACtD;IACL,CAAC;IAEa,oBAAoB,CAAC,OAAe;;;YAC9C,YAAA,IAAI,CAAC,OAAO,0CAAE,KAAK,mDAAK;YACxB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;YACpB,UAAU,CAAC,GAAS,EAAE;gBAClB,IAAI,CAAC,OAAO,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,iBAAiB,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;gBAE5F,OAAO,CAAC,GAAG,CAAC,qDAAqD,OAAO,GAAG,CAAC,CAAC;gBAE7E,IAAI,CAAC,OAAO,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;gBACpE,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;gBAChE,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;YAC/D,CAAC,CAAA,CAAC,CAAC;;KACN;IAEO,cAAc,CAAC,OAAoB;QACvC,IAAI,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YAC7C,OAAO;QAEX,IAAI,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACrE,IAAI,QAAQ;YACR,OAAO;QACX,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IACzC,CAAC;IAED,UAAU;QACN,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC;IAChC,CAAC;IAED,cAAc;QACV,IAAI,CAAC,mBAAmB,CAAC,IAAI,EAAE,CAAC;IACpC,CAAC;IAMD,IAAI,cAAc;QACd,OAAO,IAAI,CAAC,eAAe,CAAC;IAChC,CAAC;IAED,IAAI,cAAc,CAAC,KAAK;QACpB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;QAC7B,IAAI,IAAI,CAAC,eAAe,KAAK,EAAE,IAAI,IAAI,CAAC,SAAS;YAC7C,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,CAAC;IAChD,CAAC;IAcD,IACI,cAAc;QACd,OAAO,IAAI,CAAC,eAAe,CAAC;IAChC,CAAC;IAED,IACI,kBAAkB;QAClB,OAAO,IAAI,CAAC,mBAAmB,CAAC;IACpC,CAAC;IAED,IACI,qBAAqB;QACrB,OAAO,IAAI,CAAC,sBAAsB,CAAC;IACvC,CAAC;IAED,oBAAoB;QAChB,IAAI,CAAC,sBAAsB,CAAC,IAAI,EAAE,CAAC;IACvC,CAAC;IAED,IAAI,UAAU;;QACV,IAAI,CAAC,IAAI,CAAC,IAAI;YACV,OAAO,KAAK,CAAC;QAEjB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW;YACtB,OAAO,IAAI,CAAC;QAEhB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,UAAU;YACjC,OAAO,IAAI,CAAC;QAEhB,aAAO,IAAI,CAAC,IAAI,CAAC,WAAW,0CAAE,UAAU,CAAC,IAAI,CAAC,MAAM,EAAE;IAC1D,CAAC;IAED,IACI,OAAO;QACP,OAAO,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAE,CAAC;IACxC,CAAC;IAED,IACI,QAAQ;QACR,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;IACzC,CAAC;IAED,IACI,QAAQ;QACR,OAAO,IAAI,CAAC,SAAS,CAAC,YAAY,EAAE,CAAC;IACzC,CAAC;IAED,IACI,YAAY;QACZ,OAAO,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;IAC7C,CAAC;IAED,IACI,gBAAgB;QAChB,OAAO,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;IACjD,CAAC;IAED,IACI,cAAc;QACd,OAAO,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;IAC/C,CAAC;IAED,IACI,MAAM;QACN,OAAO,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;IACvC,CAAC;IAED,SAAS,CAAC,KAAoB;IAC9B,CAAC;IAED,WAAW,CAAC,IAAY;QACpB,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC;IAChC,CAAC;IAED,cAAc,CAAC,KAAoB;IACnC,CAAC;IAED,gBAAgB,CAAC,IAAY;QACzB,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC;IAC9B,CAAC;IAMD,aAAa,CAAC,OAAe;QACzB,IAAI,CAAC,SAAS,GAAG,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC;QACpC,UAAU,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;YACxB,UAAU,CAAC,GAAG,EAAE;gBACZ,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;YAC7B,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,CAAC;QACjB,CAAC,CAAC,CAAC;IACP,CAAC;IAEK,aAAa,CAAC,OAAoB;;YACpC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC3B,OAAe,CAAC,SAAS,GAAG,IAAI,CAAC;YAClC,OAAO,CAAC,OAAO,GAAG,CAAC,OAAO,CAAC,OAAO,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;YAC7C,MAAM,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;YACpK,MAAM,IAAI,OAAO,CAAO,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;YACpE,OAAe,CAAC,SAAS,GAAG,KAAK,CAAC;QACvC,CAAC;KAAA;IAED,aAAa,CAAC,OAAoB;QAC9B,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACjC,CAAC;IAOK,eAAe;;YACjB,IAAI,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC;YAEnC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC1B,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;YAC5B,IAAI,IAAI,CAAC,qBAAqB,EAAE;gBAC5B,IAAI,IAAI,CAAC,qBAAqB,CAAC,KAAK;oBAChC,IAAI,CAAC,qBAAqB,CAAC,KAAK,EAAE,CAAC;gBACvC,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC;aACrC;YAED,IAAI,OAAO;gBACP,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,CAAC;QACxD,CAAC;KAAA;IAIK,aAAa,CAAC,OAAoB;;YACpC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC7B,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;YAC/B,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,CAAC;YACrD,UAAU,CAAC,GAAS,EAAE;gBAClB,IAAI,CAAC,qBAAqB,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;YACjG,CAAC,CAAA,EAAE,GAAG,CAAC,CAAC;QACZ,CAAC;KAAA;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,YAAY,CAAC,OAAoB;QAC7B,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC/B,CAAC;IAEK,SAAS;;YACX,MAAM,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;gBAClC,OAAO,EAAE,IAAI,CAAC,YAAY;gBAC1B,eAAe,EAAE,IAAI,CAAC,eAAe,CAAC,EAAE;gBACxC,OAAO,EAAE,CAAC;gBACV,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,WAAW,EAAE,EAAE;gBACf,eAAe,EAAE,CAAC;gBAClB,OAAO,EAAE,IAAI,CAAC,OAAO;gBACrB,MAAM,EAAE,IAAI,CAAC,GAAG,EAAE;gBAClB,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;aACxB,CAAC,CAAA;YACF,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QAC3B,CAAC;KAAA;IAED,eAAe,CAAC,OAAoB;QAChC,IAAI,EAAE,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,aAAa,CAAC,qBAAqB,OAAO,CAAC,EAAE,IAAI,CAAC,CAAC;QAC1F,IAAI,CAAC,EAAE;YACH,OAAO;QACX,EAAE,CAAC,cAAc,CAAC,EAAC,KAAK,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAC,CAAC,CAAC;IAC1D,CAAC;;;YAvVJ,SAAS,SAAC;gBACP,QAAQ,EAAE,gBAAgB;gBAC1B,u9GAA8C;;aAEjD;;;YAVQ,YAAY;YADZ,kBAAkB;YAJQ,UAAU;YAMpC,cAAc;;;uBA+ClB,KAAK;2BAML,KAAK;qBAgCL,KAAK;0BASL,KAAK;0BACL,KAAK;iCACL,KAAK;+BACL,KAAK;yCAEL,KAAK;sBAEL,KAAK;0BA4DL,KAAK;wBACL,KAAK;yBACL,KAAK;2BACL,KAAK;oCACL,KAAK;+BACL,KAAK;6BACL,KAAK;6BAML,MAAM;iCAKN,MAAM;oCAKN,MAAM;sBAsBN,MAAM;uBAKN,MAAM;uBAKN,MAAM;2BAKN,MAAM;+BAKN,MAAM;6BAKN,MAAM;qBAKN,MAAM","sourcesContent":["import { AfterViewInit, Component, ElementRef, Input, Output } from '@angular/core';\r\nimport { User, ChatSource, ChatMessage, UserAccount, CommentsOrder } from '@banta/common';\r\nimport { HashTag } from '../comment-field/comment-field.component';\r\nimport { Subject, Observable, Subscription } from 'rxjs';\r\nimport { ChatBackendService } from '../../common';\r\nimport { BantaService } from '../../common';\r\nimport { ActivatedRoute } from '@angular/router';\r\n\r\n/**\r\n * Comments component\r\n */\r\n@Component({\r\n    selector: 'banta-comments',\r\n    templateUrl: './banta-comments.component.html',\r\n    styleUrls: ['./banta-comments.component.scss']\r\n})\r\nexport class BantaCommentsComponent implements AfterViewInit {\r\n    constructor(\r\n        private banta: BantaService,\r\n        private backend: ChatBackendService,\r\n        private elementRef: ElementRef<HTMLElement>,\r\n        private activatedRoute: ActivatedRoute\r\n    ) {\r\n    }\r\n\r\n    private _upvoted = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _shared = new Subject<ChatMessage>();\r\n\r\n    private _usernameSelected = new Subject<User>();\r\n    private _avatarSelected = new Subject<User>();\r\n\r\n    private _source: ChatSource;\r\n\r\n    private _subs = new Subscription();\r\n\r\n    _sortOrder: CommentsOrder = CommentsOrder.NEWEST;\r\n\r\n    get sortOrder() {\r\n        return this._sortOrder;\r\n    }\r\n\r\n    set sortOrder(value) {\r\n        if (this._sortOrder !== value) {\r\n            this._sortOrder = value;\r\n            setTimeout(() => {\r\n                this.setSourceFromTopicID(this.topicID);\r\n            });\r\n        }\r\n    }\r\n\r\n    @Input() hashtags: HashTag[] = [\r\n        {hashtag: 'error', description: 'Cause an error'},\r\n        {hashtag: 'timeout', description: 'Cause a slow timeout error'},\r\n        {hashtag: 'slow', description: 'Be slow when this message is posted'},\r\n    ];\r\n\r\n    @Input() participants: User[] = [];\r\n\r\n    ngOnInit() {\r\n        this._subs.add(\r\n            this.banta.userChanged.subscribe(user => this.user = user)\r\n        );\r\n    }\r\n\r\n    ngAfterViewInit() {\r\n        if (typeof window !== 'undefined') this.checkForSharedComment();\r\n    }\r\n\r\n    scrollToComment(commentId: ChatMessage['id']): void {\r\n        setTimeout(() => {\r\n          const comment = document.querySelectorAll(`[data-comment-id=\"${commentId}\"]`);\r\n          console.log(comment)\r\n          if (comment.length > 0) {\r\n            // comment.item(0).scroll({behavior: 'smooth'});\r\n            comment.item(0).scrollIntoView();\r\n          }\r\n        }, 1000);\r\n    }\r\n\r\n    checkForSharedComment(): void {\r\n        const commentID = this.activatedRoute.snapshot.queryParamMap.get('comment');\r\n        if (commentID) this.scrollToComment(commentID);\r\n    }\r\n\r\n    ngOnDestroy() {\r\n        this._subs.unsubscribe();\r\n    }\r\n\r\n    @Input()\r\n    get source(): ChatSource {\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() fixedHeight: boolean;\r\n    @Input() maxMessages: number;\r\n    @Input() maxVisibleMessages: number;\r\n    @Input() genericAvatarUrl: string;\r\n\r\n    @Input() shouldInterceptMessageSend?: (message: ChatMessage, source: ChatSource) => boolean | Promise<boolean>;\r\n\r\n    @Input()\r\n    get topicID(): string {\r\n        return this._topicID;\r\n    }\r\n\r\n    private _topicID: string;\r\n\r\n    set topicID(value) {\r\n        if (this._topicID !== value) {\r\n            this._topicID = value;\r\n            setTimeout(() => this.setSourceFromTopicID(value));\r\n        }\r\n    }\r\n\r\n    private async setSourceFromTopicID(topicID: string) {\r\n        this._source?.close?.();\r\n        this._source = null;\r\n        setTimeout(async () => {\r\n            this._source = await this.backend.getSourceForTopic(topicID, { sortOrder: this.sortOrder });\r\n\r\n            console.log(`[banta-comments] Subscribing to source for topic '${topicID}'`);\r\n\r\n            this._source.messageReceived.subscribe(m => this.addParticipant(m));\r\n            this._source.messageSent.subscribe(m => this.addParticipant(m));\r\n            this._source.messages.forEach(m => this.addParticipant(m));\r\n        });\r\n    }\r\n\r\n    private addParticipant(message: ChatMessage) {\r\n        if (!message || !message.user || !message.user.id)\r\n            return;\r\n\r\n        let existing = this.participants.find(x => x.id === message.user.id);\r\n        if (existing)\r\n            return;\r\n        this.participants.push(message.user);\r\n    }\r\n\r\n    showSignIn() {\r\n        this._signInSelected.next();\r\n    }\r\n\r\n    showEditAvatar() {\r\n        this._editAvatarSelected.next();\r\n    }\r\n\r\n    user: UserAccount;\r\n\r\n    private _newMessageText: string;\r\n\r\n    get newMessageText(): string {\r\n        return this._newMessageText;\r\n    }\r\n\r\n    set newMessageText(value) {\r\n        this._newMessageText = value;\r\n        if (this._newMessageText === '' && this.sendError)\r\n            setTimeout(() => this.sendError = null);\r\n    }\r\n\r\n    @Input() signInLabel = 'Sign In';\r\n    @Input() sendLabel = 'Send';\r\n    @Input() replyLabel = 'Reply';\r\n    @Input() sendingLabel = 'Sending';\r\n    @Input() permissionDeniedLabel = 'Send';\r\n    @Input() postCommentLabel = 'Post a comment';\r\n    @Input() postReplyLabel = 'Post a reply';\r\n\r\n    private _signInSelected = new Subject<void>();\r\n    private _permissionDeniedError = new Subject<void>();\r\n    private _editAvatarSelected = new Subject<void>();\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 editAvatarSelected() {\r\n        return this._editAvatarSelected;\r\n    }\r\n\r\n    @Output()\r\n    get permissionDeniedError(): Observable<void> {\r\n        return this._permissionDeniedError;\r\n    }\r\n\r\n    showPermissionDenied() {\r\n        this._permissionDeniedError.next();\r\n    }\r\n\r\n    get canComment() {\r\n        if (!this.user)\r\n            return false;\r\n\r\n        if (!this.user.permissions)\r\n            return true;\r\n\r\n        if (!this.user.permissions.canComment)\r\n            return true;\r\n\r\n        return this.user.permissions?.canComment(this.source);\r\n    }\r\n\r\n    @Output()\r\n    get upvoted() {\r\n        return this._upvoted.asObservable();\r\n    }\r\n\r\n    @Output()\r\n    get reported() {\r\n        return this._reported.asObservable();\r\n    }\r\n\r\n    @Output()\r\n    get selected() {\r\n        return this._selected.asObservable();\r\n    }\r\n\r\n    @Output()\r\n    get userSelected() {\r\n        return this._userSelected.asObservable();\r\n    }\r\n\r\n    @Output()\r\n    get usernameSelected() {\r\n        return this._usernameSelected.asObservable();\r\n    }\r\n\r\n    @Output()\r\n    get avatarSelected() {\r\n        return this._avatarSelected.asObservable();\r\n    }\r\n\r\n    @Output()\r\n    get shared() {\r\n        return this._shared.asObservable();\r\n    }\r\n\r\n    onKeyDown(event: KeyboardEvent) {\r\n    }\r\n\r\n    insertEmoji(text: string) {\r\n        this.newMessageText += text;\r\n    }\r\n\r\n    onReplyKeyDown(event: KeyboardEvent) {\r\n    }\r\n\r\n    insertReplyEmoji(text: string) {\r\n        this.replyMessage += text;\r\n    }\r\n\r\n    sending = false;\r\n    sendError: Error;\r\n    expandError = false;\r\n\r\n    indicateError(message: string) {\r\n        this.sendError = new Error(message);\r\n        setTimeout(() => {\r\n            this.expandError = true;\r\n            setTimeout(() => {\r\n                this.expandError = false;\r\n            }, 5 * 1000);\r\n        });\r\n    }\r\n\r\n    async upvoteMessage(message: ChatMessage) {\r\n        this._upvoted.next(message);\r\n        (message as any).$upvoting = true;\r\n        message.upvotes = (message.upvotes || 0) + 1;\r\n        await this.backend.upvoteMessage(message.topicId, message.parentMessageId ? message.parentMessageId : message.id, message.parentMessageId ? message.id : undefined);\r\n        await new Promise<void>(resolve => setTimeout(() => resolve(), 250));\r\n        (message as any).$upvoting = false;\r\n    }\r\n\r\n    reportMessage(message: ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    selectedMessage: ChatMessage;\r\n    selectedMessageThread: ChatSource;\r\n\r\n    replyMessage: string;\r\n\r\n    async unselectMessage() {\r\n        let message = this.selectedMessage;\r\n\r\n        this._selected.next(null);\r\n        this.selectedMessage = null;\r\n        if (this.selectedMessageThread) {\r\n            if (this.selectedMessageThread.close)\r\n                this.selectedMessageThread.close();\r\n            this.selectedMessageThread = null;\r\n        }\r\n\r\n        if (message)\r\n            setTimeout(() => this.scrollToMessage(message));\r\n    }\r\n\r\n    selectedMessageVisible = false;\r\n\r\n    async selectMessage(message: ChatMessage) {\r\n        this._selected.next(message);\r\n        this.selectedMessage = message;\r\n        setTimeout(() => this.selectedMessageVisible = true);\r\n        setTimeout(async () => {\r\n            this.selectedMessageThread = await this.backend.getSourceForThread(this.topicID, message.id);\r\n        }, 250);\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    shareMessage(message: ChatMessage) {\r\n        this._shared.next(message);\r\n    }\r\n\r\n    async sendReply() {\r\n        await this.selectedMessageThread.send({\r\n            message: this.replyMessage,\r\n            parentMessageId: this.selectedMessage.id,\r\n            upvotes: 0,\r\n            user: this.user,\r\n            submessages: [],\r\n            submessageCount: 0,\r\n            topicId: this.topicID,\r\n            sentAt: Date.now(),\r\n            updatedAt: Date.now()\r\n        })\r\n        this.replyMessage = '';\r\n    }\r\n\r\n    scrollToMessage(message: ChatMessage) {\r\n        let el = this.elementRef.nativeElement.querySelector(`[data-comment-id=\"${message.id}\"]`);\r\n        if (!el)\r\n            return;\r\n        el.scrollIntoView({block: 'center', inline: 'start'});\r\n    }\r\n}\r\n"]}
|
|
478
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"banta-comments.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/comments/banta-comments/banta-comments.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, Component, ElementRef, Input, Output } from '@angular/core';\r\nimport { User, ChatMessage, CommentsOrder } from '@banta/common';\r\nimport { HashTag } from '../comment-field/comment-field.component';\r\nimport { Subject, Observable, Subscription } from 'rxjs';\r\nimport { ActivatedRoute } from '@angular/router';\r\nimport { ChatBackendBase } from '../../chat-backend-base';\r\nimport { ChatSourceBase } from '../../chat-source-base';\r\nimport { EditEvent } from '../comment-view/comment-view.component';\r\n\r\n/**\r\n * Comments component\r\n */\r\n@Component({\r\n    selector: 'banta-comments',\r\n    templateUrl: './banta-comments.component.html',\r\n    styleUrls: ['./banta-comments.component.scss']\r\n})\r\nexport class BantaCommentsComponent {\r\n    constructor(\r\n        private backend: ChatBackendBase,\r\n        private elementRef: ElementRef<HTMLElement>,\r\n        private activatedRoute: ActivatedRoute\r\n    ) {\r\n        this.sendMessage = async (message: ChatMessage) => {\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\r\n                if (this.source.sortOrder !== CommentsOrder.NEWEST) {\r\n                    this.sortOrder = CommentsOrder.NEWEST;\r\n                }\r\n                return true;\r\n            } catch (e) {\r\n                this.handleBackendException(e, 'Could not send: ');\r\n            }\r\n        }\r\n\r\n        this.sendReply = async (message: ChatMessage) => {\r\n            try {\r\n                const intercept = await this.shouldInterceptMessageSend?.(message, this.source);\r\n                if (!intercept) {\r\n                    await this.selectedMessageThread.send(message);\r\n                }\r\n                return true;\r\n            } catch (e) {\r\n                this.handleBackendException(e, 'Could not send reply: ');\r\n            }\r\n        }\r\n    }\r\n\r\n    private handleBackendExceptionAsAlert(e: Error, prefix: string = '') {\r\n        try {\r\n            this.handleBackendException(e, prefix);\r\n        } catch (e) {\r\n            alert(e.message);\r\n        }\r\n    }\r\n\r\n    private handleBackendException(e: Error, prefix: string = '') {\r\n        let errorMessage = e.message;\r\n\r\n        if (errorMessage.startsWith('permission-denied|')) {\r\n            errorMessage = errorMessage.replace(/^permission-denied\\|/, '');\r\n\r\n            if (errorMessage.startsWith(`app-handle|`)) {\r\n                // If this is an error during authorizeAction on the backend, pass control to the user-provided \r\n                // permission-denied handler.\r\n\r\n                this.sendPermissionDenied(errorMessage.replace(/^app-handle\\|/, ''));\r\n                return;\r\n            }\r\n        }\r\n\r\n        throw new Error(`${prefix}${errorMessage}`);\r\n    }\r\n\r\n    // Lifecycle Events / Initialization\r\n\r\n    ngOnInit() {\r\n        this._subs.add(this.backend.userChanged.subscribe(user => this.user = user));\r\n        this.startLoading();\r\n\r\n        if (typeof window !== 'undefined') {\r\n            let queryString = window.location.search.substring(1);\r\n            let query = queryString.split('&')\r\n                .map(s => s.split('='))\r\n                .reduce((o, [k, v]) => (o[k] = v, o), {})\r\n            ;\r\n\r\n            const commentID = query['comment'];\r\n            if (commentID) {\r\n                this.sharedCommentID = commentID;\r\n            }\r\n        }\r\n    }\r\n\r\n    sharedCommentID: string;\r\n\r\n    ngOnDestroy() {\r\n        this._subs.unsubscribe();\r\n    }\r\n\r\n    private async setSourceFromTopicID(topicID: string) {\r\n        if (this._source) {\r\n            this._source.close();\r\n            this._source = null;\r\n        }\r\n\r\n        setTimeout(async () => {\r\n            console.log(`[banta-comments] Subscribing to topic source '${topicID}'`);\r\n            this._source = await this.backend.getSourceForTopic(topicID, { sortOrder: this.sortOrder });\r\n\r\n            if (this.sharedCommentID) {\r\n                this.navigateToSharedComment(this.sharedCommentID);\r\n                this.sharedCommentID = null;\r\n            }\r\n\r\n            this._source.messageReceived.subscribe(m => this.addParticipant(m));\r\n            this._source.messageSent.subscribe(m => this.addParticipant(m));\r\n            this._source.messages.forEach(m => this.addParticipant(m));\r\n        });\r\n    }\r\n\r\n    // Loading Screen\r\n    private _loadingMessage = '';\r\n    loadingMessageVisible = false;\r\n\r\n    get loadingMessage() {\r\n        return this._loadingMessage;\r\n    }\r\n\r\n    set loadingMessage(value) {\r\n        this.loadingMessageVisible = false;\r\n        setTimeout(() => {\r\n            this._loadingMessage = value;\r\n            this._loadingMessage = value;\r\n            setTimeout(() => {\r\n                this.loadingMessageVisible = true;\r\n            })\r\n        }, 500);\r\n    }\r\n\r\n    loading: boolean = true;\r\n    showLoadingScreen = false;\r\n    loadingStartedAt: number;\r\n    messageChangedAt: number;\r\n\r\n    private async startLoading() {\r\n        this.loadingStartedAt = this.messageChangedAt = Date.now();\r\n\r\n        if (this.updateLoading()) return;\r\n        await new Promise<void>(resolve => setTimeout(() => resolve(), 100));\r\n        if (this.updateLoading()) return;\r\n        await new Promise<void>(resolve => setTimeout(() => resolve(), 250));\r\n        if (this.updateLoading()) return;\r\n        await new Promise<void>(resolve => setTimeout(() => resolve(), 500));\r\n        if (this.updateLoading()) return;\r\n\r\n        console.log(`[Banta] Loading is taking a long time! Showing loading screen.`);\r\n        this.showLoadingScreen = true;\r\n        if (typeof window !== 'undefined')\r\n            this._loadingTimer = setInterval(() => this.updateLoading(), 1000);\r\n    }\r\n    \r\n    private _loadingTimer;\r\n    private _loadingMessageIndex = 0;\r\n\r\n    @Input() loadingMessages: string[] = [\r\n        `Just a second...`,\r\n        `We're definitely working on it.`,\r\n        `There's no need to refresh.`,\r\n        `It's definitely worth the wait!`,\r\n        `This has never happened before.`,\r\n        `We'll keep trying, but it's not looking great. \r\n            Commenting & chat services may be down. \r\n            If you continue to experience issues, please contact support.\r\n        `\r\n    ];\r\n\r\n    private updateLoading(): boolean {\r\n        if (this.source?.state && this.source?.state !== 'connecting') {\r\n            clearInterval(this._loadingTimer);\r\n            this.loadingMessage = `Here we go!`;\r\n            setTimeout(() => {\r\n                this.loading = false;\r\n            }, 750);\r\n            return true;\r\n        }\r\n\r\n        let messageSwitchTime = 5*1000;\r\n        if (this.messageChangedAt + messageSwitchTime < Date.now()) {\r\n            if (this.loadingMessages[this._loadingMessageIndex]) {\r\n                this.loadingMessage = this.loadingMessages[this._loadingMessageIndex++];\r\n                this.messageChangedAt = Date.now();\r\n            }\r\n        }\r\n\r\n        return false;\r\n    }\r\n\r\n    // Properties\r\n\r\n    private _signInSelected = new Subject<void>();\r\n    private _permissionDeniedError = new Subject<string>();\r\n    private _editAvatarSelected = new Subject<void>();\r\n    private _upvoted = new Subject<ChatMessage>();\r\n    private _reported = new Subject<ChatMessage>();\r\n    private _selected = new Subject<ChatMessage>();\r\n    private _userSelected = new Subject<ChatMessage>();\r\n    private _shared = new Subject<ChatMessage>();\r\n    private _usernameSelected = new Subject<User>();\r\n    private _avatarSelected = new Subject<User>();\r\n    private _source: ChatSourceBase;\r\n    private _subs = new Subscription();\r\n    private _sortOrder: CommentsOrder = CommentsOrder.NEWEST;\r\n    private _topicID: string;\r\n    \r\n    user: User;\r\n    selectedMessage: ChatMessage;\r\n    selectedMessageThread: ChatSourceBase;\r\n    selectedMessageVisible = false;\r\n\r\n    // Inputs\r\n\r\n    @Input() signInLabel = 'Sign In';\r\n    @Input() sendLabel = 'Send';\r\n    @Input() replyLabel = 'Reply';\r\n    @Input() sendingLabel = 'Sending';\r\n    @Input() permissionDeniedLabel = 'Send';\r\n    @Input() postCommentLabel = 'Post a comment';\r\n    @Input() postReplyLabel = 'Post a reply';\r\n    @Input() fixedHeight: boolean;\r\n    @Input() maxMessages: number;\r\n    @Input() maxVisibleMessages: number;\r\n    @Input() genericAvatarUrl: string;\r\n    @Input() shouldInterceptMessageSend?: (message: ChatMessage, source: ChatSourceBase) => boolean | Promise<boolean>;\r\n    @Input() participants: User[] = [];\r\n\r\n    @Input()\r\n    get source(): ChatSourceBase { return this._source; }\r\n    set source(value) { \r\n        this._source = value;\r\n        if (value && this.sharedCommentID) {\r\n            this.navigateToSharedComment(this.sharedCommentID);\r\n            this.sharedCommentID = null;\r\n        }\r\n    }\r\n    @Input() \r\n    hashtags: HashTag[] = [\r\n        {hashtag: 'error', description: 'Cause an error'},\r\n        {hashtag: 'timeout', description: 'Cause a slow timeout error'},\r\n        {hashtag: 'slow', description: 'Be slow when this message is posted'},\r\n    ];\r\n    @Input()\r\n    get topicID(): string { return this._topicID; }\r\n    set topicID(value) {\r\n        if (this._topicID !== value) {\r\n            this._topicID = value;\r\n            setTimeout(() => this.setSourceFromTopicID(value));\r\n        }\r\n    }\r\n\r\n    // Outputs\r\n\r\n    @Output() get signInSelected(): Observable<void> { return this._signInSelected; }\r\n    @Output() get editAvatarSelected() { return this._editAvatarSelected; }\r\n    @Output() get permissionDeniedError(): Observable<string> { return this._permissionDeniedError; }\r\n    @Output() get upvoted() { return this._upvoted.asObservable(); }\r\n    @Output() get reported() { return this._reported.asObservable(); }\r\n    @Output() get selected() { return this._selected.asObservable(); }\r\n    @Output() get userSelected() { return this._userSelected.asObservable(); }\r\n    @Output() get usernameSelected() { return this._usernameSelected.asObservable(); }\r\n    @Output() get avatarSelected() { return this._avatarSelected.asObservable(); }\r\n    @Output() get shared() { return this._shared.asObservable(); }\r\n\r\n    get sortOrder() { return this._sortOrder; }\r\n    set sortOrder(value) {\r\n        if (this._sortOrder !== value) {\r\n            this._sortOrder = value;\r\n            setTimeout(() => {\r\n                this.setSourceFromTopicID(this.topicID);\r\n            });\r\n        }\r\n    }\r\n\r\n    sendMessage: (message: ChatMessage) => void;\r\n    sendReply: (message: ChatMessage) => void;\r\n\r\n    // UI Interactions\r\n\r\n    scrollToComment(commentId: ChatMessage['id']): void {\r\n        setTimeout(() => {\r\n          const comment = document.querySelectorAll(`[data-comment-id=\"${commentId}\"]`);\r\n          if (comment.length > 0) {\r\n            // comment.item(0).scroll({behavior: 'smooth'});\r\n            comment.item(0).scrollIntoView();\r\n          }\r\n        }, 1000);\r\n    }\r\n\r\n    async navigateToSharedComment(id: string) {\r\n        let source = this.source;\r\n\r\n        await source.ready;\r\n\r\n        console.log(`Navigating to shared comment with ID '${id}'...`);\r\n        let message: ChatMessage;\r\n        \r\n        try {\r\n            message = await this.source.get(id);\r\n        } catch (e) {\r\n            console.error(`Failed to find comment from URL: ${e.message}`);\r\n            return;\r\n        }\r\n\r\n        message.transientState ??= {};\r\n\r\n        // If there is a parent message, we should instead focus that and let the \r\n        // scrollToComment and highlight do the work.\r\n\r\n        if (message.parentMessageId) {\r\n\r\n            let parentMessage = await this.source.get(message.parentMessageId);\r\n            parentMessage.transientState ??= {};\r\n            let thread = await this.selectMessage(parentMessage); \r\n\r\n            // Need to re-retrieve the message within the new chat source to affect its\r\n            // transient state.\r\n            await thread.ready;\r\n            message = await thread.get(message.id);\r\n            message.transientState ??= {};\r\n            message.transientState.highlighted = true;\r\n            console.dir(message);\r\n            await new Promise<void>(resolve => setTimeout(() => resolve(), 500));\r\n\r\n        } else {\r\n            this.selectMessage(message);\r\n        }\r\n\r\n        this.scrollToComment(id);\r\n    }\r\n\r\n    sendPermissionDenied(message: string) {\r\n        this._permissionDeniedError.next(message);\r\n    }\r\n\r\n    scrollToMessage(message: ChatMessage) {\r\n        let el = this.elementRef.nativeElement.querySelector(`[data-comment-id=\"${message.id}\"]`);\r\n        if (!el)\r\n            return;\r\n        el.scrollIntoView({block: 'center', inline: 'start'});\r\n    }\r\n\r\n    private addParticipant(message: ChatMessage) {\r\n        if (!message || !message.user || !message.user.id)\r\n            return;\r\n\r\n        let existing = this.participants.find(x => x.id === message.user.id);\r\n        if (existing)\r\n            return;\r\n        this.participants.push(message.user);\r\n    }\r\n\r\n    // Actions\r\n\r\n    async likeMessage(source: ChatSourceBase, message: ChatMessage) {\r\n        this._upvoted.next(message);\r\n        message.transientState.liking = true;\r\n\r\n        if (!message.userState?.liked)\r\n            message.likes = (message.likes || 0) + 1;\r\n        \r\n        try {\r\n            await source.likeMessage(message.id);\r\n        } catch (e) {\r\n            this.handleBackendExceptionAsAlert(e, 'Could not like this message: ');\r\n        }\r\n        \r\n        await new Promise<void>(resolve => setTimeout(() => resolve(), 250));\r\n        message.transientState.liking = false;\r\n    }\r\n\r\n    async unlikeMessage(source: ChatSourceBase, message: ChatMessage) {\r\n        this._upvoted.next(message);\r\n        message.transientState.liking = true;\r\n\r\n        if (message.userState?.liked)\r\n            message.likes = (message.likes || 0) - 1;\r\n\r\n        try {\r\n            await source.unlikeMessage(message.id);\r\n        } catch (e) {\r\n            this.handleBackendExceptionAsAlert(e, 'Failed to unlike this message: ');\r\n        }\r\n\r\n        await new Promise<void>(resolve => setTimeout(() => resolve(), 250));\r\n        message.transientState.liking = false;\r\n    }\r\n\r\n    async reportMessage(message: ChatMessage) {\r\n        this._reported.next(message);\r\n    }\r\n\r\n    async unselectMessage() {\r\n        let message = this.selectedMessage;\r\n\r\n        this._selected.next(null);\r\n        this.selectedMessage = null;\r\n        if (this.selectedMessageThread) {\r\n            if (this.selectedMessageThread.close)\r\n                this.selectedMessageThread.close();\r\n            this.selectedMessageThread = null;\r\n        }\r\n\r\n        if (message)\r\n            setTimeout(() => this.scrollToMessage(message));\r\n    }\r\n\r\n    async selectMessage(message: ChatMessage) {\r\n        this._selected.next(message);\r\n        this.selectedMessage = message;\r\n        let selectedMessageThread = await this.backend.getSourceForThread(this.topicID, message.id);\r\n\r\n        setTimeout(() => this.selectedMessageVisible = true);\r\n        setTimeout(async () => {\r\n            this.selectedMessageThread = selectedMessageThread;\r\n        }, 250);\r\n\r\n        return selectedMessageThread;\r\n    }\r\n\r\n    async showSignIn() {\r\n        this._signInSelected.next();\r\n    }\r\n\r\n    async showEditAvatar() {\r\n        this._editAvatarSelected.next();\r\n    }\r\n\r\n    async selectMessageUser(message: ChatMessage) {\r\n        this._userSelected.next(message);\r\n    }\r\n\r\n    async selectUsername(user: User) {\r\n        this._usernameSelected.next(user);\r\n    }\r\n\r\n    async selectAvatar(user: User) {\r\n        this._avatarSelected.next(user);\r\n    }\r\n\r\n    async shareMessage(message: ChatMessage) {\r\n        this._shared.next(message);\r\n    }\r\n\r\n    async deleteMessage(message: ChatMessage) {\r\n        if (!confirm(\"Are you sure you want to delete this comment? You cannot undo this action.\"))\r\n            return;\r\n\r\n        try {\r\n            await this.source.deleteMessage(message.id);\r\n        } catch (e) {\r\n            this.handleBackendExceptionAsAlert(e, `Could not delete message: `);\r\n        }\r\n    }\r\n\r\n    async editMessage(source: ChatSourceBase, message: ChatMessage, newText: string) {\r\n        try {\r\n            await source.editMessage(message.id, newText);\r\n        } catch (e) {\r\n            this.handleBackendExceptionAsAlert(e, 'Could not edit this message: ');\r\n            return;\r\n        }\r\n\r\n        message.message = newText;\r\n        message.transientState.editing = false;\r\n    }\r\n\r\n    async startEditing(message: ChatMessage) {\r\n        this.selectedMessage.transientState.editing = false;\r\n        message.transientState.editing = true;\r\n    }\r\n\r\n    async saveEdit(message: ChatMessage, text: string) {\r\n        try {\r\n            await this.source.editMessage(message.id, text);\r\n            message.transientState.editing = false;\r\n        } catch (e) {\r\n            this.handleBackendExceptionAsAlert(e, `Could not edit message: `);\r\n        }\r\n    }\r\n}\r\n"]}
|