ngx-auto-logout 1.0.0
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/README.md +247 -0
- package/dist/README.md +247 -0
- package/dist/bundles/ngx-auto-logout.umd.js +864 -0
- package/dist/bundles/ngx-auto-logout.umd.js.map +1 -0
- package/dist/bundles/ngx-auto-logout.umd.min.js +16 -0
- package/dist/bundles/ngx-auto-logout.umd.min.js.map +1 -0
- package/dist/esm2015/lib/auto-logout.component.js +351 -0
- package/dist/esm2015/lib/auto-logout.module.js +26 -0
- package/dist/esm2015/lib/auto-logout.service.js +384 -0
- package/dist/esm2015/lib/models/auto-logout-config.interface.js +10 -0
- package/dist/esm2015/lib/models/auto-logout-config.token.js +7 -0
- package/dist/esm2015/lib/provide-auto-logout.js +11 -0
- package/dist/esm2015/ngx-auto-logout.js +5 -0
- package/dist/esm2015/public-api.js +10 -0
- package/dist/esm5/lib/auto-logout.component.js +196 -0
- package/dist/esm5/lib/auto-logout.module.js +30 -0
- package/dist/esm5/lib/auto-logout.service.js +391 -0
- package/dist/esm5/lib/models/auto-logout-config.interface.js +10 -0
- package/dist/esm5/lib/models/auto-logout-config.token.js +7 -0
- package/dist/esm5/lib/provide-auto-logout.js +11 -0
- package/dist/esm5/ngx-auto-logout.js +5 -0
- package/dist/esm5/public-api.js +10 -0
- package/dist/fesm2015/ngx-auto-logout.js +787 -0
- package/dist/fesm2015/ngx-auto-logout.js.map +1 -0
- package/dist/fesm5/ngx-auto-logout.js +643 -0
- package/dist/fesm5/ngx-auto-logout.js.map +1 -0
- package/dist/lib/auto-logout.component.d.ts +81 -0
- package/dist/lib/auto-logout.module.d.ts +8 -0
- package/dist/lib/auto-logout.service.d.ts +108 -0
- package/dist/lib/models/auto-logout-config.interface.d.ts +46 -0
- package/dist/lib/models/auto-logout-config.token.d.ts +7 -0
- package/dist/lib/provide-auto-logout.d.ts +6 -0
- package/dist/ngx-auto-logout.d.ts +4 -0
- package/dist/ngx-auto-logout.metadata.json +1 -0
- package/dist/package.json +41 -0
- package/dist/public-api.d.ts +6 -0
- package/package.json +53 -0
|
@@ -0,0 +1,384 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { Injectable, NgZone } from '@angular/core';
|
|
3
|
+
import { BehaviorSubject, interval } from 'rxjs';
|
|
4
|
+
import { DEFAULT_AUTO_LOGOUT_CONFIG } from './models/auto-logout-config.interface';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* Auto logout service
|
|
8
|
+
* Angular 9 compatible: use providedIn: 'root' (official pattern)
|
|
9
|
+
*/
|
|
10
|
+
let AutoLogoutService = class AutoLogoutService {
|
|
11
|
+
/**
|
|
12
|
+
* Constructor
|
|
13
|
+
* Note: Do NOT use @Inject() here, it causes DI errors in Angular 9 VE
|
|
14
|
+
* Config should be passed via startMonitoring() method instead
|
|
15
|
+
*/
|
|
16
|
+
constructor(ngZone) {
|
|
17
|
+
this.ngZone = ngZone;
|
|
18
|
+
// 配置是否已初始化
|
|
19
|
+
this.isConfigInitialized = false;
|
|
20
|
+
// localStorage 键名
|
|
21
|
+
this.LOGIN_TIME_KEY = 'autoLogout_loginTime';
|
|
22
|
+
this.LAST_ACTIVITY_KEY = 'autoLogout_lastActivity';
|
|
23
|
+
this.EXTEND_COUNT_KEY = 'autoLogout_extendCount';
|
|
24
|
+
this.PAUSE_STATE_KEY = 'autoLogout_pauseState';
|
|
25
|
+
this.lastActivityTime = Date.now();
|
|
26
|
+
this.countdownValue = 0;
|
|
27
|
+
this.countdownSubject = new BehaviorSubject(0);
|
|
28
|
+
this.timerSubscription = null;
|
|
29
|
+
this.activityListeners = [];
|
|
30
|
+
this.warningShown = false;
|
|
31
|
+
// 是否启用活动监听
|
|
32
|
+
this.enableActivityTracking = false;
|
|
33
|
+
// 暂停状态
|
|
34
|
+
this.isPaused = false;
|
|
35
|
+
this.pauseStartTime = 0;
|
|
36
|
+
this.pausedRemainingTime = 0;
|
|
37
|
+
// Use default config, will be overridden by startMonitoring()
|
|
38
|
+
this.currentConfig = Object.assign({}, DEFAULT_AUTO_LOGOUT_CONFIG);
|
|
39
|
+
this.countdownValue = Math.floor(this.currentConfig.timeout);
|
|
40
|
+
this.countdownSubject = new BehaviorSubject(this.countdownValue);
|
|
41
|
+
}
|
|
42
|
+
getCountdown() {
|
|
43
|
+
return this.countdownSubject.asObservable();
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* 启动自动登出监控
|
|
47
|
+
*/
|
|
48
|
+
startMonitoring(config, enableTracking = false) {
|
|
49
|
+
this.stopMonitoring();
|
|
50
|
+
if (config) {
|
|
51
|
+
this.currentConfig = Object.assign(Object.assign({}, this.currentConfig), config);
|
|
52
|
+
}
|
|
53
|
+
// 恢复暂停状态
|
|
54
|
+
this.restorePauseState();
|
|
55
|
+
if (this.isPaused) {
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
// 根据模式初始化
|
|
59
|
+
if (this.currentConfig.mode === 'fixed') {
|
|
60
|
+
this.initFixedMode();
|
|
61
|
+
}
|
|
62
|
+
else {
|
|
63
|
+
this.initIdleMode(enableTracking);
|
|
64
|
+
}
|
|
65
|
+
this.warningShown = false;
|
|
66
|
+
// 启动定时器
|
|
67
|
+
this.ngZone.runOutsideAngular(() => {
|
|
68
|
+
this.timerSubscription = interval(1000).subscribe(() => {
|
|
69
|
+
this.updateCountdown();
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* 停止监控
|
|
75
|
+
*/
|
|
76
|
+
stopMonitoring() {
|
|
77
|
+
if (this.timerSubscription) {
|
|
78
|
+
this.timerSubscription.unsubscribe();
|
|
79
|
+
this.timerSubscription = null;
|
|
80
|
+
}
|
|
81
|
+
this.removeActivityListeners();
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* 清除登录时间
|
|
85
|
+
*/
|
|
86
|
+
clearLoginTime() {
|
|
87
|
+
localStorage.removeItem(this.LOGIN_TIME_KEY);
|
|
88
|
+
localStorage.removeItem(this.LAST_ACTIVITY_KEY);
|
|
89
|
+
this.resetExtendCount();
|
|
90
|
+
this.clearPauseState();
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* 延长会话
|
|
94
|
+
*/
|
|
95
|
+
extendSession(extendSeconds = 1800) {
|
|
96
|
+
if (this.isMaxExtendReached()) {
|
|
97
|
+
console.warn('已达到最大延长次数');
|
|
98
|
+
return false;
|
|
99
|
+
}
|
|
100
|
+
if (this.currentConfig.mode === 'fixed') {
|
|
101
|
+
this.lastActivityTime -= extendSeconds * 1000;
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
this.resetActivityTimer();
|
|
105
|
+
}
|
|
106
|
+
this.incrementExtendCount();
|
|
107
|
+
if (this.currentConfig.onExtended) {
|
|
108
|
+
this.currentConfig.onExtended(extendSeconds);
|
|
109
|
+
}
|
|
110
|
+
return true;
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* 检查是否可以延长
|
|
114
|
+
*/
|
|
115
|
+
canExtendSession() {
|
|
116
|
+
return !this.isMaxExtendReached() && !this.isPaused;
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* 获取剩余可延长次数
|
|
120
|
+
*/
|
|
121
|
+
getRemainingExtendTimes() {
|
|
122
|
+
if (this.currentConfig.maxExtendTimes === -1 || this.currentConfig.maxExtendTimes === undefined) {
|
|
123
|
+
return -1;
|
|
124
|
+
}
|
|
125
|
+
const usedCount = this.getExtendCount();
|
|
126
|
+
return Math.max(0, this.currentConfig.maxExtendTimes - usedCount);
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* 获取最后操作时间显示
|
|
130
|
+
*/
|
|
131
|
+
getLastActivityDisplay() {
|
|
132
|
+
if (this.currentConfig.mode !== 'idle') {
|
|
133
|
+
return '';
|
|
134
|
+
}
|
|
135
|
+
const lastActivity = this.getLastActivityTime();
|
|
136
|
+
if (!lastActivity) {
|
|
137
|
+
return '';
|
|
138
|
+
}
|
|
139
|
+
const minutesAgo = Math.floor((Date.now() - lastActivity) / 60000);
|
|
140
|
+
if (minutesAgo < 1) {
|
|
141
|
+
return '刚刚操作';
|
|
142
|
+
}
|
|
143
|
+
return `${minutesAgo}分钟前操作`;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* 暂停
|
|
147
|
+
*/
|
|
148
|
+
pause() {
|
|
149
|
+
if (this.isPaused)
|
|
150
|
+
return;
|
|
151
|
+
this.isPaused = true;
|
|
152
|
+
this.pauseStartTime = Date.now();
|
|
153
|
+
this.pausedRemainingTime = this.countdownValue;
|
|
154
|
+
this.stopMonitoring();
|
|
155
|
+
this.savePauseState();
|
|
156
|
+
if (this.currentConfig.onPaused) {
|
|
157
|
+
this.currentConfig.onPaused();
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* 恢复
|
|
162
|
+
*/
|
|
163
|
+
resume() {
|
|
164
|
+
if (!this.isPaused)
|
|
165
|
+
return;
|
|
166
|
+
const pauseDuration = Date.now() - this.pauseStartTime;
|
|
167
|
+
if (this.currentConfig.mode === 'fixed') {
|
|
168
|
+
this.lastActivityTime += pauseDuration;
|
|
169
|
+
}
|
|
170
|
+
else {
|
|
171
|
+
this.lastActivityTime += pauseDuration;
|
|
172
|
+
this.setLastActivityTime(this.lastActivityTime);
|
|
173
|
+
}
|
|
174
|
+
this.isPaused = false;
|
|
175
|
+
this.pauseStartTime = 0;
|
|
176
|
+
this.pausedRemainingTime = 0;
|
|
177
|
+
this.clearPauseState();
|
|
178
|
+
this.startMonitoring(this.currentConfig, this.enableActivityTracking);
|
|
179
|
+
if (this.currentConfig.onResumed) {
|
|
180
|
+
this.currentConfig.onResumed();
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* 检查是否暂停
|
|
185
|
+
*/
|
|
186
|
+
isAutoLogoutPaused() {
|
|
187
|
+
return this.isPaused;
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* 格式化倒计时
|
|
191
|
+
*/
|
|
192
|
+
formatCountdown(seconds) {
|
|
193
|
+
const mins = Math.floor(seconds / 60);
|
|
194
|
+
const secs = seconds % 60;
|
|
195
|
+
return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;
|
|
196
|
+
}
|
|
197
|
+
/**
|
|
198
|
+
* 判断是否警告状态
|
|
199
|
+
*/
|
|
200
|
+
isWarningState(seconds) {
|
|
201
|
+
const warningTime = this.currentConfig.warningTime || 30;
|
|
202
|
+
return seconds <= warningTime && seconds > 0;
|
|
203
|
+
}
|
|
204
|
+
/**
|
|
205
|
+
* 判断是否紧急状态(最后10秒)
|
|
206
|
+
*/
|
|
207
|
+
isUrgentState(seconds) {
|
|
208
|
+
return seconds <= 10 && seconds > 0;
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* 获取当前模式
|
|
212
|
+
*/
|
|
213
|
+
getCurrentMode() {
|
|
214
|
+
return this.currentConfig.mode;
|
|
215
|
+
}
|
|
216
|
+
// ==================== 私有方法 ====================
|
|
217
|
+
initFixedMode() {
|
|
218
|
+
const loginTime = this.getLoginTime();
|
|
219
|
+
if (!loginTime) {
|
|
220
|
+
this.setLoginTime(Date.now());
|
|
221
|
+
this.lastActivityTime = Date.now();
|
|
222
|
+
}
|
|
223
|
+
else {
|
|
224
|
+
this.lastActivityTime = loginTime;
|
|
225
|
+
}
|
|
226
|
+
this.updateCountdownValue();
|
|
227
|
+
}
|
|
228
|
+
initIdleMode(enableTracking) {
|
|
229
|
+
this.enableActivityTracking = enableTracking || true;
|
|
230
|
+
const lastActivity = this.getLastActivityTime();
|
|
231
|
+
if (!lastActivity) {
|
|
232
|
+
this.setLastActivityTime(Date.now());
|
|
233
|
+
this.lastActivityTime = Date.now();
|
|
234
|
+
}
|
|
235
|
+
else {
|
|
236
|
+
this.lastActivityTime = lastActivity;
|
|
237
|
+
}
|
|
238
|
+
this.registerActivityListeners();
|
|
239
|
+
this.updateCountdownValue();
|
|
240
|
+
}
|
|
241
|
+
updateCountdownValue() {
|
|
242
|
+
const now = Date.now();
|
|
243
|
+
const elapsed = now - this.lastActivityTime;
|
|
244
|
+
const remaining = Math.max(0, Math.floor((this.currentConfig.timeout * 1000 - elapsed) / 1000));
|
|
245
|
+
this.countdownValue = remaining;
|
|
246
|
+
this.countdownSubject.next(this.countdownValue);
|
|
247
|
+
}
|
|
248
|
+
updateCountdown() {
|
|
249
|
+
const now = Date.now();
|
|
250
|
+
const elapsed = now - this.lastActivityTime;
|
|
251
|
+
const remaining = Math.max(0, Math.floor((this.currentConfig.timeout * 1000 - elapsed) / 1000));
|
|
252
|
+
if (remaining !== this.countdownValue) {
|
|
253
|
+
this.countdownValue = remaining;
|
|
254
|
+
this.ngZone.run(() => {
|
|
255
|
+
this.countdownSubject.next(this.countdownValue);
|
|
256
|
+
});
|
|
257
|
+
// 警告回调
|
|
258
|
+
const warningTime = this.currentConfig.warningTime || 30;
|
|
259
|
+
if (remaining === warningTime && !this.warningShown) {
|
|
260
|
+
this.warningShown = true;
|
|
261
|
+
if (this.currentConfig.onWarning) {
|
|
262
|
+
this.currentConfig.onWarning(remaining);
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
// 超时回调
|
|
266
|
+
if (remaining <= 0) {
|
|
267
|
+
this.performAutoLogout();
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
performAutoLogout() {
|
|
272
|
+
this.stopMonitoring();
|
|
273
|
+
this.clearLoginTime();
|
|
274
|
+
if (this.currentConfig.onTimeout) {
|
|
275
|
+
this.currentConfig.onTimeout();
|
|
276
|
+
}
|
|
277
|
+
else {
|
|
278
|
+
// 默认行为:导航到登录页
|
|
279
|
+
// Note: Router navigation removed to avoid DI issues in Angular 9 VE
|
|
280
|
+
// Use onTimeout callback for custom navigation
|
|
281
|
+
console.warn('Auto logout timeout - implement custom navigation via onTimeout callback');
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
registerActivityListeners() {
|
|
285
|
+
const events = ['mousedown', 'mousemove', 'keypress', 'scroll', 'touchstart', 'click'];
|
|
286
|
+
events.forEach((event) => {
|
|
287
|
+
const listener = () => this.resetActivityTimer();
|
|
288
|
+
document.addEventListener(event, listener, true);
|
|
289
|
+
this.activityListeners.push({ event, listener });
|
|
290
|
+
});
|
|
291
|
+
}
|
|
292
|
+
removeActivityListeners() {
|
|
293
|
+
this.activityListeners.forEach(({ event, listener }) => {
|
|
294
|
+
document.removeEventListener(event, listener, true);
|
|
295
|
+
});
|
|
296
|
+
this.activityListeners = [];
|
|
297
|
+
}
|
|
298
|
+
resetActivityTimer() {
|
|
299
|
+
if (this.currentConfig.mode === 'idle') {
|
|
300
|
+
this.lastActivityTime = Date.now();
|
|
301
|
+
this.setLastActivityTime(this.lastActivityTime);
|
|
302
|
+
this.updateCountdownValue();
|
|
303
|
+
this.warningShown = false;
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
isMaxExtendReached() {
|
|
307
|
+
if (this.currentConfig.maxExtendTimes === -1 || this.currentConfig.maxExtendTimes === undefined) {
|
|
308
|
+
return false;
|
|
309
|
+
}
|
|
310
|
+
const usedCount = this.getExtendCount();
|
|
311
|
+
return usedCount >= this.currentConfig.maxExtendTimes;
|
|
312
|
+
}
|
|
313
|
+
getExtendCount() {
|
|
314
|
+
const countStr = localStorage.getItem(this.EXTEND_COUNT_KEY);
|
|
315
|
+
return countStr ? parseInt(countStr, 10) : 0;
|
|
316
|
+
}
|
|
317
|
+
incrementExtendCount() {
|
|
318
|
+
const currentCount = this.getExtendCount();
|
|
319
|
+
localStorage.setItem(this.EXTEND_COUNT_KEY, (currentCount + 1).toString());
|
|
320
|
+
}
|
|
321
|
+
resetExtendCount() {
|
|
322
|
+
localStorage.removeItem(this.EXTEND_COUNT_KEY);
|
|
323
|
+
}
|
|
324
|
+
getLoginTime() {
|
|
325
|
+
const timeStr = localStorage.getItem(this.LOGIN_TIME_KEY);
|
|
326
|
+
return timeStr ? parseInt(timeStr, 10) : null;
|
|
327
|
+
}
|
|
328
|
+
setLoginTime(time) {
|
|
329
|
+
localStorage.setItem(this.LOGIN_TIME_KEY, time.toString());
|
|
330
|
+
}
|
|
331
|
+
getLastActivityTime() {
|
|
332
|
+
const timeStr = localStorage.getItem(this.LAST_ACTIVITY_KEY);
|
|
333
|
+
return timeStr ? parseInt(timeStr, 10) : null;
|
|
334
|
+
}
|
|
335
|
+
setLastActivityTime(time) {
|
|
336
|
+
localStorage.setItem(this.LAST_ACTIVITY_KEY, time.toString());
|
|
337
|
+
}
|
|
338
|
+
savePauseState() {
|
|
339
|
+
const state = {
|
|
340
|
+
isPaused: true,
|
|
341
|
+
pauseStartTime: this.pauseStartTime,
|
|
342
|
+
pausedRemainingTime: this.pausedRemainingTime,
|
|
343
|
+
mode: this.currentConfig.mode,
|
|
344
|
+
lastActivityTime: this.lastActivityTime
|
|
345
|
+
};
|
|
346
|
+
localStorage.setItem(this.PAUSE_STATE_KEY, JSON.stringify(state));
|
|
347
|
+
}
|
|
348
|
+
clearPauseState() {
|
|
349
|
+
localStorage.removeItem(this.PAUSE_STATE_KEY);
|
|
350
|
+
}
|
|
351
|
+
restorePauseState() {
|
|
352
|
+
const stateStr = localStorage.getItem(this.PAUSE_STATE_KEY);
|
|
353
|
+
if (!stateStr)
|
|
354
|
+
return;
|
|
355
|
+
try {
|
|
356
|
+
const state = JSON.parse(stateStr);
|
|
357
|
+
if (state.isPaused) {
|
|
358
|
+
this.isPaused = true;
|
|
359
|
+
this.pauseStartTime = state.pauseStartTime;
|
|
360
|
+
this.pausedRemainingTime = state.pausedRemainingTime;
|
|
361
|
+
this.lastActivityTime = state.lastActivityTime;
|
|
362
|
+
this.currentConfig.mode = state.mode;
|
|
363
|
+
if (this.timerSubscription) {
|
|
364
|
+
this.timerSubscription.unsubscribe();
|
|
365
|
+
this.timerSubscription = null;
|
|
366
|
+
}
|
|
367
|
+
this.removeActivityListeners();
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
catch (e) {
|
|
371
|
+
console.error('恢复暂停状态失败:', e);
|
|
372
|
+
this.clearPauseState();
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
};
|
|
376
|
+
AutoLogoutService.ctorParameters = () => [
|
|
377
|
+
{ type: NgZone }
|
|
378
|
+
];
|
|
379
|
+
AutoLogoutService.ɵprov = i0.ɵɵdefineInjectable({ factory: function AutoLogoutService_Factory() { return new AutoLogoutService(i0.ɵɵinject(i0.NgZone)); }, token: AutoLogoutService, providedIn: "root" });
|
|
380
|
+
AutoLogoutService = __decorate([
|
|
381
|
+
Injectable({ providedIn: 'root' })
|
|
382
|
+
], AutoLogoutService);
|
|
383
|
+
export { AutoLogoutService };
|
|
384
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auto-logout.service.js","sourceRoot":"ng://ngx-auto-logout/","sources":["lib/auto-logout.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { BehaviorSubject, Observable, interval, Subscription } from 'rxjs';\nimport { AutoLogoutConfig, DEFAULT_AUTO_LOGOUT_CONFIG } from './models/auto-logout-config.interface';\n\n/**\n * Auto logout service\n * Angular 9 compatible: use providedIn: 'root' (official pattern)\n */\n@Injectable({ providedIn: 'root' })\nexport class AutoLogoutService {\n  // 当前配置\n  private currentConfig: AutoLogoutConfig;\n\n  // 配置是否已初始化\n  private isConfigInitialized = false;\n\n  // localStorage 键名\n  private readonly LOGIN_TIME_KEY = 'autoLogout_loginTime';\n  private readonly LAST_ACTIVITY_KEY = 'autoLogout_lastActivity';\n  private readonly EXTEND_COUNT_KEY = 'autoLogout_extendCount';\n  private readonly PAUSE_STATE_KEY = 'autoLogout_pauseState';\n\n  private lastActivityTime: number = Date.now();\n  private countdownValue: number = 0;\n  private countdownSubject: BehaviorSubject<number> = new BehaviorSubject<number>(0);\n  private timerSubscription: Subscription | null = null;\n  private activityListeners: any[] = [];\n  private warningShown: boolean = false;\n\n  // 是否启用活动监听\n  private enableActivityTracking: boolean = false;\n\n  // 暂停状态\n  private isPaused: boolean = false;\n  private pauseStartTime: number = 0;\n  private pausedRemainingTime: number = 0;\n\n  /**\n   * Constructor\n   * Note: Do NOT use @Inject() here, it causes DI errors in Angular 9 VE\n   * Config should be passed via startMonitoring() method instead\n   */\n  constructor(private ngZone: NgZone) {\n    // Use default config, will be overridden by startMonitoring()\n    this.currentConfig = { ...DEFAULT_AUTO_LOGOUT_CONFIG };\n    this.countdownValue = Math.floor(this.currentConfig.timeout);\n    this.countdownSubject = new BehaviorSubject<number>(this.countdownValue);\n  }\n  getCountdown(): Observable<number> {\n    return this.countdownSubject.asObservable();\n  }\n\n  /**\n   * 启动自动登出监控\n   */\n  startMonitoring(config?: Partial<AutoLogoutConfig>, enableTracking: boolean = false): void {\n    this.stopMonitoring();\n\n    if (config) {\n      this.currentConfig = { ...this.currentConfig, ...config };\n    }\n\n    // 恢复暂停状态\n    this.restorePauseState();\n\n    if (this.isPaused) {\n      return;\n    }\n\n    // 根据模式初始化\n    if (this.currentConfig.mode === 'fixed') {\n      this.initFixedMode();\n    } else {\n      this.initIdleMode(enableTracking);\n    }\n\n    this.warningShown = false;\n\n    // 启动定时器\n    this.ngZone.runOutsideAngular(() => {\n      this.timerSubscription = interval(1000).subscribe(() => {\n        this.updateCountdown();\n      });\n    });\n  }\n\n  /**\n   * 停止监控\n   */\n  stopMonitoring(): void {\n    if (this.timerSubscription) {\n      this.timerSubscription.unsubscribe();\n      this.timerSubscription = null;\n    }\n    this.removeActivityListeners();\n  }\n\n  /**\n   * 清除登录时间\n   */\n  clearLoginTime(): void {\n    localStorage.removeItem(this.LOGIN_TIME_KEY);\n    localStorage.removeItem(this.LAST_ACTIVITY_KEY);\n    this.resetExtendCount();\n    this.clearPauseState();\n  }\n\n  /**\n   * 延长会话\n   */\n  extendSession(extendSeconds: number = 1800): boolean {\n    if (this.isMaxExtendReached()) {\n      console.warn('已达到最大延长次数');\n      return false;\n    }\n\n    if (this.currentConfig.mode === 'fixed') {\n      this.lastActivityTime -= extendSeconds * 1000;\n    } else {\n      this.resetActivityTimer();\n    }\n\n    this.incrementExtendCount();\n\n    if (this.currentConfig.onExtended) {\n      this.currentConfig.onExtended(extendSeconds);\n    }\n\n    return true;\n  }\n\n  /**\n   * 检查是否可以延长\n   */\n  canExtendSession(): boolean {\n    return !this.isMaxExtendReached() && !this.isPaused;\n  }\n\n  /**\n   * 获取剩余可延长次数\n   */\n  getRemainingExtendTimes(): number {\n    if (this.currentConfig.maxExtendTimes === -1 || this.currentConfig.maxExtendTimes === undefined) {\n      return -1;\n    }\n    const usedCount = this.getExtendCount();\n    return Math.max(0, this.currentConfig.maxExtendTimes - usedCount);\n  }\n\n  /**\n   * 获取最后操作时间显示\n   */\n  getLastActivityDisplay(): string {\n    if (this.currentConfig.mode !== 'idle') {\n      return '';\n    }\n\n    const lastActivity = this.getLastActivityTime();\n    if (!lastActivity) {\n      return '';\n    }\n\n    const minutesAgo = Math.floor((Date.now() - lastActivity) / 60000);\n    if (minutesAgo < 1) {\n      return '刚刚操作';\n    }\n    return `${minutesAgo}分钟前操作`;\n  }\n\n  /**\n   * 暂停\n   */\n  pause(): void {\n    if (this.isPaused) return;\n\n    this.isPaused = true;\n    this.pauseStartTime = Date.now();\n    this.pausedRemainingTime = this.countdownValue;\n\n    this.stopMonitoring();\n    this.savePauseState();\n\n    if (this.currentConfig.onPaused) {\n      this.currentConfig.onPaused();\n    }\n  }\n\n  /**\n   * 恢复\n   */\n  resume(): void {\n    if (!this.isPaused) return;\n\n    const pauseDuration = Date.now() - this.pauseStartTime;\n\n    if (this.currentConfig.mode === 'fixed') {\n      this.lastActivityTime += pauseDuration;\n    } else {\n      this.lastActivityTime += pauseDuration;\n      this.setLastActivityTime(this.lastActivityTime);\n    }\n\n    this.isPaused = false;\n    this.pauseStartTime = 0;\n    this.pausedRemainingTime = 0;\n\n    this.clearPauseState();\n    this.startMonitoring(this.currentConfig, this.enableActivityTracking);\n\n    if (this.currentConfig.onResumed) {\n      this.currentConfig.onResumed();\n    }\n  }\n\n  /**\n   * 检查是否暂停\n   */\n  isAutoLogoutPaused(): boolean {\n    return this.isPaused;\n  }\n\n  /**\n   * 格式化倒计时\n   */\n  formatCountdown(seconds: number): string {\n    const mins = Math.floor(seconds / 60);\n    const secs = seconds % 60;\n    return `${mins.toString().padStart(2, '0')}:${secs.toString().padStart(2, '0')}`;\n  }\n\n  /**\n   * 判断是否警告状态\n   */\n  isWarningState(seconds: number): boolean {\n    const warningTime = this.currentConfig.warningTime || 30;\n    return seconds <= warningTime && seconds > 0;\n  }\n\n  /**\n   * 判断是否紧急状态（最后10秒）\n   */\n  isUrgentState(seconds: number): boolean {\n    return seconds <= 10 && seconds > 0;\n  }\n\n  /**\n   * 获取当前模式\n   */\n  getCurrentMode(): 'fixed' | 'idle' {\n    return this.currentConfig.mode;\n  }\n\n  // ==================== 私有方法 ====================\n\n  private initFixedMode(): void {\n    const loginTime = this.getLoginTime();\n    if (!loginTime) {\n      this.setLoginTime(Date.now());\n      this.lastActivityTime = Date.now();\n    } else {\n      this.lastActivityTime = loginTime;\n    }\n    this.updateCountdownValue();\n  }\n\n  private initIdleMode(enableTracking: boolean): void {\n    this.enableActivityTracking = enableTracking || true;\n\n    const lastActivity = this.getLastActivityTime();\n    if (!lastActivity) {\n      this.setLastActivityTime(Date.now());\n      this.lastActivityTime = Date.now();\n    } else {\n      this.lastActivityTime = lastActivity;\n    }\n\n    this.registerActivityListeners();\n    this.updateCountdownValue();\n  }\n\n  private updateCountdownValue(): void {\n    const now = Date.now();\n    const elapsed = now - this.lastActivityTime;\n    const remaining = Math.max(0, Math.floor((this.currentConfig.timeout * 1000 - elapsed) / 1000));\n    this.countdownValue = remaining;\n    this.countdownSubject.next(this.countdownValue);\n  }\n\n  private updateCountdown(): void {\n    const now = Date.now();\n    const elapsed = now - this.lastActivityTime;\n    const remaining = Math.max(0, Math.floor((this.currentConfig.timeout * 1000 - elapsed) / 1000));\n\n    if (remaining !== this.countdownValue) {\n      this.countdownValue = remaining;\n      this.ngZone.run(() => {\n        this.countdownSubject.next(this.countdownValue);\n      });\n\n      // 警告回调\n      const warningTime = this.currentConfig.warningTime || 30;\n      if (remaining === warningTime && !this.warningShown) {\n        this.warningShown = true;\n        if (this.currentConfig.onWarning) {\n          this.currentConfig.onWarning(remaining);\n        }\n      }\n\n      // 超时回调\n      if (remaining <= 0) {\n        this.performAutoLogout();\n      }\n    }\n  }\n\n  private performAutoLogout(): void {\n    this.stopMonitoring();\n    this.clearLoginTime();\n\n    if (this.currentConfig.onTimeout) {\n      this.currentConfig.onTimeout();\n    } else {\n      // 默认行为：导航到登录页\n      // Note: Router navigation removed to avoid DI issues in Angular 9 VE\n      // Use onTimeout callback for custom navigation\n      console.warn('Auto logout timeout - implement custom navigation via onTimeout callback');\n    }\n  }\n\n  private registerActivityListeners(): void {\n    const events = ['mousedown', 'mousemove', 'keypress', 'scroll', 'touchstart', 'click'];\n\n    events.forEach((event) => {\n      const listener = () => this.resetActivityTimer();\n      document.addEventListener(event, listener, true);\n      this.activityListeners.push({ event, listener });\n    });\n  }\n\n  private removeActivityListeners(): void {\n    this.activityListeners.forEach(({ event, listener }) => {\n      document.removeEventListener(event, listener, true);\n    });\n    this.activityListeners = [];\n  }\n\n  private resetActivityTimer(): void {\n    if (this.currentConfig.mode === 'idle') {\n      this.lastActivityTime = Date.now();\n      this.setLastActivityTime(this.lastActivityTime);\n      this.updateCountdownValue();\n      this.warningShown = false;\n    }\n  }\n\n  private isMaxExtendReached(): boolean {\n    if (this.currentConfig.maxExtendTimes === -1 || this.currentConfig.maxExtendTimes === undefined) {\n      return false;\n    }\n    const usedCount = this.getExtendCount();\n    return usedCount >= this.currentConfig.maxExtendTimes;\n  }\n\n  private getExtendCount(): number {\n    const countStr = localStorage.getItem(this.EXTEND_COUNT_KEY);\n    return countStr ? parseInt(countStr, 10) : 0;\n  }\n\n  private incrementExtendCount(): void {\n    const currentCount = this.getExtendCount();\n    localStorage.setItem(this.EXTEND_COUNT_KEY, (currentCount + 1).toString());\n  }\n\n  private resetExtendCount(): void {\n    localStorage.removeItem(this.EXTEND_COUNT_KEY);\n  }\n\n  private getLoginTime(): number | null {\n    const timeStr = localStorage.getItem(this.LOGIN_TIME_KEY);\n    return timeStr ? parseInt(timeStr, 10) : null;\n  }\n\n  private setLoginTime(time: number): void {\n    localStorage.setItem(this.LOGIN_TIME_KEY, time.toString());\n  }\n\n  private getLastActivityTime(): number | null {\n    const timeStr = localStorage.getItem(this.LAST_ACTIVITY_KEY);\n    return timeStr ? parseInt(timeStr, 10) : null;\n  }\n\n  private setLastActivityTime(time: number): void {\n    localStorage.setItem(this.LAST_ACTIVITY_KEY, time.toString());\n  }\n\n  private savePauseState(): void {\n    const state = {\n      isPaused: true,\n      pauseStartTime: this.pauseStartTime,\n      pausedRemainingTime: this.pausedRemainingTime,\n      mode: this.currentConfig.mode,\n      lastActivityTime: this.lastActivityTime\n    };\n    localStorage.setItem(this.PAUSE_STATE_KEY, JSON.stringify(state));\n  }\n\n  private clearPauseState(): void {\n    localStorage.removeItem(this.PAUSE_STATE_KEY);\n  }\n\n  private restorePauseState(): void {\n    const stateStr = localStorage.getItem(this.PAUSE_STATE_KEY);\n    if (!stateStr) return;\n\n    try {\n      const state = JSON.parse(stateStr);\n      if (state.isPaused) {\n        this.isPaused = true;\n        this.pauseStartTime = state.pauseStartTime;\n        this.pausedRemainingTime = state.pausedRemainingTime;\n        this.lastActivityTime = state.lastActivityTime;\n        this.currentConfig.mode = state.mode;\n\n        if (this.timerSubscription) {\n          this.timerSubscription.unsubscribe();\n          this.timerSubscription = null;\n        }\n        this.removeActivityListeners();\n      }\n    } catch (e) {\n      console.error('恢复暂停状态失败:', e);\n      this.clearPauseState();\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* 默认配置
|
|
3
|
+
*/
|
|
4
|
+
export const DEFAULT_AUTO_LOGOUT_CONFIG = {
|
|
5
|
+
mode: 'fixed',
|
|
6
|
+
timeout: 300,
|
|
7
|
+
warningTime: 30,
|
|
8
|
+
maxExtendTimes: -1,
|
|
9
|
+
};
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0by1sb2dvdXQtY29uZmlnLmludGVyZmFjZS5qcyIsInNvdXJjZVJvb3QiOiJuZzovL25neC1hdXRvLWxvZ291dC8iLCJzb3VyY2VzIjpbImxpYi9tb2RlbHMvYXV0by1sb2dvdXQtY29uZmlnLmludGVyZmFjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFtREE7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSwwQkFBMEIsR0FBcUI7SUFDMUQsSUFBSSxFQUFFLE9BQU87SUFDYixPQUFPLEVBQUUsR0FBRztJQUNaLFdBQVcsRUFBRSxFQUFFO0lBQ2YsY0FBYyxFQUFFLENBQUMsQ0FBQztDQUNuQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiDoh6rliqjnmbvlh7rphY3nva7mjqXlj6NcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBBdXRvTG9nb3V0Q29uZmlnIHtcbiAgLyoqXG4gICAqIOaooeW8j++8mmZpeGVkPeWbuuWumuaXtumXtCwgaWRsZT3nqbrpl7LotoXml7ZcbiAgICovXG4gIG1vZGU6ICdmaXhlZCcgfCAnaWRsZSc7XG5cbiAgLyoqXG4gICAqIOi2heaXtuaXtumXtO+8iOenku+8iVxuICAgKi9cbiAgdGltZW91dDogbnVtYmVyO1xuXG4gIC8qKlxuICAgKiDorablkYrml7bpl7TvvIjnp5LvvInvvIzpu5jorqQzMFxuICAgKi9cbiAgd2FybmluZ1RpbWU/OiBudW1iZXI7XG5cbiAgLyoqXG4gICAqIOacgOWkp+W7tumVv+asoeaVsO+8jC0xPeaXoOmZkOWItu+8jDA95LiN5YWB6K6477yMPjA95YW35L2T5qyh5pWwXG4gICAqL1xuICBtYXhFeHRlbmRUaW1lcz86IG51bWJlcjtcblxuICAvKipcbiAgICog6K2m5ZGK5Zue6LCD5Ye95pWw77yI5pu/5LujVG9hc3TvvIlcbiAgICogQHBhcmFtIHNlY29uZHMg5Ymp5L2Z56eS5pWwXG4gICAqL1xuICBvbldhcm5pbmc/OiAoc2Vjb25kczogbnVtYmVyKSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiDotoXml7blm57osIPlh73mlbDvvIjmiafooYznmbvlh7rmk43kvZzvvIlcbiAgICovXG4gIG9uVGltZW91dD86ICgpID0+IHZvaWQ7XG5cbiAgLyoqXG4gICAqIOW7tumVv+S8muivneaIkOWKn+Wbnuiwg1xuICAgKi9cbiAgb25FeHRlbmRlZD86IChzZWNvbmRzOiBudW1iZXIpID0+IHZvaWQ7XG5cbiAgLyoqXG4gICAqIOaaguWBnOWbnuiwg1xuICAgKi9cbiAgb25QYXVzZWQ/OiAoKSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiDmgaLlpI3lm57osINcbiAgICovXG4gIG9uUmVzdW1lZD86ICgpID0+IHZvaWQ7XG59XG5cbi8qKlxuICog6buY6K6k6YWN572uXG4gKi9cbmV4cG9ydCBjb25zdCBERUZBVUxUX0FVVE9fTE9HT1VUX0NPTkZJRzogQXV0b0xvZ291dENvbmZpZyA9IHtcbiAgbW9kZTogJ2ZpeGVkJyxcbiAgdGltZW91dDogMzAwLCAvLyA15YiG6ZKfXG4gIHdhcm5pbmdUaW1lOiAzMCxcbiAgbWF4RXh0ZW5kVGltZXM6IC0xLFxufTtcbiJdfQ==
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { InjectionToken } from '@angular/core';
|
|
2
|
+
/**
|
|
3
|
+
* InjectionToken for auto logout configuration
|
|
4
|
+
* Extracted to a separate file to avoid Angular 9 Ivy compilation scope issues
|
|
5
|
+
*/
|
|
6
|
+
export const AUTO_LOGOUT_CONFIG = new InjectionToken('AutoLogoutConfig');
|
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0by1sb2dvdXQtY29uZmlnLnRva2VuLmpzIiwic291cmNlUm9vdCI6Im5nOi8vbmd4LWF1dG8tbG9nb3V0LyIsInNvdXJjZXMiOlsibGliL21vZGVscy9hdXRvLWxvZ291dC1jb25maWcudG9rZW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUcvQzs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxJQUFJLGNBQWMsQ0FBbUIsa0JBQWtCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBBdXRvTG9nb3V0Q29uZmlnIH0gZnJvbSAnLi9hdXRvLWxvZ291dC1jb25maWcuaW50ZXJmYWNlJztcblxuLyoqXG4gKiBJbmplY3Rpb25Ub2tlbiBmb3IgYXV0byBsb2dvdXQgY29uZmlndXJhdGlvblxuICogRXh0cmFjdGVkIHRvIGEgc2VwYXJhdGUgZmlsZSB0byBhdm9pZCBBbmd1bGFyIDkgSXZ5IGNvbXBpbGF0aW9uIHNjb3BlIGlzc3Vlc1xuICovXG5leHBvcnQgY29uc3QgQVVUT19MT0dPVVRfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPEF1dG9Mb2dvdXRDb25maWc+KCdBdXRvTG9nb3V0Q29uZmlnJyk7XG4iXX0=
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { AUTO_LOGOUT_CONFIG } from './models/auto-logout-config.token';
|
|
2
|
+
/**
|
|
3
|
+
* 提供自动登出配置(Angular 15+ Standalone)
|
|
4
|
+
*/
|
|
5
|
+
export function provideAutoLogout(config) {
|
|
6
|
+
return {
|
|
7
|
+
provide: AUTO_LOGOUT_CONFIG,
|
|
8
|
+
useValue: config
|
|
9
|
+
};
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvdmlkZS1hdXRvLWxvZ291dC5qcyIsInNvdXJjZVJvb3QiOiJuZzovL25neC1hdXRvLWxvZ291dC8iLCJzb3VyY2VzIjpbImxpYi9wcm92aWRlLWF1dG8tbG9nb3V0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBR3ZFOztHQUVHO0FBQ0gsTUFBTSxVQUFVLGlCQUFpQixDQUFDLE1BQXdCO0lBQ3hELE9BQU87UUFDTCxPQUFPLEVBQUUsa0JBQWtCO1FBQzNCLFFBQVEsRUFBRSxNQUFNO0tBQ2pCLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUHJvdmlkZXIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEFVVE9fTE9HT1VUX0NPTkZJRyB9IGZyb20gJy4vbW9kZWxzL2F1dG8tbG9nb3V0LWNvbmZpZy50b2tlbic7XG5pbXBvcnQgeyBBdXRvTG9nb3V0Q29uZmlnIH0gZnJvbSAnLi9tb2RlbHMvYXV0by1sb2dvdXQtY29uZmlnLmludGVyZmFjZSc7XG5cbi8qKlxuICog5o+Q5L6b6Ieq5Yqo55m75Ye66YWN572u77yIQW5ndWxhciAxNSsgU3RhbmRhbG9uZe+8iVxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZUF1dG9Mb2dvdXQoY29uZmlnOiBBdXRvTG9nb3V0Q29uZmlnKTogUHJvdmlkZXIge1xuICByZXR1cm4ge1xuICAgIHByb3ZpZGU6IEFVVE9fTE9HT1VUX0NPTkZJRyxcbiAgICB1c2VWYWx1ZTogY29uZmlnXG4gIH07XG59XG4iXX0=
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './public-api';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmd4LWF1dG8tbG9nb3V0LmpzIiwic291cmNlUm9vdCI6Im5nOi8vbmd4LWF1dG8tbG9nb3V0LyIsInNvdXJjZXMiOlsibmd4LWF1dG8tbG9nb3V0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vcHVibGljLWFwaSc7XG4iXX0=
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Public API Surface of auto-logout
|
|
3
|
+
*/
|
|
4
|
+
export { DEFAULT_AUTO_LOGOUT_CONFIG } from './lib/models/auto-logout-config.interface';
|
|
5
|
+
export { AUTO_LOGOUT_CONFIG } from './lib/models/auto-logout-config.token';
|
|
6
|
+
export { AutoLogoutService } from './lib/auto-logout.service';
|
|
7
|
+
export { AutoLogoutModule } from './lib/auto-logout.module';
|
|
8
|
+
export { AutoLogoutComponent } from './lib/auto-logout.component';
|
|
9
|
+
export { provideAutoLogout } from './lib/provide-auto-logout';
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiJuZzovL25neC1hdXRvLWxvZ291dC8iLCJzb3VyY2VzIjpbInB1YmxpYy1hcGkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxPQUFPLEVBQW9CLDBCQUEwQixFQUFFLE1BQU0sMkNBQTJDLENBQUM7QUFDekcsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7QUFDM0UsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDOUQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDNUQsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDbEUsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sMkJBQTJCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogUHVibGljIEFQSSBTdXJmYWNlIG9mIGF1dG8tbG9nb3V0XG4gKi9cblxuZXhwb3J0IHsgQXV0b0xvZ291dENvbmZpZywgREVGQVVMVF9BVVRPX0xPR09VVF9DT05GSUcgfSBmcm9tICcuL2xpYi9tb2RlbHMvYXV0by1sb2dvdXQtY29uZmlnLmludGVyZmFjZSc7XG5leHBvcnQgeyBBVVRPX0xPR09VVF9DT05GSUcgfSBmcm9tICcuL2xpYi9tb2RlbHMvYXV0by1sb2dvdXQtY29uZmlnLnRva2VuJztcbmV4cG9ydCB7IEF1dG9Mb2dvdXRTZXJ2aWNlIH0gZnJvbSAnLi9saWIvYXV0by1sb2dvdXQuc2VydmljZSc7XG5leHBvcnQgeyBBdXRvTG9nb3V0TW9kdWxlIH0gZnJvbSAnLi9saWIvYXV0by1sb2dvdXQubW9kdWxlJztcbmV4cG9ydCB7IEF1dG9Mb2dvdXRDb21wb25lbnQgfSBmcm9tICcuL2xpYi9hdXRvLWxvZ291dC5jb21wb25lbnQnO1xuZXhwb3J0IHsgcHJvdmlkZUF1dG9Mb2dvdXQgfSBmcm9tICcuL2xpYi9wcm92aWRlLWF1dG8tbG9nb3V0JztcbiJdfQ==
|