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,351 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
3
|
+
import { AutoLogoutService } from './auto-logout.service';
|
|
4
|
+
let AutoLogoutComponent = class AutoLogoutComponent {
|
|
5
|
+
constructor(autoLogout) {
|
|
6
|
+
this.autoLogout = autoLogout;
|
|
7
|
+
// ==================== 输入属性 ====================
|
|
8
|
+
/**
|
|
9
|
+
* 是否显示延长按钮
|
|
10
|
+
*/
|
|
11
|
+
this.showExtendButton = true;
|
|
12
|
+
/**
|
|
13
|
+
* 是否显示暂停按钮
|
|
14
|
+
*/
|
|
15
|
+
this.showPauseButton = true;
|
|
16
|
+
/**
|
|
17
|
+
* 延长的秒数(默认30分钟)
|
|
18
|
+
*/
|
|
19
|
+
this.extendSeconds = 1800;
|
|
20
|
+
// ==================== 输出事件 ====================
|
|
21
|
+
/**
|
|
22
|
+
* 警告事件
|
|
23
|
+
*/
|
|
24
|
+
this.onWarning = new EventEmitter();
|
|
25
|
+
/**
|
|
26
|
+
* 超时事件
|
|
27
|
+
*/
|
|
28
|
+
this.onTimeout = new EventEmitter();
|
|
29
|
+
/**
|
|
30
|
+
* 延长成功事件
|
|
31
|
+
*/
|
|
32
|
+
this.onExtended = new EventEmitter();
|
|
33
|
+
/**
|
|
34
|
+
* 暂停事件
|
|
35
|
+
*/
|
|
36
|
+
this.onPaused = new EventEmitter();
|
|
37
|
+
/**
|
|
38
|
+
* 恢复事件
|
|
39
|
+
*/
|
|
40
|
+
this.onResumed = new EventEmitter();
|
|
41
|
+
// ==================== 内部状态 ====================
|
|
42
|
+
this.subscription = null;
|
|
43
|
+
this.countdown = '00:00';
|
|
44
|
+
this.isWarning = false;
|
|
45
|
+
this.isUrgent = false;
|
|
46
|
+
this.isPaused = false;
|
|
47
|
+
this.canExtend = true;
|
|
48
|
+
this.remainingExtends = -1;
|
|
49
|
+
}
|
|
50
|
+
ngOnInit() {
|
|
51
|
+
// 如果提供了自定义配置,启动监控
|
|
52
|
+
if (this.config) {
|
|
53
|
+
this.autoLogout.startMonitoring(this.config);
|
|
54
|
+
}
|
|
55
|
+
// 订阅倒计时
|
|
56
|
+
this.subscription = this.autoLogout.getCountdown().subscribe((seconds) => {
|
|
57
|
+
var _a;
|
|
58
|
+
this.countdown = this.autoLogout.formatCountdown(seconds);
|
|
59
|
+
this.isWarning = this.autoLogout.isWarningState(seconds);
|
|
60
|
+
this.isUrgent = this.autoLogout.isUrgentState(seconds);
|
|
61
|
+
// 触发警告事件
|
|
62
|
+
const warningTime = ((_a = this.config) === null || _a === void 0 ? void 0 : _a.warningTime) || 30;
|
|
63
|
+
if (seconds === warningTime) {
|
|
64
|
+
this.onWarning.emit(seconds);
|
|
65
|
+
}
|
|
66
|
+
// 触发超时事件
|
|
67
|
+
if (seconds <= 0) {
|
|
68
|
+
this.onTimeout.emit();
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
// 定期检查状态
|
|
72
|
+
setInterval(() => {
|
|
73
|
+
this.updateStatus();
|
|
74
|
+
}, 1000);
|
|
75
|
+
this.updateStatus();
|
|
76
|
+
}
|
|
77
|
+
ngOnDestroy() {
|
|
78
|
+
if (this.subscription) {
|
|
79
|
+
this.subscription.unsubscribe();
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
// ==================== 公开方法 ====================
|
|
83
|
+
/**
|
|
84
|
+
* 延长会话
|
|
85
|
+
*/
|
|
86
|
+
extendSession() {
|
|
87
|
+
this.handleExtend();
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* 暂停
|
|
91
|
+
*/
|
|
92
|
+
pause() {
|
|
93
|
+
this.handleTogglePause();
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* 恢复
|
|
97
|
+
*/
|
|
98
|
+
resume() {
|
|
99
|
+
this.handleTogglePause();
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* 获取模板上下文(用于自定义模板)
|
|
103
|
+
*/
|
|
104
|
+
get templateContext() {
|
|
105
|
+
return {
|
|
106
|
+
countdown: this.countdown,
|
|
107
|
+
isWarning: this.isWarning,
|
|
108
|
+
isUrgent: this.isUrgent,
|
|
109
|
+
isPaused: this.isPaused,
|
|
110
|
+
canExtend: this.canExtend,
|
|
111
|
+
remainingExtends: this.remainingExtends,
|
|
112
|
+
extendSession: () => this.handleExtend(),
|
|
113
|
+
togglePause: () => this.handleTogglePause(),
|
|
114
|
+
formatCountdown: (seconds) => this.autoLogout.formatCountdown(seconds)
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
// ==================== 内部方法 ====================
|
|
118
|
+
updateStatus() {
|
|
119
|
+
this.isPaused = this.autoLogout.isAutoLogoutPaused();
|
|
120
|
+
this.canExtend = this.autoLogout.canExtendSession();
|
|
121
|
+
this.remainingExtends = this.autoLogout.getRemainingExtendTimes();
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* 处理延长会话(模板调用)
|
|
125
|
+
*/
|
|
126
|
+
handleExtend() {
|
|
127
|
+
const success = this.autoLogout.extendSession(this.extendSeconds);
|
|
128
|
+
if (success) {
|
|
129
|
+
this.onExtended.emit(this.extendSeconds);
|
|
130
|
+
}
|
|
131
|
+
this.updateStatus();
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* 处理暂停/恢复切换(模板调用)
|
|
135
|
+
*/
|
|
136
|
+
handleTogglePause() {
|
|
137
|
+
if (this.isPaused) {
|
|
138
|
+
this.autoLogout.resume();
|
|
139
|
+
this.onResumed.emit();
|
|
140
|
+
}
|
|
141
|
+
else {
|
|
142
|
+
this.autoLogout.pause();
|
|
143
|
+
this.onPaused.emit();
|
|
144
|
+
}
|
|
145
|
+
this.updateStatus();
|
|
146
|
+
}
|
|
147
|
+
};
|
|
148
|
+
AutoLogoutComponent.ctorParameters = () => [
|
|
149
|
+
{ type: AutoLogoutService }
|
|
150
|
+
];
|
|
151
|
+
__decorate([
|
|
152
|
+
Input()
|
|
153
|
+
], AutoLogoutComponent.prototype, "showExtendButton", void 0);
|
|
154
|
+
__decorate([
|
|
155
|
+
Input()
|
|
156
|
+
], AutoLogoutComponent.prototype, "showPauseButton", void 0);
|
|
157
|
+
__decorate([
|
|
158
|
+
Input()
|
|
159
|
+
], AutoLogoutComponent.prototype, "extendSeconds", void 0);
|
|
160
|
+
__decorate([
|
|
161
|
+
Input()
|
|
162
|
+
], AutoLogoutComponent.prototype, "config", void 0);
|
|
163
|
+
__decorate([
|
|
164
|
+
Input()
|
|
165
|
+
], AutoLogoutComponent.prototype, "customTemplate", void 0);
|
|
166
|
+
__decorate([
|
|
167
|
+
Output()
|
|
168
|
+
], AutoLogoutComponent.prototype, "onWarning", void 0);
|
|
169
|
+
__decorate([
|
|
170
|
+
Output()
|
|
171
|
+
], AutoLogoutComponent.prototype, "onTimeout", void 0);
|
|
172
|
+
__decorate([
|
|
173
|
+
Output()
|
|
174
|
+
], AutoLogoutComponent.prototype, "onExtended", void 0);
|
|
175
|
+
__decorate([
|
|
176
|
+
Output()
|
|
177
|
+
], AutoLogoutComponent.prototype, "onPaused", void 0);
|
|
178
|
+
__decorate([
|
|
179
|
+
Output()
|
|
180
|
+
], AutoLogoutComponent.prototype, "onResumed", void 0);
|
|
181
|
+
AutoLogoutComponent = __decorate([
|
|
182
|
+
Component({
|
|
183
|
+
selector: 'ngx-auto-logout',
|
|
184
|
+
template: `
|
|
185
|
+
<!-- 默认UI模式(傻瓜模式) -->
|
|
186
|
+
<div *ngIf="!customTemplate" class="ngx-auto-logout-container">
|
|
187
|
+
<!-- 倒计时显示 -->
|
|
188
|
+
<div class="countdown-display" [class.warning]="isWarning" [class.urgent]="isUrgent" [class.paused]="isPaused">
|
|
189
|
+
<span class="icon">{{ isPaused ? '⏸️' : '⏱️' }}</span>
|
|
190
|
+
<span class="time">{{ countdown }}</span>
|
|
191
|
+
<span *ngIf="isPaused" class="pause-label">(已暂停)</span>
|
|
192
|
+
</div>
|
|
193
|
+
|
|
194
|
+
<!-- 控制按钮 -->
|
|
195
|
+
<div class="controls">
|
|
196
|
+
<!-- 延长会话按钮 -->
|
|
197
|
+
<button *ngIf="showExtendButton && canExtend" class="btn btn-extend" (click)="handleExtend()" [disabled]="!canExtend" title="延长30分钟">
|
|
198
|
+
➕ 延长
|
|
199
|
+
<span *ngIf="remainingExtends !== -1" class="extend-count"> ({{ remainingExtends }}) </span>
|
|
200
|
+
</button>
|
|
201
|
+
|
|
202
|
+
<!-- 暂停/恢复按钮 -->
|
|
203
|
+
<button
|
|
204
|
+
*ngIf="showPauseButton"
|
|
205
|
+
class="btn btn-pause"
|
|
206
|
+
[class.paused]="isPaused"
|
|
207
|
+
(click)="handleTogglePause()"
|
|
208
|
+
[title]="isPaused ? '恢复自动登出' : '暂停自动登出'"
|
|
209
|
+
>
|
|
210
|
+
{{ isPaused ? '▶️ 恢复' : '⏸️ 暂停' }}
|
|
211
|
+
</button>
|
|
212
|
+
</div>
|
|
213
|
+
</div>
|
|
214
|
+
|
|
215
|
+
<!-- 自定义模板模式(专家模式) -->
|
|
216
|
+
<ng-container *ngIf="customTemplate">
|
|
217
|
+
<ng-container [ngTemplateOutlet]="customTemplate" [ngTemplateOutletContext]="templateContext"></ng-container>
|
|
218
|
+
</ng-container>
|
|
219
|
+
`,
|
|
220
|
+
styles: [`
|
|
221
|
+
.ngx-auto-logout-container {
|
|
222
|
+
display: inline-flex;
|
|
223
|
+
align-items: center;
|
|
224
|
+
gap: 12px;
|
|
225
|
+
padding: 8px 12px;
|
|
226
|
+
background: #f8f9fa;
|
|
227
|
+
border-radius: 6px;
|
|
228
|
+
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
.countdown-display {
|
|
232
|
+
display: flex;
|
|
233
|
+
align-items: center;
|
|
234
|
+
gap: 8px;
|
|
235
|
+
padding: 6px 12px;
|
|
236
|
+
background: white;
|
|
237
|
+
border-radius: 4px;
|
|
238
|
+
font-weight: bold;
|
|
239
|
+
font-size: 16px;
|
|
240
|
+
transition: all 0.3s ease;
|
|
241
|
+
}
|
|
242
|
+
|
|
243
|
+
.countdown-display.warning {
|
|
244
|
+
background: #fff3cd;
|
|
245
|
+
color: #856404;
|
|
246
|
+
animation: blink-warning 1s ease-in-out infinite;
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
.countdown-display.urgent {
|
|
250
|
+
background: #f8d7da;
|
|
251
|
+
color: #721c24;
|
|
252
|
+
animation: blink-urgent 0.5s ease-in-out infinite;
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
.countdown-display.paused {
|
|
256
|
+
opacity: 0.6;
|
|
257
|
+
background: #e9ecef;
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
.icon {
|
|
261
|
+
font-size: 18px;
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
.time {
|
|
265
|
+
min-width: 50px;
|
|
266
|
+
text-align: center;
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
.pause-label {
|
|
270
|
+
font-size: 12px;
|
|
271
|
+
color: #6c757d;
|
|
272
|
+
font-weight: normal;
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
.controls {
|
|
276
|
+
display: flex;
|
|
277
|
+
gap: 8px;
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
.btn {
|
|
281
|
+
padding: 6px 12px;
|
|
282
|
+
font-size: 13px;
|
|
283
|
+
border: none;
|
|
284
|
+
border-radius: 4px;
|
|
285
|
+
cursor: pointer;
|
|
286
|
+
transition: all 0.2s;
|
|
287
|
+
font-weight: 500;
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
.btn-extend {
|
|
291
|
+
background: #28a745;
|
|
292
|
+
color: white;
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
.btn-extend:hover:not(:disabled) {
|
|
296
|
+
background: #218838;
|
|
297
|
+
transform: translateY(-1px);
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
.btn-extend:disabled {
|
|
301
|
+
background: #ccc;
|
|
302
|
+
cursor: not-allowed;
|
|
303
|
+
opacity: 0.6;
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
.btn-pause {
|
|
307
|
+
background: #007bff;
|
|
308
|
+
color: white;
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
.btn-pause:hover {
|
|
312
|
+
background: #0056b3;
|
|
313
|
+
transform: translateY(-1px);
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
.btn-pause.paused {
|
|
317
|
+
background: #ffc107;
|
|
318
|
+
color: #212529;
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
.extend-count {
|
|
322
|
+
font-size: 11px;
|
|
323
|
+
opacity: 0.9;
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
@keyframes blink-warning {
|
|
327
|
+
0%,
|
|
328
|
+
100% {
|
|
329
|
+
opacity: 1;
|
|
330
|
+
}
|
|
331
|
+
50% {
|
|
332
|
+
opacity: 0.7;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
@keyframes blink-urgent {
|
|
337
|
+
0%,
|
|
338
|
+
100% {
|
|
339
|
+
opacity: 1;
|
|
340
|
+
transform: scale(1);
|
|
341
|
+
}
|
|
342
|
+
50% {
|
|
343
|
+
opacity: 0.6;
|
|
344
|
+
transform: scale(1.05);
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
`]
|
|
348
|
+
})
|
|
349
|
+
], AutoLogoutComponent);
|
|
350
|
+
export { AutoLogoutComponent };
|
|
351
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auto-logout.component.js","sourceRoot":"ng://ngx-auto-logout/","sources":["lib/auto-logout.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, OnDestroy, TemplateRef } from '@angular/core';\nimport { AutoLogoutService } from './auto-logout.service';\nimport { AutoLogoutConfig } from './models/auto-logout-config.interface';\nimport { Subscription } from 'rxjs';\n\n@Component({\n  selector: 'ngx-auto-logout',\n  template: `\n    <!-- 默认UI模式（傻瓜模式） -->\n    <div *ngIf=\"!customTemplate\" class=\"ngx-auto-logout-container\">\n      <!-- 倒计时显示 -->\n      <div class=\"countdown-display\" [class.warning]=\"isWarning\" [class.urgent]=\"isUrgent\" [class.paused]=\"isPaused\">\n        <span class=\"icon\">{{ isPaused ? '⏸️' : '⏱️' }}</span>\n        <span class=\"time\">{{ countdown }}</span>\n        <span *ngIf=\"isPaused\" class=\"pause-label\">(已暂停)</span>\n      </div>\n\n      <!-- 控制按钮 -->\n      <div class=\"controls\">\n        <!-- 延长会话按钮 -->\n        <button *ngIf=\"showExtendButton && canExtend\" class=\"btn btn-extend\" (click)=\"handleExtend()\" [disabled]=\"!canExtend\" title=\"延长30分钟\">\n          ➕ 延长\n          <span *ngIf=\"remainingExtends !== -1\" class=\"extend-count\"> ({{ remainingExtends }}) </span>\n        </button>\n\n        <!-- 暂停/恢复按钮 -->\n        <button\n          *ngIf=\"showPauseButton\"\n          class=\"btn btn-pause\"\n          [class.paused]=\"isPaused\"\n          (click)=\"handleTogglePause()\"\n          [title]=\"isPaused ? '恢复自动登出' : '暂停自动登出'\"\n        >\n          {{ isPaused ? '▶️ 恢复' : '⏸️ 暂停' }}\n        </button>\n      </div>\n    </div>\n\n    <!-- 自定义模板模式（专家模式） -->\n    <ng-container *ngIf=\"customTemplate\">\n      <ng-container [ngTemplateOutlet]=\"customTemplate\" [ngTemplateOutletContext]=\"templateContext\"></ng-container>\n    </ng-container>\n  `,\n  styles: [\n    `\n      .ngx-auto-logout-container {\n        display: inline-flex;\n        align-items: center;\n        gap: 12px;\n        padding: 8px 12px;\n        background: #f8f9fa;\n        border-radius: 6px;\n        font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n      }\n\n      .countdown-display {\n        display: flex;\n        align-items: center;\n        gap: 8px;\n        padding: 6px 12px;\n        background: white;\n        border-radius: 4px;\n        font-weight: bold;\n        font-size: 16px;\n        transition: all 0.3s ease;\n      }\n\n      .countdown-display.warning {\n        background: #fff3cd;\n        color: #856404;\n        animation: blink-warning 1s ease-in-out infinite;\n      }\n\n      .countdown-display.urgent {\n        background: #f8d7da;\n        color: #721c24;\n        animation: blink-urgent 0.5s ease-in-out infinite;\n      }\n\n      .countdown-display.paused {\n        opacity: 0.6;\n        background: #e9ecef;\n      }\n\n      .icon {\n        font-size: 18px;\n      }\n\n      .time {\n        min-width: 50px;\n        text-align: center;\n      }\n\n      .pause-label {\n        font-size: 12px;\n        color: #6c757d;\n        font-weight: normal;\n      }\n\n      .controls {\n        display: flex;\n        gap: 8px;\n      }\n\n      .btn {\n        padding: 6px 12px;\n        font-size: 13px;\n        border: none;\n        border-radius: 4px;\n        cursor: pointer;\n        transition: all 0.2s;\n        font-weight: 500;\n      }\n\n      .btn-extend {\n        background: #28a745;\n        color: white;\n      }\n\n      .btn-extend:hover:not(:disabled) {\n        background: #218838;\n        transform: translateY(-1px);\n      }\n\n      .btn-extend:disabled {\n        background: #ccc;\n        cursor: not-allowed;\n        opacity: 0.6;\n      }\n\n      .btn-pause {\n        background: #007bff;\n        color: white;\n      }\n\n      .btn-pause:hover {\n        background: #0056b3;\n        transform: translateY(-1px);\n      }\n\n      .btn-pause.paused {\n        background: #ffc107;\n        color: #212529;\n      }\n\n      .extend-count {\n        font-size: 11px;\n        opacity: 0.9;\n      }\n\n      @keyframes blink-warning {\n        0%,\n        100% {\n          opacity: 1;\n        }\n        50% {\n          opacity: 0.7;\n        }\n      }\n\n      @keyframes blink-urgent {\n        0%,\n        100% {\n          opacity: 1;\n          transform: scale(1);\n        }\n        50% {\n          opacity: 0.6;\n          transform: scale(1.05);\n        }\n      }\n    `\n  ]\n})\nexport class AutoLogoutComponent implements OnInit, OnDestroy {\n  // ==================== 输入属性 ====================\n\n  /**\n   * 是否显示延长按钮\n   */\n  @Input() showExtendButton = true;\n\n  /**\n   * 是否显示暂停按钮\n   */\n  @Input() showPauseButton = true;\n\n  /**\n   * 延长的秒数（默认30分钟）\n   */\n  @Input() extendSeconds = 1800;\n\n  /**\n   * 自定义配置（会覆盖模块配置）\n   */\n  @Input() config?: Partial<AutoLogoutConfig>;\n\n  /**\n   * 自定义模板（专家模式）\n   */\n  @Input() customTemplate?: TemplateRef<any>;\n\n  // ==================== 输出事件 ====================\n\n  /**\n   * 警告事件\n   */\n  @Output() onWarning = new EventEmitter<number>();\n\n  /**\n   * 超时事件\n   */\n  @Output() onTimeout = new EventEmitter<void>();\n\n  /**\n   * 延长成功事件\n   */\n  @Output() onExtended = new EventEmitter<number>();\n\n  /**\n   * 暂停事件\n   */\n  @Output() onPaused = new EventEmitter<void>();\n\n  /**\n   * 恢复事件\n   */\n  @Output() onResumed = new EventEmitter<void>();\n\n  // ==================== 内部状态 ====================\n\n  private subscription: Subscription | null = null;\n  countdown = '00:00';\n  isWarning = false;\n  isUrgent = false;\n  isPaused = false;\n  canExtend = true;\n  remainingExtends = -1;\n\n  constructor(private autoLogout: AutoLogoutService) {}\n\n  ngOnInit() {\n    // 如果提供了自定义配置，启动监控\n    if (this.config) {\n      this.autoLogout.startMonitoring(this.config);\n    }\n\n    // 订阅倒计时\n    this.subscription = this.autoLogout.getCountdown().subscribe((seconds) => {\n      this.countdown = this.autoLogout.formatCountdown(seconds);\n      this.isWarning = this.autoLogout.isWarningState(seconds);\n      this.isUrgent = this.autoLogout.isUrgentState(seconds);\n\n      // 触发警告事件\n      const warningTime = this.config?.warningTime || 30;\n      if (seconds === warningTime) {\n        this.onWarning.emit(seconds);\n      }\n\n      // 触发超时事件\n      if (seconds <= 0) {\n        this.onTimeout.emit();\n      }\n    });\n\n    // 定期检查状态\n    setInterval(() => {\n      this.updateStatus();\n    }, 1000);\n\n    this.updateStatus();\n  }\n\n  ngOnDestroy() {\n    if (this.subscription) {\n      this.subscription.unsubscribe();\n    }\n  }\n\n  // ==================== 公开方法 ====================\n\n  /**\n   * 延长会话\n   */\n  extendSession(): void {\n    this.handleExtend();\n  }\n\n  /**\n   * 暂停\n   */\n  pause(): void {\n    this.handleTogglePause();\n  }\n\n  /**\n   * 恢复\n   */\n  resume(): void {\n    this.handleTogglePause();\n  }\n\n  /**\n   * 获取模板上下文（用于自定义模板）\n   */\n  get templateContext(): any {\n    return {\n      countdown: this.countdown,\n      isWarning: this.isWarning,\n      isUrgent: this.isUrgent,\n      isPaused: this.isPaused,\n      canExtend: this.canExtend,\n      remainingExtends: this.remainingExtends,\n      extendSession: () => this.handleExtend(),\n      togglePause: () => this.handleTogglePause(),\n      formatCountdown: (seconds: number) => this.autoLogout.formatCountdown(seconds)\n    };\n  }\n\n  // ==================== 内部方法 ====================\n\n  private updateStatus(): void {\n    this.isPaused = this.autoLogout.isAutoLogoutPaused();\n    this.canExtend = this.autoLogout.canExtendSession();\n    this.remainingExtends = this.autoLogout.getRemainingExtendTimes();\n  }\n\n  /**\n   * 处理延长会话（模板调用）\n   */\n  handleExtend(): void {\n    const success = this.autoLogout.extendSession(this.extendSeconds);\n    if (success) {\n      this.onExtended.emit(this.extendSeconds);\n    }\n    this.updateStatus();\n  }\n\n  /**\n   * 处理暂停/恢复切换（模板调用）\n   */\n  handleTogglePause(): void {\n    if (this.isPaused) {\n      this.autoLogout.resume();\n      this.onResumed.emit();\n    } else {\n      this.autoLogout.pause();\n      this.onPaused.emit();\n    }\n    this.updateStatus();\n  }\n}\n"]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
var AutoLogoutModule_1;
|
|
2
|
+
import { __decorate } from "tslib";
|
|
3
|
+
import { NgModule } from '@angular/core';
|
|
4
|
+
import { CommonModule } from '@angular/common';
|
|
5
|
+
import { AutoLogoutComponent } from './auto-logout.component';
|
|
6
|
+
/**
|
|
7
|
+
* Auto logout module
|
|
8
|
+
* Note: Service uses providedIn: 'root', module only declares/exports component
|
|
9
|
+
*/
|
|
10
|
+
let AutoLogoutModule = AutoLogoutModule_1 = class AutoLogoutModule {
|
|
11
|
+
static forRoot() {
|
|
12
|
+
return {
|
|
13
|
+
ngModule: AutoLogoutModule_1,
|
|
14
|
+
providers: []
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
};
|
|
18
|
+
AutoLogoutModule = AutoLogoutModule_1 = __decorate([
|
|
19
|
+
NgModule({
|
|
20
|
+
declarations: [AutoLogoutComponent],
|
|
21
|
+
imports: [CommonModule],
|
|
22
|
+
exports: [AutoLogoutComponent]
|
|
23
|
+
})
|
|
24
|
+
], AutoLogoutModule);
|
|
25
|
+
export { AutoLogoutModule };
|
|
26
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0by1sb2dvdXQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6Im5nOi8vbmd4LWF1dG8tbG9nb3V0LyIsInNvdXJjZXMiOlsibGliL2F1dG8tbG9nb3V0Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQXVCLE1BQU0sZUFBZSxDQUFDO0FBQzlELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUU5RDs7O0dBR0c7QUFNSCxJQUFhLGdCQUFnQix3QkFBN0IsTUFBYSxnQkFBZ0I7SUFDM0IsTUFBTSxDQUFDLE9BQU87UUFDWixPQUFPO1lBQ0wsUUFBUSxFQUFFLGtCQUFnQjtZQUMxQixTQUFTLEVBQUUsRUFBRTtTQUNkLENBQUM7SUFDSixDQUFDO0NBQ0YsQ0FBQTtBQVBZLGdCQUFnQjtJQUw1QixRQUFRLENBQUM7UUFDUixZQUFZLEVBQUUsQ0FBQyxtQkFBbUIsQ0FBQztRQUNuQyxPQUFPLEVBQUUsQ0FBQyxZQUFZLENBQUM7UUFDdkIsT0FBTyxFQUFFLENBQUMsbUJBQW1CLENBQUM7S0FDL0IsQ0FBQztHQUNXLGdCQUFnQixDQU81QjtTQVBZLGdCQUFnQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlLCBNb2R1bGVXaXRoUHJvdmlkZXJzIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgQXV0b0xvZ291dENvbXBvbmVudCB9IGZyb20gJy4vYXV0by1sb2dvdXQuY29tcG9uZW50JztcblxuLyoqXG4gKiBBdXRvIGxvZ291dCBtb2R1bGVcbiAqIE5vdGU6IFNlcnZpY2UgdXNlcyBwcm92aWRlZEluOiAncm9vdCcsIG1vZHVsZSBvbmx5IGRlY2xhcmVzL2V4cG9ydHMgY29tcG9uZW50XG4gKi9cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW0F1dG9Mb2dvdXRDb21wb25lbnRdLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcbiAgZXhwb3J0czogW0F1dG9Mb2dvdXRDb21wb25lbnRdXG59KVxuZXhwb3J0IGNsYXNzIEF1dG9Mb2dvdXRNb2R1bGUge1xuICBzdGF0aWMgZm9yUm9vdCgpOiBNb2R1bGVXaXRoUHJvdmlkZXJzPEF1dG9Mb2dvdXRNb2R1bGU+IHtcbiAgICByZXR1cm4ge1xuICAgICAgbmdNb2R1bGU6IEF1dG9Mb2dvdXRNb2R1bGUsXG4gICAgICBwcm92aWRlcnM6IFtdXG4gICAgfTtcbiAgfVxufVxuIl19
|