ngx-lift 1.7.2 → 1.7.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/pipes/array-join.pipe.mjs +3 -3
- package/esm2022/lib/pipes/byte-converter.pipe.mjs +3 -3
- package/esm2022/lib/pipes/is-https.pipe.mjs +3 -3
- package/esm2022/lib/pipes/mask.pipe.mjs +3 -3
- package/esm2022/lib/pipes/range.pipe.mjs +3 -3
- package/esm2022/lib/utils/idle-detection/idle-detection.config.mjs +7 -0
- package/esm2022/lib/utils/idle-detection/idle-detection.module.mjs +25 -0
- package/esm2022/lib/utils/idle-detection/idle-detection.service.mjs +187 -0
- package/esm2022/lib/utils/idle-detection/index.mjs +4 -0
- package/esm2022/lib/utils/index.mjs +2 -1
- package/fesm2022/ngx-lift.mjs +229 -18
- package/fesm2022/ngx-lift.mjs.map +1 -1
- package/lib/utils/idle-detection/idle-detection.config.d.ts +5 -0
- package/lib/utils/idle-detection/idle-detection.module.d.ts +13 -0
- package/lib/utils/idle-detection/idle-detection.service.d.ts +116 -0
- package/lib/utils/idle-detection/index.d.ts +3 -0
- package/lib/utils/index.d.ts +1 -0
- package/package.json +1 -1
|
@@ -8,10 +8,10 @@ export class ArrayJoinPipe {
|
|
|
8
8
|
// For non-array cases or unexpected types, return the value as is
|
|
9
9
|
return value;
|
|
10
10
|
}
|
|
11
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.
|
|
12
|
-
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.
|
|
11
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ArrayJoinPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
12
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.7", ngImport: i0, type: ArrayJoinPipe, isStandalone: true, name: "arrayJoin" }); }
|
|
13
13
|
}
|
|
14
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.
|
|
14
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ArrayJoinPipe, decorators: [{
|
|
15
15
|
type: Pipe,
|
|
16
16
|
args: [{
|
|
17
17
|
name: 'arrayJoin',
|
|
@@ -50,10 +50,10 @@ export class ByteConverterPipe {
|
|
|
50
50
|
formatNumber(value) {
|
|
51
51
|
return new Intl.NumberFormat(this.locale, { maximumFractionDigits: 2 }).format(value);
|
|
52
52
|
}
|
|
53
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.
|
|
54
|
-
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.
|
|
53
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ByteConverterPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
54
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.7", ngImport: i0, type: ByteConverterPipe, isStandalone: true, name: "byteConverter" }); }
|
|
55
55
|
}
|
|
56
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.
|
|
56
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ByteConverterPipe, decorators: [{
|
|
57
57
|
type: Pipe,
|
|
58
58
|
args: [{
|
|
59
59
|
name: 'byteConverter',
|
|
@@ -5,10 +5,10 @@ export class IsHttpsPipe {
|
|
|
5
5
|
transform(value) {
|
|
6
6
|
return isHttps(value);
|
|
7
7
|
}
|
|
8
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.
|
|
9
|
-
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.
|
|
8
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IsHttpsPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
9
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.7", ngImport: i0, type: IsHttpsPipe, isStandalone: true, name: "isHttps" }); }
|
|
10
10
|
}
|
|
11
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.
|
|
11
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IsHttpsPipe, decorators: [{
|
|
12
12
|
type: Pipe,
|
|
13
13
|
args: [{
|
|
14
14
|
name: 'isHttps',
|
|
@@ -23,10 +23,10 @@ export class MaskPipe {
|
|
|
23
23
|
.map((char, i) => (i < unmaskedPrefixLength || i > value.length - unmaskedSuffixLength - 1 ? char : maskChar))
|
|
24
24
|
.join('');
|
|
25
25
|
}
|
|
26
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.
|
|
27
|
-
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.
|
|
26
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: MaskPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
27
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.7", ngImport: i0, type: MaskPipe, isStandalone: true, name: "mask" }); }
|
|
28
28
|
}
|
|
29
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.
|
|
29
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: MaskPipe, decorators: [{
|
|
30
30
|
type: Pipe,
|
|
31
31
|
args: [{
|
|
32
32
|
name: 'mask',
|
|
@@ -6,10 +6,10 @@ export class RangePipe {
|
|
|
6
6
|
const input = value;
|
|
7
7
|
return range(...input);
|
|
8
8
|
}
|
|
9
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.
|
|
10
|
-
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.
|
|
9
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RangePipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
10
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.7", ngImport: i0, type: RangePipe, isStandalone: true, name: "range" }); }
|
|
11
11
|
}
|
|
12
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.
|
|
12
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: RangePipe, decorators: [{
|
|
13
13
|
type: Pipe,
|
|
14
14
|
args: [{
|
|
15
15
|
name: 'range',
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { makeEnvironmentProviders } from '@angular/core';
|
|
2
|
+
export class IdleDetectionConfig {
|
|
3
|
+
}
|
|
4
|
+
export function provideIdleDetectionConfig(config) {
|
|
5
|
+
return makeEnvironmentProviders([{ provide: IdleDetectionConfig, useValue: config }]);
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWRsZS1kZXRlY3Rpb24uY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LWxpZnQvc3JjL2xpYi91dGlscy9pZGxlLWRldGVjdGlvbi9pZGxlLWRldGVjdGlvbi5jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLHdCQUF3QixFQUFDLE1BQU0sZUFBZSxDQUFDO0FBRXZELE1BQU0sT0FBTyxtQkFBbUI7Q0FHL0I7QUFFRCxNQUFNLFVBQVUsMEJBQTBCLENBQUMsTUFBMkI7SUFDcEUsT0FBTyx3QkFBd0IsQ0FBQyxDQUFDLEVBQUMsT0FBTyxFQUFFLG1CQUFtQixFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQUMsQ0FBQyxDQUFDLENBQUM7QUFDdEYsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7bWFrZUVudmlyb25tZW50UHJvdmlkZXJzfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuZXhwb3J0IGNsYXNzIElkbGVEZXRlY3Rpb25Db25maWcge1xuICBpZGxlRHVyYXRpb25JblNlY29uZHM/OiBudW1iZXI7XG4gIHRpbWVvdXREdXJhdGlvbkluU2Vjb25kcz86IG51bWJlcjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHByb3ZpZGVJZGxlRGV0ZWN0aW9uQ29uZmlnKGNvbmZpZzogSWRsZURldGVjdGlvbkNvbmZpZykge1xuICByZXR1cm4gbWFrZUVudmlyb25tZW50UHJvdmlkZXJzKFt7cHJvdmlkZTogSWRsZURldGVjdGlvbkNvbmZpZywgdXNlVmFsdWU6IGNvbmZpZ31dKTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { NgModule } from '@angular/core';
|
|
2
|
+
import { provideIdleDetectionConfig } from './idle-detection.config';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Idle detection module.
|
|
6
|
+
* @deprecated use provideIdleDetectionConfig(config: IdleDetectionConfig) instead
|
|
7
|
+
*/
|
|
8
|
+
export class IdleDetectionModule {
|
|
9
|
+
static forRoot(config) {
|
|
10
|
+
return {
|
|
11
|
+
ngModule: IdleDetectionModule,
|
|
12
|
+
providers: [provideIdleDetectionConfig(config)],
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
|
|
16
|
+
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionModule }); }
|
|
17
|
+
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionModule }); }
|
|
18
|
+
}
|
|
19
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionModule, decorators: [{
|
|
20
|
+
type: NgModule,
|
|
21
|
+
args: [{
|
|
22
|
+
imports: [],
|
|
23
|
+
}]
|
|
24
|
+
}] });
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWRsZS1kZXRlY3Rpb24ubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LWxpZnQvc3JjL2xpYi91dGlscy9pZGxlLWRldGVjdGlvbi9pZGxlLWRldGVjdGlvbi5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFzQixRQUFRLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFFNUQsT0FBTyxFQUFzQiwwQkFBMEIsRUFBQyxNQUFNLHlCQUF5QixDQUFDOztBQUV4Rjs7O0dBR0c7QUFJSCxNQUFNLE9BQU8sbUJBQW1CO0lBQzlCLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBMkI7UUFDeEMsT0FBTztZQUNMLFFBQVEsRUFBRSxtQkFBbUI7WUFDN0IsU0FBUyxFQUFFLENBQUMsMEJBQTBCLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDaEQsQ0FBQztJQUNKLENBQUM7OEdBTlUsbUJBQW1COytHQUFuQixtQkFBbUI7K0dBQW5CLG1CQUFtQjs7MkZBQW5CLG1CQUFtQjtrQkFIL0IsUUFBUTttQkFBQztvQkFDUixPQUFPLEVBQUUsRUFBRTtpQkFDWiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7TW9kdWxlV2l0aFByb3ZpZGVycywgTmdNb2R1bGV9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQge0lkbGVEZXRlY3Rpb25Db25maWcsIHByb3ZpZGVJZGxlRGV0ZWN0aW9uQ29uZmlnfSBmcm9tICcuL2lkbGUtZGV0ZWN0aW9uLmNvbmZpZyc7XG5cbi8qKlxuICogSWRsZSBkZXRlY3Rpb24gbW9kdWxlLlxuICogQGRlcHJlY2F0ZWQgdXNlIHByb3ZpZGVJZGxlRGV0ZWN0aW9uQ29uZmlnKGNvbmZpZzogSWRsZURldGVjdGlvbkNvbmZpZykgaW5zdGVhZFxuICovXG5ATmdNb2R1bGUoe1xuICBpbXBvcnRzOiBbXSxcbn0pXG5leHBvcnQgY2xhc3MgSWRsZURldGVjdGlvbk1vZHVsZSB7XG4gIHN0YXRpYyBmb3JSb290KGNvbmZpZzogSWRsZURldGVjdGlvbkNvbmZpZyk6IE1vZHVsZVdpdGhQcm92aWRlcnM8SWRsZURldGVjdGlvbk1vZHVsZT4ge1xuICAgIHJldHVybiB7XG4gICAgICBuZ01vZHVsZTogSWRsZURldGVjdGlvbk1vZHVsZSxcbiAgICAgIHByb3ZpZGVyczogW3Byb3ZpZGVJZGxlRGV0ZWN0aW9uQ29uZmlnKGNvbmZpZyldLFxuICAgIH07XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
import { Injectable, Optional } from '@angular/core';
|
|
2
|
+
import { fromEvent, merge, Subject, throttleTime } from 'rxjs';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "./idle-detection.config";
|
|
5
|
+
/**
|
|
6
|
+
* Service for detecting user idle time and implementing a countdown.
|
|
7
|
+
*/
|
|
8
|
+
export class IdleDetectionService {
|
|
9
|
+
/**
|
|
10
|
+
* Constructs the IdleDetectionService.
|
|
11
|
+
* @param config - Optional configuration for idle and timeout durations.
|
|
12
|
+
*/
|
|
13
|
+
constructor(config) {
|
|
14
|
+
/**
|
|
15
|
+
* The list of interruption events that will end the idle detection.
|
|
16
|
+
*/
|
|
17
|
+
this.interruptionEvents = [
|
|
18
|
+
'click',
|
|
19
|
+
'keydown',
|
|
20
|
+
'keypress',
|
|
21
|
+
'mousemove',
|
|
22
|
+
'mousedown',
|
|
23
|
+
'scroll',
|
|
24
|
+
'wheel',
|
|
25
|
+
'touchmove',
|
|
26
|
+
'pointermove',
|
|
27
|
+
'resize',
|
|
28
|
+
];
|
|
29
|
+
/**
|
|
30
|
+
* The default idle duration in seconds (19 minutes).
|
|
31
|
+
*/
|
|
32
|
+
this.idleDuration = 19 * 60;
|
|
33
|
+
/**
|
|
34
|
+
* The default timeout duration in seconds (1 minute).
|
|
35
|
+
*/
|
|
36
|
+
this.timeoutDuration = 60;
|
|
37
|
+
/**
|
|
38
|
+
* Flag to indicate if countdown is in progress.
|
|
39
|
+
*/
|
|
40
|
+
this.isCountingDown = false;
|
|
41
|
+
/**
|
|
42
|
+
* The current countdown value.
|
|
43
|
+
*/
|
|
44
|
+
this.countdown = this.timeoutDuration;
|
|
45
|
+
/**
|
|
46
|
+
* Subject to emit when idle period ends.
|
|
47
|
+
*/
|
|
48
|
+
this.idleEndSubject = new Subject();
|
|
49
|
+
/**
|
|
50
|
+
* Subject to emit the countdown value.
|
|
51
|
+
*/
|
|
52
|
+
this.countdownSubject = new Subject();
|
|
53
|
+
/**
|
|
54
|
+
* Subject to emit when countdown ends.
|
|
55
|
+
*/
|
|
56
|
+
this.countdownEndSubject = new Subject();
|
|
57
|
+
if (config) {
|
|
58
|
+
this.setConfig(config);
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Starts to watch for user inactivity.
|
|
63
|
+
*/
|
|
64
|
+
startWatching() {
|
|
65
|
+
this.setupInterruptionEvents();
|
|
66
|
+
this.startIdleTimer();
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Resets the idle timer when user activity is detected.
|
|
70
|
+
*/
|
|
71
|
+
resetTimer() {
|
|
72
|
+
this.startIdleTimer();
|
|
73
|
+
if (this.isCountingDown) {
|
|
74
|
+
this.stopCountdown();
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Sets up the interruption events that will end the idle detection.
|
|
79
|
+
* Listens to a set of events on the document (e.g. click, keydown, mousemove, etc.).
|
|
80
|
+
* When any of these events is triggered, the idle timer is reset.
|
|
81
|
+
* Uses `throttleTime` operator to only trigger the reset when the events are spaced
|
|
82
|
+
* out by at least 1000ms (1 second).
|
|
83
|
+
* @private
|
|
84
|
+
*/
|
|
85
|
+
setupInterruptionEvents() {
|
|
86
|
+
if (!this.interruptionSubscription) {
|
|
87
|
+
const throttledInterruptionEvents = this.interruptionEvents.map((eventName) => fromEvent(document, eventName).pipe(throttleTime(1000)));
|
|
88
|
+
this.interruptionSubscription = merge(...throttledInterruptionEvents).subscribe(() => this.resetTimer());
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Starts the idle timer.
|
|
93
|
+
* When the timer expires, it emits an event through onIdleEnd() and starts the countdown.
|
|
94
|
+
*/
|
|
95
|
+
startIdleTimer() {
|
|
96
|
+
clearTimeout(this.idleTimer);
|
|
97
|
+
this.idleTimer = window.setTimeout(() => {
|
|
98
|
+
// after idle period, user inactivity detected
|
|
99
|
+
this.idleEndSubject.next();
|
|
100
|
+
this.startCountdown();
|
|
101
|
+
}, this.idleDuration * 1000);
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Starts the countdown.
|
|
105
|
+
*/
|
|
106
|
+
startCountdown() {
|
|
107
|
+
this.isCountingDown = true;
|
|
108
|
+
this.countdownSubject.next(this.countdown);
|
|
109
|
+
this.countdownTimer = window.setInterval(() => {
|
|
110
|
+
this.countdown--;
|
|
111
|
+
this.countdownSubject.next(this.countdown);
|
|
112
|
+
if (this.countdown <= 0) {
|
|
113
|
+
this.stopCountdown();
|
|
114
|
+
this.countdownEndSubject.next();
|
|
115
|
+
}
|
|
116
|
+
}, 1000);
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Stops the countdown.
|
|
120
|
+
*/
|
|
121
|
+
stopCountdown() {
|
|
122
|
+
clearInterval(this.countdownTimer);
|
|
123
|
+
this.isCountingDown = false;
|
|
124
|
+
// reset countdown
|
|
125
|
+
this.countdown = this.timeoutDuration;
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Returns an observable that emits when the user has been idle for a long period.
|
|
129
|
+
* Developers can use this to perform actions like opening a dialog.
|
|
130
|
+
*
|
|
131
|
+
* user has been inactive for a long period (idleDuration), at this moment, idle detection phase ends, onIdleEnd event is emitted, and then enter countdown/timeout phase.
|
|
132
|
+
* During the countdown phase:
|
|
133
|
+
* - if user has any activity, countdown phase immediately ends and restart the idle detection phase.
|
|
134
|
+
* - else, countdownEnd event will be emitted when timeoutDuration is over.
|
|
135
|
+
* @returns {Observable<void>} - Observable for idle end event.
|
|
136
|
+
*/
|
|
137
|
+
onIdleEnd() {
|
|
138
|
+
return this.idleEndSubject.asObservable();
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Returns an observable that emits when the countdown ends.
|
|
142
|
+
* Usually means the user has been inactive for a very long time and should be logged out.
|
|
143
|
+
* @returns {Observable<void>} - Observable for countdown end event.
|
|
144
|
+
*/
|
|
145
|
+
onTimeoutEnd() {
|
|
146
|
+
return this.countdownEndSubject.asObservable();
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Returns an observable that emits the countdown value every second.
|
|
150
|
+
* @returns {Observable<number>} - Observable for countdown value.
|
|
151
|
+
*/
|
|
152
|
+
onCountDown() {
|
|
153
|
+
return this.countdownSubject.asObservable();
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Clears all timers when the component is destroyed.
|
|
157
|
+
*/
|
|
158
|
+
clearTimers() {
|
|
159
|
+
clearTimeout(this.idleTimer);
|
|
160
|
+
clearInterval(this.countdownTimer);
|
|
161
|
+
this.interruptionSubscription?.unsubscribe();
|
|
162
|
+
this.interruptionSubscription = undefined;
|
|
163
|
+
}
|
|
164
|
+
/**
|
|
165
|
+
* Sets the idle and timeout durations based on the provided configuration.
|
|
166
|
+
* @param config - Configuration object with idle and timeout durations.
|
|
167
|
+
*/
|
|
168
|
+
setConfig(config) {
|
|
169
|
+
if (config.idleDurationInSeconds) {
|
|
170
|
+
this.idleDuration = config.idleDurationInSeconds;
|
|
171
|
+
}
|
|
172
|
+
if (config.timeoutDurationInSeconds) {
|
|
173
|
+
this.countdown = this.timeoutDuration = config.timeoutDurationInSeconds;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionService, deps: [{ token: i1.IdleDetectionConfig, optional: true }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
177
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionService, providedIn: 'root' }); }
|
|
178
|
+
}
|
|
179
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: IdleDetectionService, decorators: [{
|
|
180
|
+
type: Injectable,
|
|
181
|
+
args: [{
|
|
182
|
+
providedIn: 'root',
|
|
183
|
+
}]
|
|
184
|
+
}], ctorParameters: () => [{ type: i1.IdleDetectionConfig, decorators: [{
|
|
185
|
+
type: Optional
|
|
186
|
+
}] }] });
|
|
187
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"idle-detection.service.js","sourceRoot":"","sources":["../../../../../../projects/ngx-lift/src/lib/utils/idle-detection/idle-detection.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {Injectable, Optional} from '@angular/core';\nimport {fromEvent, merge, Subject, Subscription, throttleTime} from 'rxjs';\n\nimport {IdleDetectionConfig} from './idle-detection.config';\n\n/**\n * Service for detecting user idle time and implementing a countdown.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class IdleDetectionService {\n  /**\n   * The list of interruption events that will end the idle detection.\n   */\n  private interruptionEvents = [\n    'click',\n    'keydown',\n    'keypress',\n    'mousemove',\n    'mousedown',\n    'scroll',\n    'wheel',\n    'touchmove',\n    'pointermove',\n    'resize',\n  ];\n  private interruptionSubscription?: Subscription;\n\n  /**\n   * The default idle duration in seconds (19 minutes).\n   */\n  private idleDuration = 19 * 60;\n\n  /**\n   * The default timeout duration in seconds (1 minute).\n   */\n  private timeoutDuration = 60;\n\n  /**\n   * Timer for idle detection.\n   */\n  private idleTimer?: number;\n\n  /**\n   * Timer for countdown.\n   */\n  private countdownTimer?: number;\n\n  /**\n   * Flag to indicate if countdown is in progress.\n   */\n  private isCountingDown = false;\n\n  /**\n   * The current countdown value.\n   */\n  private countdown = this.timeoutDuration;\n\n  /**\n   * Subject to emit when idle period ends.\n   */\n  private idleEndSubject = new Subject<void>();\n\n  /**\n   * Subject to emit the countdown value.\n   */\n  private countdownSubject = new Subject<number>();\n\n  /**\n   * Subject to emit when countdown ends.\n   */\n  private countdownEndSubject = new Subject<void>();\n\n  /**\n   * Constructs the IdleDetectionService.\n   * @param config - Optional configuration for idle and timeout durations.\n   */\n  constructor(@Optional() config: IdleDetectionConfig) {\n    if (config) {\n      this.setConfig(config);\n    }\n  }\n\n  /**\n   * Starts to watch for user inactivity.\n   */\n  startWatching() {\n    this.setupInterruptionEvents();\n    this.startIdleTimer();\n  }\n\n  /**\n   * Resets the idle timer when user activity is detected.\n   */\n  resetTimer() {\n    this.startIdleTimer();\n\n    if (this.isCountingDown) {\n      this.stopCountdown();\n    }\n  }\n\n  /**\n   * Sets up the interruption events that will end the idle detection.\n   * Listens to a set of events on the document (e.g. click, keydown, mousemove, etc.).\n   * When any of these events is triggered, the idle timer is reset.\n   * Uses `throttleTime` operator to only trigger the reset when the events are spaced\n   * out by at least 1000ms (1 second).\n   * @private\n   */\n  private setupInterruptionEvents() {\n    if (!this.interruptionSubscription) {\n      const throttledInterruptionEvents = this.interruptionEvents.map((eventName) =>\n        fromEvent(document, eventName).pipe(throttleTime(1000)),\n      );\n      this.interruptionSubscription = merge(...throttledInterruptionEvents).subscribe(() => this.resetTimer());\n    }\n  }\n\n  /**\n   * Starts the idle timer.\n   * When the timer expires, it emits an event through onIdleEnd() and starts the countdown.\n   */\n  private startIdleTimer() {\n    clearTimeout(this.idleTimer);\n\n    this.idleTimer = window.setTimeout(() => {\n      // after idle period, user inactivity detected\n      this.idleEndSubject.next();\n      this.startCountdown();\n    }, this.idleDuration * 1000);\n  }\n\n  /**\n   * Starts the countdown.\n   */\n  private startCountdown() {\n    this.isCountingDown = true;\n    this.countdownSubject.next(this.countdown);\n\n    this.countdownTimer = window.setInterval(() => {\n      this.countdown--;\n      this.countdownSubject.next(this.countdown);\n\n      if (this.countdown <= 0) {\n        this.stopCountdown();\n        this.countdownEndSubject.next();\n      }\n    }, 1000);\n  }\n\n  /**\n   * Stops the countdown.\n   */\n  private stopCountdown() {\n    clearInterval(this.countdownTimer);\n    this.isCountingDown = false;\n\n    // reset countdown\n    this.countdown = this.timeoutDuration;\n  }\n\n  /**\n   * Returns an observable that emits when the user has been idle for a long period.\n   * Developers can use this to perform actions like opening a dialog.\n   *\n   * user has been inactive for a long period (idleDuration), at this moment, idle detection phase ends, onIdleEnd event is emitted, and then enter countdown/timeout phase.\n   * During the countdown phase:\n   *  - if user has any activity, countdown phase immediately ends and restart the idle detection phase.\n   *  - else, countdownEnd event will be emitted when timeoutDuration is over.\n   * @returns {Observable<void>} - Observable for idle end event.\n   */\n  onIdleEnd() {\n    return this.idleEndSubject.asObservable();\n  }\n\n  /**\n   * Returns an observable that emits when the countdown ends.\n   * Usually means the user has been inactive for a very long time and should be logged out.\n   * @returns {Observable<void>} - Observable for countdown end event.\n   */\n  onTimeoutEnd() {\n    return this.countdownEndSubject.asObservable();\n  }\n\n  /**\n   * Returns an observable that emits the countdown value every second.\n   * @returns {Observable<number>} - Observable for countdown value.\n   */\n  onCountDown() {\n    return this.countdownSubject.asObservable();\n  }\n\n  /**\n   * Clears all timers when the component is destroyed.\n   */\n  clearTimers() {\n    clearTimeout(this.idleTimer);\n    clearInterval(this.countdownTimer);\n\n    this.interruptionSubscription?.unsubscribe();\n    this.interruptionSubscription = undefined;\n  }\n\n  /**\n   * Sets the idle and timeout durations based on the provided configuration.\n   * @param config - Configuration object with idle and timeout durations.\n   */\n  setConfig(config: IdleDetectionConfig) {\n    if (config.idleDurationInSeconds) {\n      this.idleDuration = config.idleDurationInSeconds;\n    }\n\n    if (config.timeoutDurationInSeconds) {\n      this.countdown = this.timeoutDuration = config.timeoutDurationInSeconds;\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
export * from './idle-detection.config';
|
|
2
|
+
export * from './idle-detection.module';
|
|
3
|
+
export * from './idle-detection.service';
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtbGlmdC9zcmMvbGliL3V0aWxzL2lkbGUtZGV0ZWN0aW9uL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMseUJBQXlCLENBQUM7QUFDeEMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLDBCQUEwQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9pZGxlLWRldGVjdGlvbi5jb25maWcnO1xuZXhwb3J0ICogZnJvbSAnLi9pZGxlLWRldGVjdGlvbi5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi9pZGxlLWRldGVjdGlvbi5zZXJ2aWNlJztcbiJdfQ==
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
export * from './difference-in-days.util';
|
|
2
2
|
export * from './form.util';
|
|
3
|
+
export * from './idle-detection';
|
|
3
4
|
export * from './is-empty.util';
|
|
4
5
|
export * from './is-equal.util';
|
|
5
6
|
export * from './omit-by.util';
|
|
6
7
|
export * from './pick-by.util';
|
|
7
8
|
export * from './range.util';
|
|
8
9
|
export * from './url.util';
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtbGlmdC9zcmMvbGliL3V0aWxzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxrQkFBa0IsQ0FBQztBQUNqQyxjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsaUJBQWlCLENBQUM7QUFDaEMsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsWUFBWSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9kaWZmZXJlbmNlLWluLWRheXMudXRpbCc7XG5leHBvcnQgKiBmcm9tICcuL2Zvcm0udXRpbCc7XG5leHBvcnQgKiBmcm9tICcuL2lkbGUtZGV0ZWN0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vaXMtZW1wdHkudXRpbCc7XG5leHBvcnQgKiBmcm9tICcuL2lzLWVxdWFsLnV0aWwnO1xuZXhwb3J0ICogZnJvbSAnLi9vbWl0LWJ5LnV0aWwnO1xuZXhwb3J0ICogZnJvbSAnLi9waWNrLWJ5LnV0aWwnO1xuZXhwb3J0ICogZnJvbSAnLi9yYW5nZS51dGlsJztcbmV4cG9ydCAqIGZyb20gJy4vdXJsLnV0aWwnO1xuIl19
|