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.
Files changed (37) hide show
  1. package/README.md +247 -0
  2. package/dist/README.md +247 -0
  3. package/dist/bundles/ngx-auto-logout.umd.js +864 -0
  4. package/dist/bundles/ngx-auto-logout.umd.js.map +1 -0
  5. package/dist/bundles/ngx-auto-logout.umd.min.js +16 -0
  6. package/dist/bundles/ngx-auto-logout.umd.min.js.map +1 -0
  7. package/dist/esm2015/lib/auto-logout.component.js +351 -0
  8. package/dist/esm2015/lib/auto-logout.module.js +26 -0
  9. package/dist/esm2015/lib/auto-logout.service.js +384 -0
  10. package/dist/esm2015/lib/models/auto-logout-config.interface.js +10 -0
  11. package/dist/esm2015/lib/models/auto-logout-config.token.js +7 -0
  12. package/dist/esm2015/lib/provide-auto-logout.js +11 -0
  13. package/dist/esm2015/ngx-auto-logout.js +5 -0
  14. package/dist/esm2015/public-api.js +10 -0
  15. package/dist/esm5/lib/auto-logout.component.js +196 -0
  16. package/dist/esm5/lib/auto-logout.module.js +30 -0
  17. package/dist/esm5/lib/auto-logout.service.js +391 -0
  18. package/dist/esm5/lib/models/auto-logout-config.interface.js +10 -0
  19. package/dist/esm5/lib/models/auto-logout-config.token.js +7 -0
  20. package/dist/esm5/lib/provide-auto-logout.js +11 -0
  21. package/dist/esm5/ngx-auto-logout.js +5 -0
  22. package/dist/esm5/public-api.js +10 -0
  23. package/dist/fesm2015/ngx-auto-logout.js +787 -0
  24. package/dist/fesm2015/ngx-auto-logout.js.map +1 -0
  25. package/dist/fesm5/ngx-auto-logout.js +643 -0
  26. package/dist/fesm5/ngx-auto-logout.js.map +1 -0
  27. package/dist/lib/auto-logout.component.d.ts +81 -0
  28. package/dist/lib/auto-logout.module.d.ts +8 -0
  29. package/dist/lib/auto-logout.service.d.ts +108 -0
  30. package/dist/lib/models/auto-logout-config.interface.d.ts +46 -0
  31. package/dist/lib/models/auto-logout-config.token.d.ts +7 -0
  32. package/dist/lib/provide-auto-logout.d.ts +6 -0
  33. package/dist/ngx-auto-logout.d.ts +4 -0
  34. package/dist/ngx-auto-logout.metadata.json +1 -0
  35. package/dist/package.json +41 -0
  36. package/dist/public-api.d.ts +6 -0
  37. 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