@banta/sdk 4.7.12 → 4.7.13
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { Component, Input } from "@angular/core";
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
|
-
import * as i1 from "
|
|
3
|
+
import * as i1 from "../../lib/common/timer-pool.service";
|
|
4
4
|
import * as i2 from "@angular/common";
|
|
5
5
|
export class TimestampComponent {
|
|
6
6
|
constructor(timerPool) {
|
|
@@ -120,4 +120,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
120
120
|
}], ctorParameters: function () { return [{ type: i1.TimerPool }]; }, propDecorators: { value: [{
|
|
121
121
|
type: Input
|
|
122
122
|
}] } });
|
|
123
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
123
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timestamp.component.js","sourceRoot":"","sources":["../../../../../projects/sdk/src/lib/common/timestamp.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;;;;AAejD,MAAM,OAAO,kBAAkB;IAC3B,YACY,SAAoB;QAApB,cAAS,GAAT,SAAS,CAAW;QAKhC,aAAQ,GAAG,EAAE,CAAC;QACd,YAAO,GAAG,EAAE,CAAC;QAGL,kBAAa,GAAW,CAAC,CAAC;QAE1B,eAAU,GAAG,KAAK,CAAC;QAY3B,iBAAY,GAAG,KAAK,CAAC;IArBrB,CAAC;IAWD,WAAW;QACP,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,IAAI,CAAC,gBAAgB,EAAE,EAAE,CAAC;IAC9B,CAAC;IAED,IACI,KAAK;QACL,OAAO,IAAI,CAAC,MAAM,CAAC;IACvB,CAAC;IAID,MAAM;QACF,IAAI,IAAI,CAAC,UAAU;YACf,OAAO;QAEX,IAAI,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACrB,IAAI,IAAI,GAAG,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC;QAC5B,IAAI,MAAM,GAAG,IAAI,GAAC,EAAE,CAAC;QACrB,IAAI,IAAI,GAAG,MAAM,GAAG,EAAE,CAAC;QACvB,IAAI,GAAG,GAAG,IAAI,GAAG,EAAE,CAAC;QACpB,IAAI,IAAI,GAAG,GAAG,GAAG,CAAC,CAAC;QACnB,IAAI,KAAK,GAAG,GAAG,GAAG,EAAE,CAAC;QACrB,IAAI,IAAI,GAAG,GAAG,GAAG,GAAG,CAAC;QACrB,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,UAAU,GAAG,CAAC,CAAC;QAEnB,IAAI,IAAI,GAAG,IAAI,EAAE;YACb,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;YACzB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,OAAO;SACV;QAED,IAAI,IAAI,GAAG,KAAK,EAAE;YACd,IAAI,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,KAAK,CAAC,CAAC;YAEtC,IAAI,MAAM,KAAK,CAAC;gBACZ,IAAI,CAAC,QAAQ,GAAG,GAAG,MAAM,YAAY,CAAC;;gBAEtC,IAAI,CAAC,QAAQ,GAAG,GAAG,MAAM,aAAa,CAAC;SAE9C;aAAM,IAAI,IAAI,GAAG,IAAI,EAAE;YACpB,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;YAEpC,IAAI,KAAK,KAAK,CAAC;gBACX,IAAI,CAAC,QAAQ,GAAG,GAAG,KAAK,WAAW,CAAC;;gBAEpC,IAAI,CAAC,QAAQ,GAAG,GAAG,KAAK,YAAY,CAAC;SAC5C;aAAM,IAAI,IAAI,GAAG,GAAG,EAAE;YACnB,IAAI,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,GAAG,CAAC,CAAC;YAClC,IAAI,IAAI,KAAK,CAAC;gBACV,IAAI,CAAC,QAAQ,GAAG,GAAG,IAAI,UAAU,CAAC;;gBAElC,IAAI,CAAC,QAAQ,GAAG,GAAG,IAAI,WAAW,CAAC;SAC1C;aAAM,IAAI,IAAI,GAAG,IAAI,EAAE;YACpB,IAAI,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;YACpC,IAAI,KAAK,KAAK,CAAC;gBACX,IAAI,CAAC,QAAQ,GAAG,GAAG,KAAK,WAAW,CAAC;;gBAEpC,IAAI,CAAC,QAAQ,GAAG,GAAG,KAAK,YAAY,CAAC;YAEzC,UAAU,GAAG,IAAI,GAAG,EAAE,GAAG,EAAE,CAAC;SAC/B;aAAM,IAAI,IAAI,GAAG,MAAM,EAAE;YACtB,IAAI,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,GAAG,MAAM,CAAC,CAAC;YACxC,IAAI,OAAO,KAAK,CAAC;gBACb,IAAI,CAAC,QAAQ,GAAG,GAAG,OAAO,aAAa,CAAC;;gBAExC,IAAI,CAAC,QAAQ,GAAG,GAAG,OAAO,cAAc,CAAC;YAC7C,UAAU,GAAG,IAAI,GAAG,EAAE,CAAC;SAC1B;aAAM,IAAI,IAAI,GAAG,KAAM,EAAE;YACtB,IAAI,CAAC,QAAQ,GAAG,oBAAoB,CAAC;YACrC,UAAU,GAAG,IAAI,GAAG,EAAE,CAAC;SAC1B;aAAM;YACH,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;YAC3B,UAAU,GAAG,IAAI,GAAG,EAAE,CAAC;SAC1B;QAED,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE;YAC/B,IAAI,IAAI,CAAC,aAAa,KAAK,UAAU,EAAE;gBACnC,IAAI,CAAC,aAAa,GAAG,UAAU,CAAC;gBAChC,IAAI,CAAC,gBAAgB,EAAE,EAAE,CAAC;gBAC1B,IAAI,UAAU,GAAG,CAAC,EAAE;oBAChB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,UAAU,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;iBACpF;aACJ;SACJ;IACL,CAAC;IAED,IAAI,KAAK,CAAC,CAAC;QACP,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE;YACnB,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;YAChB,IAAI,CAAC,MAAM,EAAE,CAAC;SACjB;IACL,CAAC;;gHA5GQ,kBAAkB;oGAAlB,kBAAkB,mFAVjB;;;;;;;KAOT;4FAGQ,kBAAkB;kBAZ9B,SAAS;+BACI,iBAAiB,YACjB;;;;;;;KAOT;gGAwBG,KAAK;sBADR,KAAK","sourcesContent":["import { Component, Input } from \"@angular/core\";\r\nimport { TimerPool } from \"../../lib/common/timer-pool.service\";\r\n\r\n@Component({\r\n    selector: 'banta-timestamp',\r\n    template: `\r\n        <span *ngIf=\"showAbsolute\" [title]=\"value | date : 'short'\">\r\n            {{value | date : 'shortDate'}}\r\n        </span>\r\n        <span *ngIf=\"!showAbsolute\" [title]=\"value | date : 'short'\">\r\n            {{relative}}\r\n        </span>\r\n    `,\r\n    styles: [``]\r\n})\r\nexport class TimestampComponent {\r\n    constructor(\r\n        private timerPool: TimerPool\r\n    ) {\r\n    }\r\n\r\n    private _value : number;\r\n    relative = '';\r\n    tooltip = '';\r\n\r\n    private timerUnsubscribe: () => void;\r\n    private timerInterval: number = 0;\r\n\r\n    private _destroyed = false;\r\n\r\n    ngOnDestroy() {\r\n        this._destroyed = true;\r\n        this.timerUnsubscribe?.();\r\n    }\r\n\r\n    @Input()\r\n    get value() {\r\n        return this._value;\r\n    }\r\n\r\n    showAbsolute = false;\r\n\r\n    update() {\r\n        if (this._destroyed)\r\n            return;\r\n        \r\n        let now = Date.now();\r\n        let diff = now - this.value;\r\n        let minute = 1000*60;\r\n        let hour = minute * 60;\r\n        let day = hour * 24;\r\n        let week = day * 7;\r\n        let month = day * 30;\r\n        let year = day * 365;\r\n        this.showAbsolute = false;\r\n        let updateTime = 0;\r\n\r\n        if (diff > year) {\r\n            this.showAbsolute = true;\r\n            this.relative = 'abs';\r\n            return;\r\n        }\r\n\r\n        if (diff > month) {\r\n            let months = Math.floor(diff / month);\r\n\r\n            if (months === 1)\r\n                this.relative = `${months} month ago`;\r\n            else\r\n                this.relative = `${months} months ago`;\r\n            \r\n        } else if (diff > week) {\r\n            let weeks = Math.floor(diff / week);\r\n\r\n            if (weeks === 1)\r\n                this.relative = `${weeks} week ago`;\r\n            else\r\n                this.relative = `${weeks} weeks ago`;\r\n        } else if (diff > day) {\r\n            let days = Math.floor(diff / day);\r\n            if (days === 1)\r\n                this.relative = `${days} day ago`;\r\n            else\r\n                this.relative = `${days} days ago`;\r\n        } else if (diff > hour) {\r\n            let hours = Math.floor(diff / hour);\r\n            if (hours === 1)\r\n                this.relative = `${hours} hour ago`;\r\n            else\r\n                this.relative = `${hours} hours ago`;\r\n            \r\n            updateTime = 1000 * 60 * 30;\r\n        } else if (diff > minute) {\r\n            let minutes = Math.floor(diff / minute);\r\n            if (minutes === 1)\r\n                this.relative = `${minutes} minute ago`;\r\n            else\r\n                this.relative = `${minutes} minutes ago`;\r\n            updateTime = 1000 * 45;\r\n        } else if (diff > 30_000) {\r\n            this.relative = `about a minute ago`;\r\n            updateTime = 1000 * 60;\r\n        } else {\r\n            this.relative = `just now`;\r\n            updateTime = 1000 * 30;\r\n        }\r\n        \r\n        if (typeof window !== 'undefined') {\r\n            if (this.timerInterval !== updateTime) {\r\n                this.timerInterval = updateTime;\r\n                this.timerUnsubscribe?.();\r\n                if (updateTime > 0) {\r\n                    this.timerUnsubscribe = this.timerPool.addTimer(updateTime, () => this.update());\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    set value(v) {\r\n        if (this._value !== v) {\r\n            this._value = v;\r\n            this.update();\r\n        }\r\n    }\r\n}"]}
|
package/fesm2015/banta-sdk.mjs
CHANGED
|
@@ -2,8 +2,9 @@ import { Observable, Subject, BehaviorSubject, Subscription } from 'rxjs';
|
|
|
2
2
|
import { publish, take } from 'rxjs/operators';
|
|
3
3
|
import * as i0 from '@angular/core';
|
|
4
4
|
import { Component, Input, ViewChild, Pipe, Inject, Optional, Output, HostBinding, NgModule, ViewChildren, Directive, TemplateRef, ContentChild, Injectable as Injectable$1 } from '@angular/core';
|
|
5
|
-
import
|
|
6
|
-
import
|
|
5
|
+
import { __decorate, __awaiter } from 'tslib';
|
|
6
|
+
import * as i1$2 from '@banta/common';
|
|
7
|
+
import { Injectable, CommentsOrder, SocketRPC, RpcEvent, DurableSocket } from '@banta/common';
|
|
7
8
|
import * as i2 from '@angular/common';
|
|
8
9
|
import { CommonModule } from '@angular/common';
|
|
9
10
|
import * as i2$1 from '@angular/material/icon';
|
|
@@ -11,22 +12,20 @@ import { MatIconModule } from '@angular/material/icon';
|
|
|
11
12
|
import * as marked from 'marked';
|
|
12
13
|
import createDOMPurify from 'dompurify';
|
|
13
14
|
import twemoji$1 from 'twemoji';
|
|
14
|
-
import * as i1
|
|
15
|
+
import * as i1 from '@angular/platform-browser';
|
|
15
16
|
import * as i3 from '@angular/cdk/bidi';
|
|
16
17
|
import * as i4 from '@angular/material/progress-spinner';
|
|
17
18
|
import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
|
|
18
19
|
import * as i6 from '@angular/material/button';
|
|
19
20
|
import { MatButtonModule } from '@angular/material/button';
|
|
20
|
-
import {
|
|
21
|
-
import * as i1$3 from '@banta/common';
|
|
22
|
-
import { Injectable, CommentsOrder, SocketRPC, RpcEvent, DurableSocket } from '@banta/common';
|
|
21
|
+
import { TimerPool as TimerPool$1 } from 'projects/sdk/src/lib/common/timer-pool.service';
|
|
23
22
|
import * as i4$1 from '@angular/forms';
|
|
24
23
|
import { FormsModule } from '@angular/forms';
|
|
25
24
|
import * as i6$1 from '@angular/material/form-field';
|
|
26
25
|
import { MatFormFieldModule } from '@angular/material/form-field';
|
|
27
26
|
import * as i7 from '@angular/material/input';
|
|
28
27
|
import { MatInputModule } from '@angular/material/input';
|
|
29
|
-
import * as i1$
|
|
28
|
+
import * as i1$1 from '@angular/cdk/overlay';
|
|
30
29
|
import { OverlayModule } from '@angular/cdk/overlay';
|
|
31
30
|
import * as i4$2 from '@angular/cdk/portal';
|
|
32
31
|
import { PortalModule } from '@angular/cdk/portal';
|
|
@@ -58,6 +57,89 @@ function lazyConnection(options) {
|
|
|
58
57
|
return obs.pipe(publish()).refCount();
|
|
59
58
|
}
|
|
60
59
|
|
|
60
|
+
/**
|
|
61
|
+
* Provides a way to hook in to a shared set of timers, instead of creating a timer per instance.
|
|
62
|
+
* This is very useful for cases where the update is not extremely time-sensitive, but happens at scale.
|
|
63
|
+
* The principal use case is the TimestampComponent. When several hundred (or several thousand) comments are
|
|
64
|
+
* being displayed, we do not want to trigger thousands of independent relative timestamp updates, because over
|
|
65
|
+
* time the updates will saturate the CPU since they don't perfectly align.
|
|
66
|
+
*/
|
|
67
|
+
let TimerPool = class TimerPool {
|
|
68
|
+
constructor() {
|
|
69
|
+
this.subscriptions = new Map();
|
|
70
|
+
this.newSubscriptions = new Map();
|
|
71
|
+
this.removedSubscriptions = new Map();
|
|
72
|
+
}
|
|
73
|
+
addTimer(interval, callback) {
|
|
74
|
+
var _a;
|
|
75
|
+
if (interval <= 0) {
|
|
76
|
+
console.warn(`Refusing to set timer with interval of ${interval}!`);
|
|
77
|
+
return () => { };
|
|
78
|
+
}
|
|
79
|
+
let state;
|
|
80
|
+
let sizeWas = this.subscriptions.size;
|
|
81
|
+
if (!this.subscriptions.has(interval)) {
|
|
82
|
+
state = { subscribers: [] };
|
|
83
|
+
state.handle = setInterval(() => {
|
|
84
|
+
console.debug(`[Banta/TimerPool] Notifying ${state.subscribers.length} subs [${interval}ms]`);
|
|
85
|
+
state.subscribers.forEach(sub => sub());
|
|
86
|
+
}, interval);
|
|
87
|
+
this.subscriptions.set(interval, state);
|
|
88
|
+
}
|
|
89
|
+
else {
|
|
90
|
+
state = this.subscriptions.get(interval);
|
|
91
|
+
}
|
|
92
|
+
state.subscribers.push(callback);
|
|
93
|
+
// Debug information //////////////////////////
|
|
94
|
+
//
|
|
95
|
+
if (!this.newSubscriptions.has(interval))
|
|
96
|
+
this.newSubscriptions.set(interval, 0);
|
|
97
|
+
this.newSubscriptions.set(interval, ((_a = this.newSubscriptions.get(interval)) !== null && _a !== void 0 ? _a : 0) + 1);
|
|
98
|
+
clearTimeout(this.newSubscriptionsNotice);
|
|
99
|
+
this.newSubscriptionsNotice = setTimeout(() => {
|
|
100
|
+
for (let [interval, count] of this.newSubscriptions) {
|
|
101
|
+
console.debug(`[Banta/TimerPool] ${count} new subscriptions to ${interval}ms [${state.subscribers.length} total]`);
|
|
102
|
+
}
|
|
103
|
+
this.newSubscriptions.clear();
|
|
104
|
+
});
|
|
105
|
+
//
|
|
106
|
+
///////////////////////////////////////////////
|
|
107
|
+
if (sizeWas === 0) {
|
|
108
|
+
console.debug(`[Banta/TimerPool] No longer idle.`);
|
|
109
|
+
}
|
|
110
|
+
// Unsubscribe function
|
|
111
|
+
return () => {
|
|
112
|
+
var _a;
|
|
113
|
+
let state = this.subscriptions.get(interval);
|
|
114
|
+
let index = state.subscribers.indexOf(callback);
|
|
115
|
+
if (index >= 0)
|
|
116
|
+
state.subscribers.splice(index, 1);
|
|
117
|
+
if (state.subscribers.length === 0) {
|
|
118
|
+
clearInterval(state.handle);
|
|
119
|
+
this.subscriptions.delete(interval);
|
|
120
|
+
}
|
|
121
|
+
if (!this.removedSubscriptions.has(interval))
|
|
122
|
+
this.removedSubscriptions.set(interval, 0);
|
|
123
|
+
this.removedSubscriptions.set(interval, ((_a = this.removedSubscriptions.get(interval)) !== null && _a !== void 0 ? _a : 0) + 1);
|
|
124
|
+
// Debug information ////////////////////////////////////////////////////////////////////
|
|
125
|
+
clearTimeout(this.removedSubscriptionsNotice);
|
|
126
|
+
this.removedSubscriptionsNotice = setTimeout(() => {
|
|
127
|
+
var _a, _b;
|
|
128
|
+
for (let [interval, count] of this.removedSubscriptions) {
|
|
129
|
+
let state = this.subscriptions.get(interval);
|
|
130
|
+
console.debug(`[Banta/TimerPool] ${count} unsubscribed from ${interval}ms [${(_b = (_a = state === null || state === void 0 ? void 0 : state.subscribers) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : 0} remain]`);
|
|
131
|
+
}
|
|
132
|
+
if (this.subscriptions.size === 0)
|
|
133
|
+
console.debug(`[Banta/TimerPool] All subscriptions have been removed. Now idle.`);
|
|
134
|
+
this.removedSubscriptions.clear();
|
|
135
|
+
});
|
|
136
|
+
};
|
|
137
|
+
}
|
|
138
|
+
};
|
|
139
|
+
TimerPool = __decorate([
|
|
140
|
+
Injectable()
|
|
141
|
+
], TimerPool);
|
|
142
|
+
|
|
61
143
|
class TimestampComponent {
|
|
62
144
|
constructor(timerPool) {
|
|
63
145
|
this.timerPool = timerPool;
|
|
@@ -156,7 +238,7 @@ class TimestampComponent {
|
|
|
156
238
|
}
|
|
157
239
|
}
|
|
158
240
|
}
|
|
159
|
-
TimestampComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TimestampComponent, deps: [{ token:
|
|
241
|
+
TimestampComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TimestampComponent, deps: [{ token: TimerPool }], target: i0.ɵɵFactoryTarget.Component });
|
|
160
242
|
TimestampComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: TimestampComponent, selector: "banta-timestamp", inputs: { value: "value" }, ngImport: i0, template: `
|
|
161
243
|
<span *ngIf="showAbsolute" [title]="value | date : 'short'">
|
|
162
244
|
{{value | date : 'shortDate'}}
|
|
@@ -175,7 +257,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
175
257
|
{{relative}}
|
|
176
258
|
</span>
|
|
177
259
|
` }]
|
|
178
|
-
}], ctorParameters: function () { return [{ type:
|
|
260
|
+
}], ctorParameters: function () { return [{ type: TimerPool }]; }, propDecorators: { value: [{
|
|
179
261
|
type: Input
|
|
180
262
|
}] } });
|
|
181
263
|
|
|
@@ -278,7 +360,7 @@ class BantaMarkdownToHtmlPipe {
|
|
|
278
360
|
}));
|
|
279
361
|
}
|
|
280
362
|
}
|
|
281
|
-
BantaMarkdownToHtmlPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaMarkdownToHtmlPipe, deps: [{ token: i1
|
|
363
|
+
BantaMarkdownToHtmlPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaMarkdownToHtmlPipe, deps: [{ token: i1.DomSanitizer }, { token: BANTA_SDK_OPTIONS, optional: true }], target: i0.ɵɵFactoryTarget.Pipe });
|
|
282
364
|
BantaMarkdownToHtmlPipe.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "14.2.12", ngImport: i0, type: BantaMarkdownToHtmlPipe, name: "markdownToHtml" });
|
|
283
365
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaMarkdownToHtmlPipe, decorators: [{
|
|
284
366
|
type: Pipe,
|
|
@@ -286,7 +368,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
286
368
|
name: 'markdownToHtml'
|
|
287
369
|
}]
|
|
288
370
|
}], ctorParameters: function () {
|
|
289
|
-
return [{ type: i1
|
|
371
|
+
return [{ type: i1.DomSanitizer }, { type: undefined, decorators: [{
|
|
290
372
|
type: Inject,
|
|
291
373
|
args: [BANTA_SDK_OPTIONS]
|
|
292
374
|
}, {
|
|
@@ -304,14 +386,14 @@ class BantaTrustResourceUrlPipe {
|
|
|
304
386
|
return this.sanitizer.bypassSecurityTrustResourceUrl(value);
|
|
305
387
|
}
|
|
306
388
|
}
|
|
307
|
-
BantaTrustResourceUrlPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaTrustResourceUrlPipe, deps: [{ token: i1
|
|
389
|
+
BantaTrustResourceUrlPipe.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaTrustResourceUrlPipe, deps: [{ token: i1.DomSanitizer }], target: i0.ɵɵFactoryTarget.Pipe });
|
|
308
390
|
BantaTrustResourceUrlPipe.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "14.2.12", ngImport: i0, type: BantaTrustResourceUrlPipe, name: "trustResourceUrl" });
|
|
309
391
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaTrustResourceUrlPipe, decorators: [{
|
|
310
392
|
type: Pipe,
|
|
311
393
|
args: [{
|
|
312
394
|
name: 'trustResourceUrl'
|
|
313
395
|
}]
|
|
314
|
-
}], ctorParameters: function () { return [{ type: i1
|
|
396
|
+
}], ctorParameters: function () { return [{ type: i1.DomSanitizer }]; } });
|
|
315
397
|
|
|
316
398
|
class BantaMentionLinkerPipe {
|
|
317
399
|
transform(value, links) {
|
|
@@ -524,89 +606,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
524
606
|
type: Output
|
|
525
607
|
}] } });
|
|
526
608
|
|
|
527
|
-
/**
|
|
528
|
-
* Provides a way to hook in to a shared set of timers, instead of creating a timer per instance.
|
|
529
|
-
* This is very useful for cases where the update is not extremely time-sensitive, but happens at scale.
|
|
530
|
-
* The principal use case is the TimestampComponent. When several hundred (or several thousand) comments are
|
|
531
|
-
* being displayed, we do not want to trigger thousands of independent relative timestamp updates, because over
|
|
532
|
-
* time the updates will saturate the CPU since they don't perfectly align.
|
|
533
|
-
*/
|
|
534
|
-
let TimerPool = class TimerPool {
|
|
535
|
-
constructor() {
|
|
536
|
-
this.subscriptions = new Map();
|
|
537
|
-
this.newSubscriptions = new Map();
|
|
538
|
-
this.removedSubscriptions = new Map();
|
|
539
|
-
}
|
|
540
|
-
addTimer(interval, callback) {
|
|
541
|
-
var _a;
|
|
542
|
-
if (interval <= 0) {
|
|
543
|
-
console.warn(`Refusing to set timer with interval of ${interval}!`);
|
|
544
|
-
return () => { };
|
|
545
|
-
}
|
|
546
|
-
let state;
|
|
547
|
-
let sizeWas = this.subscriptions.size;
|
|
548
|
-
if (!this.subscriptions.has(interval)) {
|
|
549
|
-
state = { subscribers: [] };
|
|
550
|
-
state.handle = setInterval(() => {
|
|
551
|
-
console.debug(`[Banta/TimerPool] Notifying ${state.subscribers.length} subs [${interval}ms]`);
|
|
552
|
-
state.subscribers.forEach(sub => sub());
|
|
553
|
-
}, interval);
|
|
554
|
-
this.subscriptions.set(interval, state);
|
|
555
|
-
}
|
|
556
|
-
else {
|
|
557
|
-
state = this.subscriptions.get(interval);
|
|
558
|
-
}
|
|
559
|
-
state.subscribers.push(callback);
|
|
560
|
-
// Debug information //////////////////////////
|
|
561
|
-
//
|
|
562
|
-
if (!this.newSubscriptions.has(interval))
|
|
563
|
-
this.newSubscriptions.set(interval, 0);
|
|
564
|
-
this.newSubscriptions.set(interval, ((_a = this.newSubscriptions.get(interval)) !== null && _a !== void 0 ? _a : 0) + 1);
|
|
565
|
-
clearTimeout(this.newSubscriptionsNotice);
|
|
566
|
-
this.newSubscriptionsNotice = setTimeout(() => {
|
|
567
|
-
for (let [interval, count] of this.newSubscriptions) {
|
|
568
|
-
console.debug(`[Banta/TimerPool] ${count} new subscriptions to ${interval}ms [${state.subscribers.length} total]`);
|
|
569
|
-
}
|
|
570
|
-
this.newSubscriptions.clear();
|
|
571
|
-
});
|
|
572
|
-
//
|
|
573
|
-
///////////////////////////////////////////////
|
|
574
|
-
if (sizeWas === 0) {
|
|
575
|
-
console.debug(`[Banta/TimerPool] No longer idle.`);
|
|
576
|
-
}
|
|
577
|
-
// Unsubscribe function
|
|
578
|
-
return () => {
|
|
579
|
-
var _a;
|
|
580
|
-
let state = this.subscriptions.get(interval);
|
|
581
|
-
let index = state.subscribers.indexOf(callback);
|
|
582
|
-
if (index >= 0)
|
|
583
|
-
state.subscribers.splice(index, 1);
|
|
584
|
-
if (state.subscribers.length === 0) {
|
|
585
|
-
clearInterval(state.handle);
|
|
586
|
-
this.subscriptions.delete(interval);
|
|
587
|
-
}
|
|
588
|
-
if (!this.removedSubscriptions.has(interval))
|
|
589
|
-
this.removedSubscriptions.set(interval, 0);
|
|
590
|
-
this.removedSubscriptions.set(interval, ((_a = this.removedSubscriptions.get(interval)) !== null && _a !== void 0 ? _a : 0) + 1);
|
|
591
|
-
// Debug information ////////////////////////////////////////////////////////////////////
|
|
592
|
-
clearTimeout(this.removedSubscriptionsNotice);
|
|
593
|
-
this.removedSubscriptionsNotice = setTimeout(() => {
|
|
594
|
-
var _a, _b;
|
|
595
|
-
for (let [interval, count] of this.removedSubscriptions) {
|
|
596
|
-
let state = this.subscriptions.get(interval);
|
|
597
|
-
console.debug(`[Banta/TimerPool] ${count} unsubscribed from ${interval}ms [${(_b = (_a = state === null || state === void 0 ? void 0 : state.subscribers) === null || _a === void 0 ? void 0 : _a.length) !== null && _b !== void 0 ? _b : 0} remain]`);
|
|
598
|
-
}
|
|
599
|
-
if (this.subscriptions.size === 0)
|
|
600
|
-
console.debug(`[Banta/TimerPool] All subscriptions have been removed. Now idle.`);
|
|
601
|
-
this.removedSubscriptions.clear();
|
|
602
|
-
});
|
|
603
|
-
};
|
|
604
|
-
}
|
|
605
|
-
};
|
|
606
|
-
TimerPool = __decorate([
|
|
607
|
-
Injectable()
|
|
608
|
-
], TimerPool);
|
|
609
|
-
|
|
610
609
|
const COMPONENTS$3 = [
|
|
611
610
|
TimestampComponent,
|
|
612
611
|
LightboxComponent,
|
|
@@ -7242,13 +7241,13 @@ class EmojiSelectorPanelComponent {
|
|
|
7242
7241
|
this.categories = this.pairs(cats).map(pair => pair[1]);
|
|
7243
7242
|
}
|
|
7244
7243
|
}
|
|
7245
|
-
EmojiSelectorPanelComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: EmojiSelectorPanelComponent, deps: [{ token: i1
|
|
7244
|
+
EmojiSelectorPanelComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: EmojiSelectorPanelComponent, deps: [{ token: i1.DomSanitizer }, { token: BANTA_SDK_OPTIONS, optional: true }], target: i0.ɵɵFactoryTarget.Component });
|
|
7246
7245
|
EmojiSelectorPanelComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: EmojiSelectorPanelComponent, selector: "emoji-selector-panel", outputs: { selected: "selected" }, ngImport: i0, template: "<div class=\"search-box\" *ngIf=\"searchVisible\">\r\n\t<a mat-icon-button href=\"javascript:;\" (click)=\"hideSearch()\">\r\n\t\t<mat-icon>arrow_back</mat-icon>\r\n\t</a>\r\n\t<mat-form-field appearance=\"outline\" floatLabel=\"always\">\r\n\t\t<mat-label>Search for emoji</mat-label>\r\n\t\t<input name=\"search\" type=\"text\" matInput placeholder=\"Start typing\" [(ngModel)]=\"searchQuery\" />\r\n\t</mat-form-field>\r\n</div>\r\n\r\n<div class=\"selector\">\r\n\t<ng-container *ngIf=\"searchVisible\">\r\n\t\t<div class=\"emoji-list\">\r\n\t\t\t<a href=\"javascript:;\" (click)=\"select(emoji.char)\" \r\n\t\t\t\t*ngFor=\"let emoji of searchResults\" [innerHtml]=\"emoji.html || ''\">\r\n\t\t\t</a>\r\n\t\t</div>\r\n\t</ng-container>\r\n\t<ng-container *ngIf=\"!searchVisible\">\r\n\t\t<div class=\"categories\">\r\n\t\t\t<ng-container *ngIf=\"!searchVisible\">\r\n\t\t\t\t<a [title]=\"humanize(category.name)\" [class.active]=\"activeCategory === category.name\" mat-icon-button *ngFor=\"let category of categories\" (click)=\"activeCategory = category.name\">\r\n\t\t\t\t\t<mat-icon>{{category.icon}}</mat-icon>\r\n\t\t\t\t</a>\r\n\r\n\t\t\t\t<a title=\"Search\" [class.active] mat-icon-button (click)=\"showSearch()\">\r\n\t\t\t\t\t<mat-icon>search</mat-icon>\r\n\t\t\t\t</a>\r\n\t\t\t</ng-container>\r\n\t\t</div>\r\n\t\t<ng-container *ngFor=\"let category of categories\">\r\n\t\t\t<div class=\"emoji-list\" *ngIf=\"activeCategory && activeCategory == category.name\">\r\n\t\t\t\t<a href=\"javascript:;\" (click)=\"select(emoji.char)\" \r\n\t\t\t\t\t*ngFor=\"let emoji of category.emojis\" [innerHtml]=\"emoji.html || ''\">\r\n\t\t\t\t</a>\r\n\t\t\t</div>\r\n\t\t</ng-container>\r\n\t</ng-container>\r\n</div>", styles: [":host{background:#111;color:#fff;border:1px solid #333;border-radius:5px;padding:.5em;width:calc(9*(32px + 1em));max-width:calc(100vw - 1.5em - 5px)}.selector{display:flex;flex-direction:column}.categories a{opacity:.25;transition:.4s opacity ease-in-out}.categories a:hover{opacity:.5}.categories a.active{opacity:1}.emoji-list{flex-grow:1;overflow-y:auto;height:20em}.emoji-list a{display:inline-block;padding:2px;margin:4px;background-color:#111}.emoji-list a ::ng-deep .emoji{width:32px;height:32px}.emoji-list a:hover{background-color:#333}.search-box{display:flex;align-items:baseline}.search-box mat-form-field{flex-grow:1}@media (max-width: 500px){.selector{flex-direction:row;height:27em}.emoji-list{height:auto}}:host-context(.banta-mobile) .selector{flex-direction:row;height:27em}:host-context(.banta-mobile) .emoji-list{height:auto}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4$1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i2$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i6.MatAnchor, selector: "a[mat-button], a[mat-raised-button], a[mat-icon-button], a[mat-fab], a[mat-mini-fab], a[mat-stroked-button], a[mat-flat-button]", inputs: ["disabled", "disableRipple", "color", "tabIndex"], exportAs: ["matButton", "matAnchor"] }, { kind: "component", type: i6$1.MatFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i6$1.MatLabel, selector: "mat-label" }, { kind: "directive", type: i7.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }] });
|
|
7247
7246
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: EmojiSelectorPanelComponent, decorators: [{
|
|
7248
7247
|
type: Component,
|
|
7249
7248
|
args: [{ selector: 'emoji-selector-panel', template: "<div class=\"search-box\" *ngIf=\"searchVisible\">\r\n\t<a mat-icon-button href=\"javascript:;\" (click)=\"hideSearch()\">\r\n\t\t<mat-icon>arrow_back</mat-icon>\r\n\t</a>\r\n\t<mat-form-field appearance=\"outline\" floatLabel=\"always\">\r\n\t\t<mat-label>Search for emoji</mat-label>\r\n\t\t<input name=\"search\" type=\"text\" matInput placeholder=\"Start typing\" [(ngModel)]=\"searchQuery\" />\r\n\t</mat-form-field>\r\n</div>\r\n\r\n<div class=\"selector\">\r\n\t<ng-container *ngIf=\"searchVisible\">\r\n\t\t<div class=\"emoji-list\">\r\n\t\t\t<a href=\"javascript:;\" (click)=\"select(emoji.char)\" \r\n\t\t\t\t*ngFor=\"let emoji of searchResults\" [innerHtml]=\"emoji.html || ''\">\r\n\t\t\t</a>\r\n\t\t</div>\r\n\t</ng-container>\r\n\t<ng-container *ngIf=\"!searchVisible\">\r\n\t\t<div class=\"categories\">\r\n\t\t\t<ng-container *ngIf=\"!searchVisible\">\r\n\t\t\t\t<a [title]=\"humanize(category.name)\" [class.active]=\"activeCategory === category.name\" mat-icon-button *ngFor=\"let category of categories\" (click)=\"activeCategory = category.name\">\r\n\t\t\t\t\t<mat-icon>{{category.icon}}</mat-icon>\r\n\t\t\t\t</a>\r\n\r\n\t\t\t\t<a title=\"Search\" [class.active] mat-icon-button (click)=\"showSearch()\">\r\n\t\t\t\t\t<mat-icon>search</mat-icon>\r\n\t\t\t\t</a>\r\n\t\t\t</ng-container>\r\n\t\t</div>\r\n\t\t<ng-container *ngFor=\"let category of categories\">\r\n\t\t\t<div class=\"emoji-list\" *ngIf=\"activeCategory && activeCategory == category.name\">\r\n\t\t\t\t<a href=\"javascript:;\" (click)=\"select(emoji.char)\" \r\n\t\t\t\t\t*ngFor=\"let emoji of category.emojis\" [innerHtml]=\"emoji.html || ''\">\r\n\t\t\t\t</a>\r\n\t\t\t</div>\r\n\t\t</ng-container>\r\n\t</ng-container>\r\n</div>", styles: [":host{background:#111;color:#fff;border:1px solid #333;border-radius:5px;padding:.5em;width:calc(9*(32px + 1em));max-width:calc(100vw - 1.5em - 5px)}.selector{display:flex;flex-direction:column}.categories a{opacity:.25;transition:.4s opacity ease-in-out}.categories a:hover{opacity:.5}.categories a.active{opacity:1}.emoji-list{flex-grow:1;overflow-y:auto;height:20em}.emoji-list a{display:inline-block;padding:2px;margin:4px;background-color:#111}.emoji-list a ::ng-deep .emoji{width:32px;height:32px}.emoji-list a:hover{background-color:#333}.search-box{display:flex;align-items:baseline}.search-box mat-form-field{flex-grow:1}@media (max-width: 500px){.selector{flex-direction:row;height:27em}.emoji-list{height:auto}}:host-context(.banta-mobile) .selector{flex-direction:row;height:27em}:host-context(.banta-mobile) .emoji-list{height:auto}\n"] }]
|
|
7250
7249
|
}], ctorParameters: function () {
|
|
7251
|
-
return [{ type: i1
|
|
7250
|
+
return [{ type: i1.DomSanitizer }, { type: undefined, decorators: [{
|
|
7252
7251
|
type: Inject,
|
|
7253
7252
|
args: [BANTA_SDK_OPTIONS]
|
|
7254
7253
|
}, {
|
|
@@ -7323,7 +7322,7 @@ class EmojiSelectorButtonComponent {
|
|
|
7323
7322
|
this.overlayRef.attach(this.selectorPanelTemplate);
|
|
7324
7323
|
}
|
|
7325
7324
|
}
|
|
7326
|
-
EmojiSelectorButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: EmojiSelectorButtonComponent, deps: [{ token: i0.ElementRef }, { token: i1$
|
|
7325
|
+
EmojiSelectorButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: EmojiSelectorButtonComponent, deps: [{ token: i0.ElementRef }, { token: i1$1.Overlay }], target: i0.ɵɵFactoryTarget.Component });
|
|
7327
7326
|
EmojiSelectorButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: EmojiSelectorButtonComponent, selector: "emoji-selector-button", inputs: { overlayX: "overlayX", overlayY: "overlayY", originX: "originX", originY: "originY" }, outputs: { selected: "selected" }, viewQueries: [{ propertyName: "selectorPanelTemplate", first: true, predicate: ["selectorPanelTemplate"], descendants: true }], ngImport: i0, template: `
|
|
7328
7327
|
<button #button type="button" mat-icon-button (click)="show()">
|
|
7329
7328
|
<mat-icon>emoji_emotions</mat-icon>
|
|
@@ -7348,7 +7347,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
7348
7347
|
></emoji-selector-panel>
|
|
7349
7348
|
</ng-template>
|
|
7350
7349
|
`, styles: [":host{display:block;position:relative}button{color:#666}\n"] }]
|
|
7351
|
-
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1$
|
|
7350
|
+
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1$1.Overlay }]; }, propDecorators: { selectorPanelTemplate: [{
|
|
7352
7351
|
type: ViewChild,
|
|
7353
7352
|
args: ['selectorPanelTemplate']
|
|
7354
7353
|
}], selected: [{
|
|
@@ -8082,12 +8081,12 @@ class AttachmentButtonComponent {
|
|
|
8082
8081
|
});
|
|
8083
8082
|
}
|
|
8084
8083
|
}
|
|
8085
|
-
AttachmentButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: AttachmentButtonComponent, deps: [{ token: i1$
|
|
8084
|
+
AttachmentButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: AttachmentButtonComponent, deps: [{ token: i1$2.CDNProvider }], target: i0.ɵɵFactoryTarget.Component });
|
|
8086
8085
|
AttachmentButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: AttachmentButtonComponent, selector: "banta-attachment-button", outputs: { addedAttachment: "addedAttachment", attachmentError: "attachmentError" }, viewQueries: [{ propertyName: "fileInput", first: true, predicate: ["fileUpload"], descendants: true }], ngImport: i0, template: "<button matTooltip=\"Add an image or gif\" #button type=\"button\" mat-icon-button (click)=\"show()\">\r\n\t<mat-icon>image</mat-icon>\r\n</button>\r\n<input style=\"display: none;\" #fileUpload [multiple]=\"false\" (change)=\"fileChange($event)\" type=\"file\" >", styles: ["button{color:#666}\n"], dependencies: [{ kind: "component", type: i2$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i6.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "directive", type: i11.MatTooltip, selector: "[matTooltip]", exportAs: ["matTooltip"] }] });
|
|
8087
8086
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: AttachmentButtonComponent, decorators: [{
|
|
8088
8087
|
type: Component,
|
|
8089
8088
|
args: [{ selector: 'banta-attachment-button', template: "<button matTooltip=\"Add an image or gif\" #button type=\"button\" mat-icon-button (click)=\"show()\">\r\n\t<mat-icon>image</mat-icon>\r\n</button>\r\n<input style=\"display: none;\" #fileUpload [multiple]=\"false\" (change)=\"fileChange($event)\" type=\"file\" >", styles: ["button{color:#666}\n"] }]
|
|
8090
|
-
}], ctorParameters: function () { return [{ type: i1$
|
|
8089
|
+
}], ctorParameters: function () { return [{ type: i1$2.CDNProvider }]; }, propDecorators: { fileInput: [{
|
|
8091
8090
|
type: ViewChild,
|
|
8092
8091
|
args: ['fileUpload', { static: false }]
|
|
8093
8092
|
}], addedAttachment: [{
|