@banta/sdk 4.7.9 → 4.7.11
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/esm2020/lib/banta-sdk.module.mjs +5 -6
- package/esm2020/lib/chat-backend.mjs +18 -6
- package/esm2020/lib/chat-source-base.mjs +1 -1
- package/esm2020/lib/chat-source.mjs +39 -10
- package/esm2020/lib/comments/banta-comments/banta-comments.component.mjs +2 -2
- package/esm2020/lib/comments/comment-field/comment-field.component.mjs +5 -2
- package/esm2020/lib/common/attachment/attachment.component.mjs +3 -5
- package/esm2020/lib/common/attachments/attachments.component.mjs +6 -3
- package/esm2020/lib/common/common.module.mjs +10 -1
- package/esm2020/lib/common/timer-pool.service.mjs +83 -0
- package/esm2020/lib/common/timestamp.component.mjs +24 -14
- package/fesm2015/banta-sdk.mjs +117 -51
- package/fesm2015/banta-sdk.mjs.map +1 -1
- package/fesm2020/banta-sdk.mjs +113 -51
- package/fesm2020/banta-sdk.mjs.map +1 -1
- package/lib/chat-backend.d.ts +5 -0
- package/lib/chat-source-base.d.ts +1 -0
- package/lib/chat-source.d.ts +4 -0
- package/lib/common/attachments/attachments.component.d.ts +1 -0
- package/lib/common/common.module.d.ts +2 -0
- package/lib/common/timer-pool.service.d.ts +15 -0
- package/lib/common/timestamp.component.d.ts +6 -1
- package/package.json +1 -1
|
@@ -63,15 +63,13 @@ export class BantaAttachmentComponent {
|
|
|
63
63
|
|| !this.attachment.url);
|
|
64
64
|
}
|
|
65
65
|
get isImageAttachment() {
|
|
66
|
-
|
|
67
|
-
return true;
|
|
68
|
-
return false;
|
|
66
|
+
return this.attachment?.type?.startsWith('image/');
|
|
69
67
|
}
|
|
70
68
|
get hasFrame() {
|
|
71
69
|
if (!this.attachment)
|
|
72
70
|
return false;
|
|
73
71
|
return this.attachment.type === 'iframe' || (this.attachment.type === 'card'
|
|
74
|
-
&& this.attachment.card
|
|
72
|
+
&& this.attachment.card?.player);
|
|
75
73
|
}
|
|
76
74
|
get frameUrl() {
|
|
77
75
|
if (!this.attachment)
|
|
@@ -111,4 +109,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
111
109
|
type: HostBinding,
|
|
112
110
|
args: ['class.loading']
|
|
113
111
|
}] } });
|
|
114
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.ts","../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, HostBinding, Input, Output } from \"@angular/core\";\r\nimport { ChatMessageAttachment } from \"@banta/common\";\r\nimport { Subject } from \"rxjs\";\r\n\r\n@Component({\r\n    selector: 'banta-attachment',\r\n    templateUrl: './attachment.component.html',\r\n    styleUrls: ['./attachment.component.scss']\r\n})\r\nexport class BantaAttachmentComponent {\r\n    constructor(\r\n        private elementRef: ElementRef<HTMLElement>\r\n    ) {\r\n\r\n    }\r\n\r\n    private _attachment: ChatMessageAttachment;\r\n    @Input() get attachment() { return this._attachment; }\r\n    set attachment(value) {\r\n        this._attachment = value;\r\n        this.checkLoad();\r\n    }\r\n    @Input() loading = false;\r\n    @Input() editing = false;\r\n    @Input() loadingMessage: string = 'Please wait...';\r\n    @Input() error = false;\r\n    @Input() errorMessage: string = 'An error has occurred';\r\n    @Output() removed = new Subject<void>();\r\n    @Output() activated = new Subject<void>();\r\n    @Output() loaded = new Subject<void>();\r\n\r\n    ngOnInit() {\r\n        if (typeof window !== 'undefined') {\r\n            setTimeout(() => {\r\n                if (!window['twttr'])\r\n                    return;\r\n                window['twttr'].widgets.load();\r\n            }, 100);\r\n        }\r\n    }\r\n\r\n    private _viewLoaded = false;\r\n    ngAfterViewInit() {\r\n        this._viewLoaded = true;\r\n        this.checkLoad();\r\n    }\r\n\r\n    private checkLoad() {\r\n        if (!this._attachment || !this._viewLoaded || !this.elementRef?.nativeElement)\r\n            return;\r\n\r\n        if (typeof window === 'undefined')\r\n            this.loaded.next();\r\n        else\r\n            setTimeout(() => this.loaded.next(), 250);\r\n    }\r\n\r\n    activate() {\r\n        this.activated.next();\r\n    }\r\n\r\n    remove() {\r\n        this.removed.next();\r\n    }\r\n\r\n    get isError() {\r\n        return this.error || this.attachment?.transientState?.error;\r\n    }\r\n\r\n    get theErrorMessage() {\r\n        return this.errorMessage || this.attachment?.transientState?.errorMessage;\r\n    }\r\n\r\n    @HostBinding('class.loading')\r\n    get isLoading() {\r\n        return this.editing && (\r\n            this.loading || !this.attachment || this.attachment.transientState?.loading \r\n            || !this.attachment.url\r\n        );\r\n    }\r\n\r\n    get isImageAttachment() {\r\n        if (this.attachment.type.startsWith('image/'))\r\n            return true;\r\n        return false;\r\n    }\r\n\r\n    get hasFrame() {\r\n        if (!this.attachment)\r\n            return false;\r\n\r\n        return this.attachment.type === 'iframe' || (\r\n            this.attachment.type === 'card' \r\n            && this.attachment.card.player\r\n        );\r\n    }\r\n\r\n    get frameUrl() {\r\n        if (!this.attachment)\r\n            return undefined;\r\n        \r\n        if (this.attachment.type === 'iframe') {\r\n            return this.attachment.url;\r\n        } else if (this.attachment.type === 'card') {\r\n            return this.attachment.card.player;\r\n        }\r\n    }\r\n}","<button type=\"button\" (click)=\"remove()\" mat-mini-fab color=\"primary\" class=\"remove-button\" *ngIf=\"editing\">\r\n    <mat-icon>close</mat-icon>\r\n</button>\r\n\r\n<ng-container *ngIf=\"isError\">\r\n    <mat-icon class=\"error\">close</mat-icon>\r\n    <em class=\"error\">{{theErrorMessage}}</em>\r\n</ng-container>\r\n<ng-container *ngIf=\"!isError\">\r\n    <ng-container *ngIf=\"isLoading\">\r\n        <mat-spinner></mat-spinner>\r\n        <em>{{loadingMessage}}</em>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"!isLoading && attachment\">\r\n        <iframe *ngIf=\"hasFrame\"\r\n            sandbox=\"allow-scripts allow-popups allow-same-origin allow-presentation\" \r\n            [src]=\"frameUrl | trustResourceUrl\"></iframe>\r\n        <a *ngIf=\"attachment.type === 'card'\" class=\"card-attachment\" [href]=\"attachment.url\" target=\"_blank\" [class.has-image]=\"attachment.card.image\">\r\n            <img \r\n                *ngIf=\"attachment.card.image\"\r\n                class=\"thumbnail\" \r\n                [src]=\"attachment.card.image\"\r\n                />\r\n            <div class=\"description\">\r\n                <h1>{{attachment.card.title}}</h1>\r\n                <div class=\"summary\">\r\n                    {{attachment.card.description}}\r\n                </div>\r\n                <cite>{{attachment.card.url}}</cite>\r\n            </div>\r\n        </a>\r\n        <a class=\"image-attachment\" *ngIf=\"isImageAttachment && attachment.url\" href=\"javascript:;\" (click)=\"activate()\">\r\n            <img [src]=\"attachment.url\" alt=\"Image Attachment\">\r\n        </a>\r\n        <blockquote *ngIf=\"attachment.type === 'tweet'\" \r\n            class=\"twitter-tweet\">\r\n                <p lang=\"en\" dir=\"ltr\"></p>\r\n                <a [href]=\"attachment.url\"></a>\r\n        </blockquote>\r\n    </ng-container>\r\n</ng-container>"]}
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.ts","../../../../../../projects/sdk/src/lib/common/attachment/attachment.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, HostBinding, Input, Output } from \"@angular/core\";\r\nimport { ChatMessageAttachment } from \"@banta/common\";\r\nimport { Subject } from \"rxjs\";\r\n\r\n@Component({\r\n    selector: 'banta-attachment',\r\n    templateUrl: './attachment.component.html',\r\n    styleUrls: ['./attachment.component.scss']\r\n})\r\nexport class BantaAttachmentComponent {\r\n    constructor(\r\n        private elementRef: ElementRef<HTMLElement>\r\n    ) {\r\n\r\n    }\r\n\r\n    private _attachment: ChatMessageAttachment;\r\n    @Input() get attachment() { return this._attachment; }\r\n    set attachment(value) {\r\n        this._attachment = value;\r\n        this.checkLoad();\r\n    }\r\n    @Input() loading = false;\r\n    @Input() editing = false;\r\n    @Input() loadingMessage: string = 'Please wait...';\r\n    @Input() error = false;\r\n    @Input() errorMessage: string = 'An error has occurred';\r\n    @Output() removed = new Subject<void>();\r\n    @Output() activated = new Subject<void>();\r\n    @Output() loaded = new Subject<void>();\r\n\r\n    ngOnInit() {\r\n        if (typeof window !== 'undefined') {\r\n            setTimeout(() => {\r\n                if (!window['twttr'])\r\n                    return;\r\n                window['twttr'].widgets.load();\r\n            }, 100);\r\n        }\r\n    }\r\n\r\n    private _viewLoaded = false;\r\n    ngAfterViewInit() {\r\n        this._viewLoaded = true;\r\n        this.checkLoad();\r\n    }\r\n\r\n    private checkLoad() {\r\n        if (!this._attachment || !this._viewLoaded || !this.elementRef?.nativeElement)\r\n            return;\r\n\r\n        if (typeof window === 'undefined')\r\n            this.loaded.next();\r\n        else\r\n            setTimeout(() => this.loaded.next(), 250);\r\n    }\r\n\r\n    activate() {\r\n        this.activated.next();\r\n    }\r\n\r\n    remove() {\r\n        this.removed.next();\r\n    }\r\n\r\n    get isError() {\r\n        return this.error || this.attachment?.transientState?.error;\r\n    }\r\n\r\n    get theErrorMessage() {\r\n        return this.errorMessage || this.attachment?.transientState?.errorMessage;\r\n    }\r\n\r\n    @HostBinding('class.loading')\r\n    get isLoading() {\r\n        return this.editing && (\r\n            this.loading || !this.attachment || this.attachment.transientState?.loading \r\n            || !this.attachment.url\r\n        );\r\n    }\r\n\r\n    get isImageAttachment() {\r\n        return this.attachment?.type?.startsWith('image/');\r\n    }\r\n\r\n    get hasFrame() {\r\n        if (!this.attachment)\r\n            return false;\r\n\r\n        return this.attachment.type === 'iframe' || (\r\n            this.attachment.type === 'card' \r\n            && this.attachment.card?.player\r\n        );\r\n    }\r\n\r\n    get frameUrl() {\r\n        if (!this.attachment)\r\n            return undefined;\r\n        \r\n        if (this.attachment.type === 'iframe') {\r\n            return this.attachment.url;\r\n        } else if (this.attachment.type === 'card') {\r\n            return this.attachment.card.player;\r\n        }\r\n    }\r\n}","<button type=\"button\" (click)=\"remove()\" mat-mini-fab color=\"primary\" class=\"remove-button\" *ngIf=\"editing\">\r\n    <mat-icon>close</mat-icon>\r\n</button>\r\n\r\n<ng-container *ngIf=\"isError\">\r\n    <mat-icon class=\"error\">close</mat-icon>\r\n    <em class=\"error\">{{theErrorMessage}}</em>\r\n</ng-container>\r\n<ng-container *ngIf=\"!isError\">\r\n    <ng-container *ngIf=\"isLoading\">\r\n        <mat-spinner></mat-spinner>\r\n        <em>{{loadingMessage}}</em>\r\n    </ng-container>\r\n    <ng-container *ngIf=\"!isLoading && attachment\">\r\n        <iframe *ngIf=\"hasFrame\"\r\n            sandbox=\"allow-scripts allow-popups allow-same-origin allow-presentation\" \r\n            [src]=\"frameUrl | trustResourceUrl\"></iframe>\r\n        <a *ngIf=\"attachment.type === 'card'\" class=\"card-attachment\" [href]=\"attachment.url\" target=\"_blank\" [class.has-image]=\"attachment.card.image\">\r\n            <img \r\n                *ngIf=\"attachment.card.image\"\r\n                class=\"thumbnail\" \r\n                [src]=\"attachment.card.image\"\r\n                />\r\n            <div class=\"description\">\r\n                <h1>{{attachment.card.title}}</h1>\r\n                <div class=\"summary\">\r\n                    {{attachment.card.description}}\r\n                </div>\r\n                <cite>{{attachment.card.url}}</cite>\r\n            </div>\r\n        </a>\r\n        <a class=\"image-attachment\" *ngIf=\"isImageAttachment && attachment.url\" href=\"javascript:;\" (click)=\"activate()\">\r\n            <img [src]=\"attachment.url\" alt=\"Image Attachment\">\r\n        </a>\r\n        <blockquote *ngIf=\"attachment.type === 'tweet'\" \r\n            class=\"twitter-tweet\">\r\n                <p lang=\"en\" dir=\"ltr\"></p>\r\n                <a [href]=\"attachment.url\"></a>\r\n        </blockquote>\r\n    </ng-container>\r\n</ng-container>"]}
|
|
@@ -42,11 +42,14 @@ export class BantaAttachmentsComponent {
|
|
|
42
42
|
.filter(x => x.type === 'image/png')
|
|
43
43
|
.map(x => x.url));
|
|
44
44
|
}
|
|
45
|
+
get validAttachments() {
|
|
46
|
+
return this.attachments.filter(x => x.type);
|
|
47
|
+
}
|
|
45
48
|
get inlineAttachments() {
|
|
46
|
-
return this.
|
|
49
|
+
return this.validAttachments.filter(x => x.type !== 'card' && (x.style === 'inline' || !x.style));
|
|
47
50
|
}
|
|
48
51
|
get blockAttachments() {
|
|
49
|
-
return this.
|
|
52
|
+
return this.validAttachments.filter(x => x.style === 'block' || x.type === 'card');
|
|
50
53
|
}
|
|
51
54
|
attachmentId(index, attachment) {
|
|
52
55
|
return attachment.url;
|
|
@@ -69,4 +72,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
69
72
|
}], loaded: [{
|
|
70
73
|
type: Output
|
|
71
74
|
}] } });
|
|
72
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
75
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachments.component.js","sourceRoot":"","sources":["../../../../../../projects/sdk/src/lib/common/attachments/attachments.component.ts","../../../../../../projects/sdk/src/lib/common/attachments/attachments.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, ViewChild } from \"@angular/core\";\r\nimport { ChatMessage, ChatMessageAttachment } from \"@banta/common\";\r\nimport { Subject } from \"rxjs\";\r\nimport { LightboxComponent } from \"../lightbox/lightbox.component\";\r\n\r\n@Component({\r\n    selector: 'banta-attachments',\r\n    templateUrl: './attachments.component.html',\r\n    styleUrls: ['./attachments.component.scss']\r\n})\r\nexport class BantaAttachmentsComponent {\r\n    @Input() attachments: ChatMessageAttachment[];\r\n    @Input() editing = false;\r\n    @ViewChild('lightbox') lightbox: LightboxComponent;\r\n    @Output() remove = new Subject<ChatMessageAttachment>();\r\n    @Output() loaded = new Subject<void>();\r\n\r\n    loadedAttachments = new WeakMap<ChatMessageAttachment, boolean>();\r\n\r\n    ngAfterViewInit() {\r\n\r\n    }\r\n\r\n    markAttachmentLoaded(attachment: ChatMessageAttachment) {\r\n        this.loadedAttachments.set(attachment, true);\r\n\r\n        if (this.allAttachmentsLoaded)\r\n            this.loaded.next();\r\n    }\r\n\r\n    isAttachmentLoaded(attachment: ChatMessageAttachment) {\r\n        return this.loadedAttachments.has(attachment);\r\n    }\r\n\r\n    get allAttachmentsLoaded() {\r\n        return this.attachments.every(x => this.isAttachmentLoaded(x));\r\n    }\r\n\r\n    removeAttachment(attachment: ChatMessageAttachment) {\r\n        this.remove.next(attachment);\r\n    }\r\n\r\n    isImageAttachment(attachment: ChatMessageAttachment) {\r\n        if (attachment.type.startsWith('image/'))\r\n            return true;\r\n        return false;\r\n    }\r\n\r\n    isCardAttachment(attachment: ChatMessageAttachment) {\r\n        if (['card'].includes(attachment.type))\r\n            return true;\r\n        return false;\r\n    }\r\n\r\n    showLightbox(image: ChatMessageAttachment) {\r\n        this.lightbox.open(\r\n            image.url,\r\n            this.attachments\r\n                .filter(x => x.type === 'image/png')\r\n                .map(x => x.url)\r\n        )\r\n    }\r\n\r\n    get validAttachments() {\r\n        return this.attachments.filter(x => x.type);\r\n    }\r\n\r\n    get inlineAttachments() {\r\n        return this.validAttachments.filter(x => x.type !== 'card' && (x.style === 'inline' || !x.style));\r\n    }\r\n\r\n    get blockAttachments() {\r\n        return this.validAttachments.filter(x => x.style === 'block' || x.type === 'card');\r\n    }\r\n\r\n    attachmentId(index: number, attachment: ChatMessageAttachment) {\r\n        return attachment.url;\r\n    }\r\n}","<ng-container *ngIf=\"attachments?.length > 0\">\r\n    <banta-lightbox #lightbox></banta-lightbox>\r\n    <div class=\"block-attachments\">\r\n        <ng-container *ngFor=\"let attachment of blockAttachments; trackBy: attachmentId\">\r\n            <banta-attachment \r\n                [attachment]=\"attachment\"\r\n                [editing]=\"editing\"\r\n                (loaded)=\"markAttachmentLoaded(attachment)\"\r\n                (removed)=\"removeAttachment(attachment)\"\r\n                (activated)=\"showLightbox(attachment)\"\r\n                ></banta-attachment>\r\n        </ng-container>\r\n    </div>\r\n\r\n    <div \r\n        class=\"inline-attachments\" \r\n        [class.single]=\"attachments?.length === 1\" \r\n        *ngIf=\"attachments && attachments?.length > 0\"\r\n        >\r\n        <ng-container *ngFor=\"let attachment of inlineAttachments; trackBy: attachmentId\">\r\n            <banta-attachment \r\n                [attachment]=\"attachment\"\r\n                [editing]=\"editing\"\r\n                (loaded)=\"markAttachmentLoaded(attachment)\"\r\n                (removed)=\"removeAttachment(attachment)\"\r\n                (activated)=\"showLightbox(attachment)\"\r\n                ></banta-attachment>\r\n        </ng-container>\r\n    </div>\r\n</ng-container>"]}
|
|
@@ -10,6 +10,7 @@ import { MatButtonModule } from '@angular/material/button';
|
|
|
10
10
|
import { BantaTrustResourceUrlPipe } from './trust-resource-url.pipe';
|
|
11
11
|
import { BantaAttachmentComponent } from './attachment/attachment.component';
|
|
12
12
|
import { BantaMentionLinkerPipe } from './mention-linker.pipe';
|
|
13
|
+
import { TimerPool } from 'projects/sdk/src/lib/common/timer-pool.service';
|
|
13
14
|
import * as i0 from "@angular/core";
|
|
14
15
|
const COMPONENTS = [
|
|
15
16
|
TimestampComponent,
|
|
@@ -21,6 +22,14 @@ const COMPONENTS = [
|
|
|
21
22
|
BantaAttachmentsComponent
|
|
22
23
|
];
|
|
23
24
|
export class BantaCommonModule {
|
|
25
|
+
static forRoot() {
|
|
26
|
+
return {
|
|
27
|
+
ngModule: BantaCommonModule,
|
|
28
|
+
providers: [
|
|
29
|
+
TimerPool
|
|
30
|
+
]
|
|
31
|
+
};
|
|
32
|
+
}
|
|
24
33
|
}
|
|
25
34
|
BantaCommonModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: BantaCommonModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
|
26
35
|
BantaCommonModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "14.2.12", ngImport: i0, type: BantaCommonModule, declarations: [TimestampComponent,
|
|
@@ -56,4 +65,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
56
65
|
exports: COMPONENTS
|
|
57
66
|
}]
|
|
58
67
|
}] });
|
|
59
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3Nkay9zcmMvbGliL2NvbW1vbi9jb21tb24ubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQXVCLE1BQU0sZUFBZSxDQUFDO0FBQzlELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQzNELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUNsRSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDdkQsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDbEUsT0FBTyxFQUFFLHlCQUF5QixFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDaEYsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFDOUUsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQzNELE9BQU8sRUFBRSx5QkFBeUIsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ3RFLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBQzdFLE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBQy9ELE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxnREFBZ0QsQ0FBQzs7QUFFM0UsTUFBTSxVQUFVLEdBQUc7SUFDZixrQkFBa0I7SUFDbEIsaUJBQWlCO0lBQ2pCLHVCQUF1QjtJQUN2QixzQkFBc0I7SUFDdEIseUJBQXlCO0lBQ3pCLHdCQUF3QjtJQUN4Qix5QkFBeUI7Q0FDNUIsQ0FBQztBQVlGLE1BQU0sT0FBTyxpQkFBaUI7SUFDMUIsTUFBTSxDQUFDLE9BQU87UUFDVixPQUFPO1lBQ0gsUUFBUSxFQUFFLGlCQUFpQjtZQUMzQixTQUFTLEVBQUU7Z0JBQ1AsU0FBUzthQUNaO1NBQ0osQ0FBQTtJQUNMLENBQUM7OytHQVJRLGlCQUFpQjtnSEFBakIsaUJBQWlCLGlCQW5CMUIsa0JBQWtCO1FBQ2xCLGlCQUFpQjtRQUNqQix1QkFBdUI7UUFDdkIsc0JBQXNCO1FBQ3RCLHlCQUF5QjtRQUN6Qix3QkFBd0I7UUFDeEIseUJBQXlCLGFBTXJCLFlBQVk7UUFDWixhQUFhO1FBQ2Isd0JBQXdCO1FBQ3hCLGVBQWUsYUFmbkIsa0JBQWtCO1FBQ2xCLGlCQUFpQjtRQUNqQix1QkFBdUI7UUFDdkIsc0JBQXNCO1FBQ3RCLHlCQUF5QjtRQUN6Qix3QkFBd0I7UUFDeEIseUJBQXlCO2dIQWFoQixpQkFBaUIsWUFQdEIsWUFBWTtRQUNaLGFBQWE7UUFDYix3QkFBd0I7UUFDeEIsZUFBZTs0RkFJVixpQkFBaUI7a0JBVjdCLFFBQVE7bUJBQUM7b0JBQ04sWUFBWSxFQUFFLFVBQVU7b0JBQ3hCLE9BQU8sRUFBRTt3QkFDTCxZQUFZO3dCQUNaLGFBQWE7d0JBQ2Isd0JBQXdCO3dCQUN4QixlQUFlO3FCQUNsQjtvQkFDRCxPQUFPLEVBQUUsVUFBVTtpQkFDdEIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSwgTW9kdWxlV2l0aFByb3ZpZGVycyB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBUaW1lc3RhbXBDb21wb25lbnQgfSBmcm9tICcuL3RpbWVzdGFtcC5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5pbXBvcnQgeyBMaWdodGJveENvbXBvbmVudCB9IGZyb20gJy4vbGlnaHRib3gvbGlnaHRib3guY29tcG9uZW50JztcclxuaW1wb3J0IHsgTWF0SWNvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2ljb24nO1xyXG5pbXBvcnQgeyBCYW50YU1hcmtkb3duVG9IdG1sUGlwZSB9IGZyb20gJy4vbWFya2Rvd24tdG8taHRtbC5waXBlJztcclxuaW1wb3J0IHsgQmFudGFBdHRhY2htZW50c0NvbXBvbmVudCB9IGZyb20gJy4vYXR0YWNobWVudHMvYXR0YWNobWVudHMuY29tcG9uZW50JztcclxuaW1wb3J0IHsgTWF0UHJvZ3Jlc3NTcGlubmVyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvcHJvZ3Jlc3Mtc3Bpbm5lcic7XHJcbmltcG9ydCB7IE1hdEJ1dHRvbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2J1dHRvbic7XHJcbmltcG9ydCB7IEJhbnRhVHJ1c3RSZXNvdXJjZVVybFBpcGUgfSBmcm9tICcuL3RydXN0LXJlc291cmNlLXVybC5waXBlJztcclxuaW1wb3J0IHsgQmFudGFBdHRhY2htZW50Q29tcG9uZW50IH0gZnJvbSAnLi9hdHRhY2htZW50L2F0dGFjaG1lbnQuY29tcG9uZW50JztcclxuaW1wb3J0IHsgQmFudGFNZW50aW9uTGlua2VyUGlwZSB9IGZyb20gJy4vbWVudGlvbi1saW5rZXIucGlwZSc7XHJcbmltcG9ydCB7IFRpbWVyUG9vbCB9IGZyb20gJ3Byb2plY3RzL3Nkay9zcmMvbGliL2NvbW1vbi90aW1lci1wb29sLnNlcnZpY2UnO1xyXG5cclxuY29uc3QgQ09NUE9ORU5UUyA9IFtcclxuICAgIFRpbWVzdGFtcENvbXBvbmVudCxcclxuICAgIExpZ2h0Ym94Q29tcG9uZW50LFxyXG4gICAgQmFudGFNYXJrZG93blRvSHRtbFBpcGUsXHJcbiAgICBCYW50YU1lbnRpb25MaW5rZXJQaXBlLFxyXG4gICAgQmFudGFUcnVzdFJlc291cmNlVXJsUGlwZSxcclxuICAgIEJhbnRhQXR0YWNobWVudENvbXBvbmVudCxcclxuICAgIEJhbnRhQXR0YWNobWVudHNDb21wb25lbnRcclxuXTtcclxuXHJcbkBOZ01vZHVsZSh7XHJcbiAgICBkZWNsYXJhdGlvbnM6IENPTVBPTkVOVFMsXHJcbiAgICBpbXBvcnRzOiBbXHJcbiAgICAgICAgQ29tbW9uTW9kdWxlLFxyXG4gICAgICAgIE1hdEljb25Nb2R1bGUsXHJcbiAgICAgICAgTWF0UHJvZ3Jlc3NTcGlubmVyTW9kdWxlLFxyXG4gICAgICAgIE1hdEJ1dHRvbk1vZHVsZVxyXG4gICAgXSxcclxuICAgIGV4cG9ydHM6IENPTVBPTkVOVFNcclxufSlcclxuZXhwb3J0IGNsYXNzIEJhbnRhQ29tbW9uTW9kdWxlIHtcclxuICAgIHN0YXRpYyBmb3JSb290KCk6IE1vZHVsZVdpdGhQcm92aWRlcnM8QmFudGFDb21tb25Nb2R1bGU+IHtcclxuICAgICAgICByZXR1cm4ge1xyXG4gICAgICAgICAgICBuZ01vZHVsZTogQmFudGFDb21tb25Nb2R1bGUsXHJcbiAgICAgICAgICAgIHByb3ZpZGVyczogW1xyXG4gICAgICAgICAgICAgICAgVGltZXJQb29sXHJcbiAgICAgICAgICAgIF1cclxuICAgICAgICB9XHJcbiAgICB9XHJcbn0iXX0=
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { Injectable } from "@banta/common";
|
|
3
|
+
/**
|
|
4
|
+
* Provides a way to hook in to a shared set of timers, instead of creating a timer per instance.
|
|
5
|
+
* This is very useful for cases where the update is not extremely time-sensitive, but happens at scale.
|
|
6
|
+
* The principal use case is the TimestampComponent. When several hundred (or several thousand) comments are
|
|
7
|
+
* being displayed, we do not want to trigger thousands of independent relative timestamp updates, because over
|
|
8
|
+
* time the updates will saturate the CPU since they don't perfectly align.
|
|
9
|
+
*/
|
|
10
|
+
let TimerPool = class TimerPool {
|
|
11
|
+
constructor() {
|
|
12
|
+
this.subscriptions = new Map();
|
|
13
|
+
this.newSubscriptions = new Map();
|
|
14
|
+
this.removedSubscriptions = new Map();
|
|
15
|
+
}
|
|
16
|
+
addTimer(interval, callback) {
|
|
17
|
+
if (interval <= 0) {
|
|
18
|
+
console.warn(`Refusing to set timer with interval of ${interval}!`);
|
|
19
|
+
return () => { };
|
|
20
|
+
}
|
|
21
|
+
let state;
|
|
22
|
+
let sizeWas = this.subscriptions.size;
|
|
23
|
+
if (!this.subscriptions.has(interval)) {
|
|
24
|
+
state = { subscribers: [] };
|
|
25
|
+
state.handle = setInterval(() => {
|
|
26
|
+
console.debug(`[Banta/TimerPool] Notifying ${state.subscribers.length} subs [${interval}ms]`);
|
|
27
|
+
state.subscribers.forEach(sub => sub());
|
|
28
|
+
}, interval);
|
|
29
|
+
this.subscriptions.set(interval, state);
|
|
30
|
+
}
|
|
31
|
+
else {
|
|
32
|
+
state = this.subscriptions.get(interval);
|
|
33
|
+
}
|
|
34
|
+
state.subscribers.push(callback);
|
|
35
|
+
// Debug information //////////////////////////
|
|
36
|
+
//
|
|
37
|
+
if (!this.newSubscriptions.has(interval))
|
|
38
|
+
this.newSubscriptions.set(interval, 0);
|
|
39
|
+
this.newSubscriptions.set(interval, (this.newSubscriptions.get(interval) ?? 0) + 1);
|
|
40
|
+
clearTimeout(this.newSubscriptionsNotice);
|
|
41
|
+
this.newSubscriptionsNotice = setTimeout(() => {
|
|
42
|
+
for (let [interval, count] of this.newSubscriptions) {
|
|
43
|
+
console.debug(`[Banta/TimerPool] ${count} new subscriptions to ${interval}ms [${state.subscribers.length} total]`);
|
|
44
|
+
}
|
|
45
|
+
this.newSubscriptions.clear();
|
|
46
|
+
});
|
|
47
|
+
//
|
|
48
|
+
///////////////////////////////////////////////
|
|
49
|
+
if (sizeWas === 0) {
|
|
50
|
+
console.debug(`[Banta/TimerPool] No longer idle.`);
|
|
51
|
+
}
|
|
52
|
+
// Unsubscribe function
|
|
53
|
+
return () => {
|
|
54
|
+
let state = this.subscriptions.get(interval);
|
|
55
|
+
let index = state.subscribers.indexOf(callback);
|
|
56
|
+
if (index >= 0)
|
|
57
|
+
state.subscribers.splice(index, 1);
|
|
58
|
+
if (state.subscribers.length === 0) {
|
|
59
|
+
clearInterval(state.handle);
|
|
60
|
+
this.subscriptions.delete(interval);
|
|
61
|
+
}
|
|
62
|
+
if (!this.removedSubscriptions.has(interval))
|
|
63
|
+
this.removedSubscriptions.set(interval, 0);
|
|
64
|
+
this.removedSubscriptions.set(interval, (this.removedSubscriptions.get(interval) ?? 0) + 1);
|
|
65
|
+
// Debug information ////////////////////////////////////////////////////////////////////
|
|
66
|
+
clearTimeout(this.removedSubscriptionsNotice);
|
|
67
|
+
this.removedSubscriptionsNotice = setTimeout(() => {
|
|
68
|
+
for (let [interval, count] of this.removedSubscriptions) {
|
|
69
|
+
let state = this.subscriptions.get(interval);
|
|
70
|
+
console.debug(`[Banta/TimerPool] ${count} unsubscribed from ${interval}ms [${state?.subscribers?.length ?? 0} remain]`);
|
|
71
|
+
}
|
|
72
|
+
if (this.subscriptions.size === 0)
|
|
73
|
+
console.debug(`[Banta/TimerPool] All subscriptions have been removed. Now idle.`);
|
|
74
|
+
this.removedSubscriptions.clear();
|
|
75
|
+
});
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
TimerPool = __decorate([
|
|
80
|
+
Injectable()
|
|
81
|
+
], TimerPool);
|
|
82
|
+
export { TimerPool };
|
|
83
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timer-pool.service.js","sourceRoot":"","sources":["../../../../../projects/sdk/src/lib/common/timer-pool.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from \"@banta/common\";\r\n\r\ninterface TimerState {\r\n    handle?;\r\n    subscribers: (() => void)[];\r\n}\r\n\r\n/**\r\n * Provides a way to hook in to a shared set of timers, instead of creating a timer per instance.\r\n * This is very useful for cases where the update is not extremely time-sensitive, but happens at scale.\r\n * The principal use case is the TimestampComponent. When several hundred (or several thousand) comments are \r\n * being displayed, we do not want to trigger thousands of independent relative timestamp updates, because over \r\n * time the updates will saturate the CPU since they don't perfectly align. \r\n */\r\n@Injectable()\r\nexport class TimerPool {\r\n    private subscriptions = new Map<number, TimerState>();\r\n\r\n    private newSubscriptionsNotice;\r\n    private newSubscriptions = new Map<number, number>();\r\n\r\n    private removedSubscriptionsNotice;\r\n    private removedSubscriptions = new Map<number, number>();\r\n\r\n    addTimer(interval: number, callback: () => void) {\r\n        if (interval <= 0) {\r\n            console.warn(`Refusing to set timer with interval of ${interval}!`);\r\n            return () => {};\r\n        }\r\n\r\n        let state: TimerState;\r\n        let sizeWas = this.subscriptions.size;\r\n\r\n        if (!this.subscriptions.has(interval)) {\r\n            state = { subscribers: [] };\r\n            state.handle = setInterval(() => {\r\n                console.debug(`[Banta/TimerPool] Notifying ${state.subscribers.length} subs [${interval}ms]`);\r\n                state.subscribers.forEach(sub => sub());\r\n            }, interval);\r\n            this.subscriptions.set(interval, state);\r\n        } else {\r\n            state = this.subscriptions.get(interval);\r\n        }\r\n\r\n        state.subscribers.push(callback);\r\n\r\n        // Debug information //////////////////////////\r\n        //\r\n\r\n        if (!this.newSubscriptions.has(interval))\r\n            this.newSubscriptions.set(interval, 0);\r\n        this.newSubscriptions.set(interval, (this.newSubscriptions.get(interval) ?? 0) + 1);\r\n\r\n        clearTimeout(this.newSubscriptionsNotice);\r\n        this.newSubscriptionsNotice = setTimeout(() => {\r\n            for (let [interval, count] of this.newSubscriptions) {\r\n                console.debug(`[Banta/TimerPool] ${count} new subscriptions to ${interval}ms [${state.subscribers.length} total]`);\r\n            }\r\n\r\n            this.newSubscriptions.clear();\r\n        });\r\n\r\n        //\r\n        ///////////////////////////////////////////////\r\n\r\n        if (sizeWas === 0) {\r\n            console.debug(`[Banta/TimerPool] No longer idle.`);\r\n        }\r\n\r\n        // Unsubscribe function \r\n\r\n        return () => {\r\n            let state = this.subscriptions.get(interval);\r\n            let index = state.subscribers.indexOf(callback);\r\n            if (index >= 0)\r\n                state.subscribers.splice(index, 1);\r\n\r\n            if (state.subscribers.length === 0) {\r\n                clearInterval(state.handle);\r\n                this.subscriptions.delete(interval);\r\n            }\r\n\r\n            if (!this.removedSubscriptions.has(interval))\r\n                this.removedSubscriptions.set(interval, 0);\r\n            this.removedSubscriptions.set(interval, (this.removedSubscriptions.get(interval) ?? 0) + 1);\r\n\r\n            // Debug information ////////////////////////////////////////////////////////////////////\r\n\r\n            clearTimeout(this.removedSubscriptionsNotice);\r\n            this.removedSubscriptionsNotice = setTimeout(() => {\r\n                for (let [interval, count] of this.removedSubscriptions) {\r\n                    let state = this.subscriptions.get(interval);\r\n                    console.debug(`[Banta/TimerPool] ${count} unsubscribed from ${interval}ms [${state?.subscribers?.length ?? 0} remain]`);\r\n                }\r\n\r\n                if (this.subscriptions.size === 0)\r\n                    console.debug(`[Banta/TimerPool] All subscriptions have been removed. Now idle.`);\r\n\r\n                this.removedSubscriptions.clear();\r\n            });\r\n        };\r\n    }\r\n}"]}
|
|
@@ -1,21 +1,26 @@
|
|
|
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 "projects/sdk/src/lib/common/timer-pool.service";
|
|
4
|
+
import * as i2 from "@angular/common";
|
|
4
5
|
export class TimestampComponent {
|
|
5
|
-
constructor() {
|
|
6
|
+
constructor(timerPool) {
|
|
7
|
+
this.timerPool = timerPool;
|
|
6
8
|
this.relative = '';
|
|
7
9
|
this.tooltip = '';
|
|
8
|
-
this.
|
|
10
|
+
this.timerInterval = 0;
|
|
11
|
+
this._destroyed = false;
|
|
9
12
|
this.showAbsolute = false;
|
|
10
13
|
}
|
|
11
14
|
ngOnDestroy() {
|
|
12
|
-
|
|
13
|
-
|
|
15
|
+
this._destroyed = true;
|
|
16
|
+
this.timerUnsubscribe?.();
|
|
14
17
|
}
|
|
15
18
|
get value() {
|
|
16
19
|
return this._value;
|
|
17
20
|
}
|
|
18
21
|
update() {
|
|
22
|
+
if (this._destroyed)
|
|
23
|
+
return;
|
|
19
24
|
let now = Date.now();
|
|
20
25
|
let diff = now - this.value;
|
|
21
26
|
let minute = 1000 * 60;
|
|
@@ -77,18 +82,23 @@ export class TimestampComponent {
|
|
|
77
82
|
updateTime = 1000 * 30;
|
|
78
83
|
}
|
|
79
84
|
if (typeof window !== 'undefined') {
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
this.
|
|
85
|
+
if (this.timerInterval !== updateTime) {
|
|
86
|
+
this.timerInterval = updateTime;
|
|
87
|
+
this.timerUnsubscribe?.();
|
|
88
|
+
if (updateTime > 0) {
|
|
89
|
+
this.timerUnsubscribe = this.timerPool.addTimer(updateTime, () => this.update());
|
|
90
|
+
}
|
|
83
91
|
}
|
|
84
92
|
}
|
|
85
93
|
}
|
|
86
94
|
set value(v) {
|
|
87
|
-
this._value
|
|
88
|
-
|
|
95
|
+
if (this._value !== v) {
|
|
96
|
+
this._value = v;
|
|
97
|
+
this.update();
|
|
98
|
+
}
|
|
89
99
|
}
|
|
90
100
|
}
|
|
91
|
-
TimestampComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TimestampComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
101
|
+
TimestampComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TimestampComponent, deps: [{ token: i1.TimerPool }], target: i0.ɵɵFactoryTarget.Component });
|
|
92
102
|
TimestampComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: TimestampComponent, selector: "banta-timestamp", inputs: { value: "value" }, ngImport: i0, template: `
|
|
93
103
|
<span *ngIf="showAbsolute" [title]="value | date : 'short'">
|
|
94
104
|
{{value | date : 'shortDate'}}
|
|
@@ -96,7 +106,7 @@ TimestampComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", ver
|
|
|
96
106
|
<span *ngIf="!showAbsolute" [title]="value | date : 'short'">
|
|
97
107
|
{{relative}}
|
|
98
108
|
</span>
|
|
99
|
-
`, isInline: true, styles: [""], dependencies: [{ kind: "directive", type:
|
|
109
|
+
`, isInline: true, styles: [""], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i2.DatePipe, name: "date" }] });
|
|
100
110
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: TimestampComponent, decorators: [{
|
|
101
111
|
type: Component,
|
|
102
112
|
args: [{ selector: 'banta-timestamp', template: `
|
|
@@ -107,7 +117,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
107
117
|
{{relative}}
|
|
108
118
|
</span>
|
|
109
119
|
` }]
|
|
110
|
-
}], propDecorators: { value: [{
|
|
120
|
+
}], ctorParameters: function () { return [{ type: i1.TimerPool }]; }, propDecorators: { value: [{
|
|
111
121
|
type: Input
|
|
112
122
|
}] } });
|
|
113
|
-
//# 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":"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","sourcesContent":["import { Component, Input } from \"@angular/core\";\r\nimport { TimerPool } from \"projects/sdk/src/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}"]}
|