@reskin/core 0.0.21 → 0.1.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 (45) hide show
  1. package/bundles/reskin-core-directives.umd.js +303 -163
  2. package/bundles/reskin-core-directives.umd.js.map +1 -1
  3. package/bundles/reskin-core-guards.umd.js +119 -32
  4. package/bundles/reskin-core-guards.umd.js.map +1 -1
  5. package/bundles/reskin-core-interceptors.umd.js +310 -92
  6. package/bundles/reskin-core-interceptors.umd.js.map +1 -1
  7. package/bundles/reskin-core-utils.umd.js +220 -77
  8. package/bundles/reskin-core-utils.umd.js.map +1 -1
  9. package/directives/auth.directive.d.ts +56 -9
  10. package/directives/load.styles.directive.d.ts +45 -5
  11. package/directives/string.template.outlet.directive.d.ts +68 -11
  12. package/esm2015/directives/auth.directive.js +71 -30
  13. package/esm2015/directives/load.styles.directive.js +84 -15
  14. package/esm2015/directives/string.template.outlet.directive.js +118 -60
  15. package/esm2015/guards/auth.guard.js +117 -30
  16. package/esm2015/interceptors/blob.interceptor.js +67 -28
  17. package/esm2015/interceptors/cache.interceptor.js +46 -14
  18. package/esm2015/interceptors/error.interceptor.js +104 -12
  19. package/esm2015/interceptors/public-api.js +2 -1
  20. package/esm2015/interceptors/token.interceptor.js +86 -42
  21. package/esm2015/interceptors/types.js +5 -0
  22. package/esm2015/utils/array.js +42 -22
  23. package/esm2015/utils/dom.js +29 -11
  24. package/esm2015/utils/form.js +44 -13
  25. package/esm2015/utils/store.js +101 -26
  26. package/fesm2015/reskin-core-directives.js +269 -103
  27. package/fesm2015/reskin-core-directives.js.map +1 -1
  28. package/fesm2015/reskin-core-guards.js +116 -29
  29. package/fesm2015/reskin-core-guards.js.map +1 -1
  30. package/fesm2015/reskin-core-interceptors.js +302 -91
  31. package/fesm2015/reskin-core-interceptors.js.map +1 -1
  32. package/fesm2015/reskin-core-utils.js +212 -68
  33. package/fesm2015/reskin-core-utils.js.map +1 -1
  34. package/guards/auth.guard.d.ts +85 -5
  35. package/interceptors/blob.interceptor.d.ts +30 -3
  36. package/interceptors/cache.interceptor.d.ts +28 -4
  37. package/interceptors/error.interceptor.d.ts +43 -2
  38. package/interceptors/public-api.d.ts +1 -0
  39. package/interceptors/token.interceptor.d.ts +41 -12
  40. package/interceptors/types.d.ts +68 -0
  41. package/package.json +1 -1
  42. package/utils/array.d.ts +8 -1
  43. package/utils/dom.d.ts +32 -5
  44. package/utils/form.d.ts +37 -2
  45. package/utils/store.d.ts +56 -15
@@ -1,20 +1,77 @@
1
1
  import { OnChanges, SimpleChanges, TemplateRef, ViewContainerRef } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
- export declare class RkStringTemplateOutletDirective<_T = unknown> implements OnChanges {
4
- private viewContainer;
3
+ /**
4
+ * 字符串模板输出上下文
5
+ */
6
+ export declare class RkStringTemplateOutletContext<T = any> {
7
+ /** 隐式上下文值 */
8
+ $implicit: T | null;
9
+ }
10
+ /**
11
+ * 字符串模板输出指令
12
+ * 支持动态渲染字符串或 TemplateRef
13
+ *
14
+ * @example
15
+ * ```html
16
+ * <!-- 渲染字符串 -->
17
+ * <ng-container *rkStringTemplateOutlet="'Hello World'">
18
+ * <ng-template>默认模板</ng-template>
19
+ * </ng-container>
20
+ *
21
+ * <!-- 渲染 TemplateRef -->
22
+ * <ng-container *rkStringTemplateOutlet="customTemplate; context: { $implicit: data }">
23
+ * </ng-container>
24
+ * <ng-template #customTemplate let-item>
25
+ * <div>{{ item }}</div>
26
+ * </ng-template>
27
+ *
28
+ * <!-- 动态切换 -->
29
+ * <ng-container *rkStringTemplateOutlet="isTemplate ? templateRef : 'Static Text'">
30
+ * </ng-container>
31
+ * ```
32
+ */
33
+ export declare class RkStringTemplateOutletDirective<T = unknown> implements OnChanges {
34
+ private viewContainerRef;
5
35
  private templateRef;
6
36
  private embeddedViewRef;
7
37
  private context;
8
- rkStringTemplateOutletContext: any | null;
9
- rkStringTemplateOutlet: any | TemplateRef<any>;
10
- static rkTemplateContextGuard<T>(_dir: RkStringTemplateOutletDirective<T>, _ctx: any): _ctx is RkStringTemplateOutletContext;
38
+ /**
39
+ * 输出内容
40
+ * 可以是字符串或 TemplateRef
41
+ */
42
+ rkStringTemplateOutlet: T | TemplateRef<RkStringTemplateOutletContext<T>> | null;
43
+ /**
44
+ * 模板上下文
45
+ * 当 rkStringTemplateOutlet 是 TemplateRef 时使用
46
+ */
47
+ rkStringTemplateOutletContext: RkStringTemplateOutletContext<T> | null;
48
+ /**
49
+ * 类型守卫
50
+ * 用于 Angular 模板类型检查
51
+ */
52
+ static rkTemplateContextGuard<T>(_dir: RkStringTemplateOutletDirective<T>, _ctx: any): _ctx is RkStringTemplateOutletContext<T>;
53
+ constructor(viewContainerRef: ViewContainerRef, templateRef: TemplateRef<RkStringTemplateOutletContext<T>>);
54
+ ngOnChanges(changes: SimpleChanges): void;
55
+ /**
56
+ * 判断是否需要重建视图
57
+ */
58
+ private shouldRecreateView;
59
+ /**
60
+ * 检查 outlet 类型是否变化
61
+ */
62
+ private isOutletTypeChanged;
63
+ /**
64
+ * 检查上下文结构是否变化
65
+ */
66
+ private isContextShapeChanged;
67
+ /**
68
+ * 重建视图
69
+ */
11
70
  private recreateView;
71
+ /**
72
+ * 更新视图上下文
73
+ */
12
74
  private updateContext;
13
- constructor(viewContainer: ViewContainerRef, templateRef: TemplateRef<any>);
14
- ngOnChanges(changes: SimpleChanges): void;
15
75
  static ɵfac: i0.ɵɵFactoryDeclaration<RkStringTemplateOutletDirective<any>, never>;
16
- static ɵdir: i0.ɵɵDirectiveDeclaration<RkStringTemplateOutletDirective<any>, "[rkStringTemplateOutlet]", never, { "rkStringTemplateOutletContext": "rkStringTemplateOutletContext"; "rkStringTemplateOutlet": "rkStringTemplateOutlet"; }, {}, never>;
17
- }
18
- export declare class RkStringTemplateOutletContext {
19
- $implicit: any;
76
+ static ɵdir: i0.ɵɵDirectiveDeclaration<RkStringTemplateOutletDirective<any>, "[rkStringTemplateOutlet]", never, { "rkStringTemplateOutlet": "rkStringTemplateOutlet"; "rkStringTemplateOutletContext": "rkStringTemplateOutletContext"; }, {}, never>;
20
77
  }
@@ -1,62 +1,103 @@
1
1
  import { Directive, Input } from '@angular/core';
2
- import { map } from 'rxjs/operators';
2
+ import { Subject } from 'rxjs';
3
+ import { map, takeUntil } from 'rxjs/operators';
3
4
  import * as i0 from "@angular/core";
4
5
  import * as i1 from "@reskin/core/services";
6
+ /**
7
+ * 权限指令
8
+ * 根据权限标识符控制模板的显示与隐藏
9
+ *
10
+ * @example
11
+ * ```html
12
+ * <!-- 基础用法 -->
13
+ * <div *rkAuth="'user:edit'">有编辑权限才显示</div>
14
+ *
15
+ * <!-- 多个权限(或关系) -->
16
+ * <div *rkAuth="['user:edit', 'user:delete']">有任一权限即显示</div>
17
+ *
18
+ * <!-- 使用 then/else 模板 -->
19
+ * <ng-container *rkAuth="'user:edit'; then hasAuth; else noAuth"></ng-container>
20
+ * <ng-template #hasAuth let-menu>
21
+ * <p>有权限,菜单信息:{{ menu | json }}</p>
22
+ * </ng-template>
23
+ * <ng-template #noAuth>
24
+ * <p>无权限</p>
25
+ * </ng-template>
26
+ * ```
27
+ */
5
28
  export class RkAuthDirective {
6
- constructor(menu, viewContainerRef, templateRef) {
7
- this.menu = menu;
29
+ constructor(menuService, viewContainerRef, templateRef) {
30
+ this.menuService = menuService;
8
31
  this.viewContainerRef = viewContainerRef;
9
32
  this.templateRef = templateRef;
10
33
  this.condition = '';
34
+ this.destroy$ = new Subject();
11
35
  this.thenTemplateRef = templateRef;
12
36
  }
13
37
  /**
14
- * 传递多个标识符时,逗号分隔,匹配按"或者"运算
15
- * @param condition
38
+ * 权限标识符
39
+ * 传递多个标识符时,匹配按"或"运算
16
40
  */
17
41
  set rkAuth(condition) {
18
42
  this.condition = condition;
19
43
  this.updateView();
20
44
  }
45
+ /**
46
+ * 有权限时显示的模板
47
+ */
21
48
  set rkAuthThen(templateRef) {
22
49
  this.thenTemplateRef = templateRef;
23
50
  this.updateView();
24
51
  }
52
+ /**
53
+ * 无权限时显示的模板
54
+ */
25
55
  set rkAuthElse(templateRef) {
26
56
  this.elseTemplateRef = templateRef;
27
57
  this.updateView();
28
58
  }
29
59
  ngOnDestroy() {
30
- var _a;
31
- (_a = this.authSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
60
+ this.destroy$.next();
61
+ this.destroy$.complete();
32
62
  }
63
+ /**
64
+ * 更新视图显示
65
+ */
33
66
  updateView() {
34
- var _a;
35
- (_a = this.authSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
36
- this.authSubscription = this.hasAuth(this.condition).subscribe((context) => {
67
+ if (!this.condition) {
37
68
  this.viewContainerRef.clear();
38
- if (context.status) {
39
- if (this.thenTemplateRef) {
40
- this.viewContainerRef.createEmbeddedView(this.thenTemplateRef, context);
41
- }
42
- }
43
- else {
44
- if (this.elseTemplateRef) {
45
- this.viewContainerRef.createEmbeddedView(this.elseTemplateRef, context);
46
- }
47
- }
69
+ return;
70
+ }
71
+ this.checkAuth(this.condition)
72
+ .pipe(takeUntil(this.destroy$))
73
+ .subscribe((context) => {
74
+ this.renderView(context);
48
75
  });
49
76
  }
50
- hasAuth(tags) {
51
- const tagList = typeof tags === 'string' ? [tags] : tags;
52
- return this.menu.requestData().pipe(map((json) => {
53
- const menus = json.data.filter((menu) => tagList.includes(menu.SYSTEM_RESOURCE_GUARD_ID));
54
- if (!menus.length)
55
- return { $implicit: [], status: false };
56
- return {
57
- $implicit: typeof tags === 'string' ? menus[0] : menus,
58
- status: !!menus.length,
77
+ /**
78
+ * 渲染视图
79
+ */
80
+ renderView(context) {
81
+ this.viewContainerRef.clear();
82
+ if (context.status && this.thenTemplateRef) {
83
+ this.viewContainerRef.createEmbeddedView(this.thenTemplateRef, context);
84
+ }
85
+ else if (!context.status && this.elseTemplateRef) {
86
+ this.viewContainerRef.createEmbeddedView(this.elseTemplateRef, context);
87
+ }
88
+ }
89
+ /**
90
+ * 检查权限
91
+ */
92
+ checkAuth(tags) {
93
+ const tagList = Array.isArray(tags) ? tags : [tags];
94
+ return this.menuService.requestData().pipe(map((response) => {
95
+ const matchedMenus = response.data.filter((menu) => tagList.includes(menu.SYSTEM_RESOURCE_GUARD_ID));
96
+ const context = {
97
+ $implicit: Array.isArray(tags) ? matchedMenus : matchedMenus[0] || null,
98
+ status: matchedMenus.length > 0,
59
99
  };
100
+ return context;
60
101
  }));
61
102
  }
62
103
  }
@@ -74,4 +115,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImpo
74
115
  }], rkAuthElse: [{
75
116
  type: Input
76
117
  }] } });
77
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJyYXJ5L2NvcmUvZGlyZWN0aXZlcy9hdXRoLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBNEMsTUFBTSxlQUFlLENBQUM7QUFFM0YsT0FBTyxFQUFFLEdBQUcsRUFBRSxNQUFNLGdCQUFnQixDQUFDOzs7QUFNckMsTUFBTSxPQUFPLGVBQWU7SUF1QnhCLFlBQ1ksSUFBbUIsRUFDbkIsZ0JBQWtDLEVBQ25DLFdBQTZCO1FBRjVCLFNBQUksR0FBSixJQUFJLENBQWU7UUFDbkIscUJBQWdCLEdBQWhCLGdCQUFnQixDQUFrQjtRQUNuQyxnQkFBVyxHQUFYLFdBQVcsQ0FBa0I7UUFLaEMsY0FBUyxHQUFzQixFQUFFLENBQUM7UUFIdEMsSUFBSSxDQUFDLGVBQWUsR0FBRyxXQUFXLENBQUM7SUFDdkMsQ0FBQztJQTVCRDs7O09BR0c7SUFDSCxJQUNJLE1BQU0sQ0FBQyxTQUE0QjtRQUNuQyxJQUFJLENBQUMsU0FBUyxHQUFHLFNBQVMsQ0FBQztRQUMzQixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDdEIsQ0FBQztJQUVELElBQ0ksVUFBVSxDQUFDLFdBQTZCO1FBQ3hDLElBQUksQ0FBQyxlQUFlLEdBQUcsV0FBVyxDQUFDO1FBQ25DLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBRUQsSUFDSSxVQUFVLENBQUMsV0FBNkI7UUFDeEMsSUFBSSxDQUFDLGVBQWUsR0FBRyxXQUFXLENBQUM7UUFDbkMsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUFlRCxXQUFXOztRQUNQLE1BQUEsSUFBSSxDQUFDLGdCQUFnQiwwQ0FBRSxXQUFXLEVBQUUsQ0FBQztJQUN6QyxDQUFDO0lBRU8sVUFBVTs7UUFDZCxNQUFBLElBQUksQ0FBQyxnQkFBZ0IsMENBQUUsV0FBVyxFQUFFLENBQUM7UUFDckMsSUFBSSxDQUFDLGdCQUFnQixHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFO1lBQ3ZFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUM5QixJQUFJLE9BQU8sQ0FBQyxNQUFNLEVBQUU7Z0JBQ2hCLElBQUksSUFBSSxDQUFDLGVBQWUsRUFBRTtvQkFDdEIsSUFBSSxDQUFDLGdCQUFnQixDQUFDLGtCQUFrQixDQUFDLElBQUksQ0FBQyxlQUFlLEVBQUUsT0FBTyxDQUFDLENBQUM7aUJBQzNFO2FBQ0o7aUJBQU07Z0JBQ0gsSUFBSSxJQUFJLENBQUMsZUFBZSxFQUFFO29CQUN0QixJQUFJLENBQUMsZ0JBQWdCLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLGVBQWUsRUFBRSxPQUFPLENBQUMsQ0FBQztpQkFDM0U7YUFDSjtRQUNMLENBQUMsQ0FBQyxDQUFDO0lBQ1AsQ0FBQztJQUVPLE9BQU8sQ0FBQyxJQUF1QjtRQUNuQyxNQUFNLE9BQU8sR0FBYSxPQUFPLElBQUksS0FBSyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQztRQUNuRSxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUMsSUFBSSxDQUMvQixHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRTtZQUNULE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsSUFBSSxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyx3QkFBd0IsQ0FBQyxDQUFDLENBQUM7WUFDMUYsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNO2dCQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsRUFBRSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsQ0FBQztZQUMzRCxPQUFPO2dCQUNILFNBQVMsRUFBRSxPQUFPLElBQUksS0FBSyxRQUFRLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsS0FBSztnQkFDdEQsTUFBTSxFQUFFLENBQUMsQ0FBQyxLQUFLLENBQUMsTUFBTTthQUN6QixDQUFDO1FBQ04sQ0FBQyxDQUFDLENBQ0wsQ0FBQztJQUNOLENBQUM7OzZHQXBFUSxlQUFlO2lHQUFmLGVBQWU7NEZBQWYsZUFBZTtrQkFIM0IsU0FBUzttQkFBQztvQkFDUCxRQUFRLEVBQUUsVUFBVTtpQkFDdkI7NkpBT08sTUFBTTtzQkFEVCxLQUFLO2dCQU9GLFVBQVU7c0JBRGIsS0FBSztnQkFPRixVQUFVO3NCQURiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIElucHV0LCBPbkRlc3Ryb3ksIFRlbXBsYXRlUmVmLCBWaWV3Q29udGFpbmVyUmVmIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IFN1YnNjcmlwdGlvbiB9IGZyb20gJ3J4anMnO1xyXG5pbXBvcnQgeyBtYXAgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XHJcbmltcG9ydCB7IFJrTWVudVNlcnZpY2UgfSBmcm9tICdAcmVza2luL2NvcmUvc2VydmljZXMnO1xyXG5cclxuQERpcmVjdGl2ZSh7XHJcbiAgICBzZWxlY3RvcjogJ1tya0F1dGhdJyxcclxufSlcclxuZXhwb3J0IGNsYXNzIFJrQXV0aERpcmVjdGl2ZSBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XHJcbiAgICAvKipcclxuICAgICAqIOS8oOmAkuWkmuS4quagh+ivhuespuaXtu+8jOmAl+WPt+WIhumalO+8jOWMuemFjeaMiVwi5oiW6ICFXCLov5DnrpdcclxuICAgICAqIEBwYXJhbSBjb25kaXRpb25cclxuICAgICAqL1xyXG4gICAgQElucHV0KClcclxuICAgIHNldCBya0F1dGgoY29uZGl0aW9uOiBzdHJpbmcgfCBzdHJpbmdbXSkge1xyXG4gICAgICAgIHRoaXMuY29uZGl0aW9uID0gY29uZGl0aW9uO1xyXG4gICAgICAgIHRoaXMudXBkYXRlVmlldygpO1xyXG4gICAgfVxyXG5cclxuICAgIEBJbnB1dCgpXHJcbiAgICBzZXQgcmtBdXRoVGhlbih0ZW1wbGF0ZVJlZjogVGVtcGxhdGVSZWY8YW55Pikge1xyXG4gICAgICAgIHRoaXMudGhlblRlbXBsYXRlUmVmID0gdGVtcGxhdGVSZWY7XHJcbiAgICAgICAgdGhpcy51cGRhdGVWaWV3KCk7XHJcbiAgICB9XHJcblxyXG4gICAgQElucHV0KClcclxuICAgIHNldCBya0F1dGhFbHNlKHRlbXBsYXRlUmVmOiBUZW1wbGF0ZVJlZjxhbnk+KSB7XHJcbiAgICAgICAgdGhpcy5lbHNlVGVtcGxhdGVSZWYgPSB0ZW1wbGF0ZVJlZjtcclxuICAgICAgICB0aGlzLnVwZGF0ZVZpZXcoKTtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdHJ1Y3RvcihcclxuICAgICAgICBwcml2YXRlIG1lbnU6IFJrTWVudVNlcnZpY2UsXHJcbiAgICAgICAgcHJpdmF0ZSB2aWV3Q29udGFpbmVyUmVmOiBWaWV3Q29udGFpbmVyUmVmLFxyXG4gICAgICAgIHB1YmxpYyB0ZW1wbGF0ZVJlZjogVGVtcGxhdGVSZWY8YW55PixcclxuICAgICkge1xyXG4gICAgICAgIHRoaXMudGhlblRlbXBsYXRlUmVmID0gdGVtcGxhdGVSZWY7XHJcbiAgICB9XHJcblxyXG4gICAgcHJpdmF0ZSBjb25kaXRpb246IHN0cmluZyB8IHN0cmluZ1tdID0gJyc7XHJcbiAgICBwcml2YXRlIHRoZW5UZW1wbGF0ZVJlZj86IFRlbXBsYXRlUmVmPGFueT47XHJcbiAgICBwcml2YXRlIGVsc2VUZW1wbGF0ZVJlZj86IFRlbXBsYXRlUmVmPGFueT47XHJcbiAgICBwcml2YXRlIGF1dGhTdWJzY3JpcHRpb24/OiBTdWJzY3JpcHRpb247XHJcblxyXG4gICAgbmdPbkRlc3Ryb3koKSB7XHJcbiAgICAgICAgdGhpcy5hdXRoU3Vic2NyaXB0aW9uPy51bnN1YnNjcmliZSgpO1xyXG4gICAgfVxyXG5cclxuICAgIHByaXZhdGUgdXBkYXRlVmlldygpIHtcclxuICAgICAgICB0aGlzLmF1dGhTdWJzY3JpcHRpb24/LnVuc3Vic2NyaWJlKCk7XHJcbiAgICAgICAgdGhpcy5hdXRoU3Vic2NyaXB0aW9uID0gdGhpcy5oYXNBdXRoKHRoaXMuY29uZGl0aW9uKS5zdWJzY3JpYmUoKGNvbnRleHQpID0+IHtcclxuICAgICAgICAgICAgdGhpcy52aWV3Q29udGFpbmVyUmVmLmNsZWFyKCk7XHJcbiAgICAgICAgICAgIGlmIChjb250ZXh0LnN0YXR1cykge1xyXG4gICAgICAgICAgICAgICAgaWYgKHRoaXMudGhlblRlbXBsYXRlUmVmKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgdGhpcy52aWV3Q29udGFpbmVyUmVmLmNyZWF0ZUVtYmVkZGVkVmlldyh0aGlzLnRoZW5UZW1wbGF0ZVJlZiwgY29udGV4dCk7XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBpZiAodGhpcy5lbHNlVGVtcGxhdGVSZWYpIHtcclxuICAgICAgICAgICAgICAgICAgICB0aGlzLnZpZXdDb250YWluZXJSZWYuY3JlYXRlRW1iZWRkZWRWaWV3KHRoaXMuZWxzZVRlbXBsYXRlUmVmLCBjb250ZXh0KTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfVxyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG5cclxuICAgIHByaXZhdGUgaGFzQXV0aCh0YWdzOiBzdHJpbmcgfCBzdHJpbmdbXSkge1xyXG4gICAgICAgIGNvbnN0IHRhZ0xpc3Q6IHN0cmluZ1tdID0gdHlwZW9mIHRhZ3MgPT09ICdzdHJpbmcnID8gW3RhZ3NdIDogdGFncztcclxuICAgICAgICByZXR1cm4gdGhpcy5tZW51LnJlcXVlc3REYXRhKCkucGlwZShcclxuICAgICAgICAgICAgbWFwKChqc29uKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBtZW51cyA9IGpzb24uZGF0YS5maWx0ZXIoKG1lbnUpID0+IHRhZ0xpc3QuaW5jbHVkZXMobWVudS5TWVNURU1fUkVTT1VSQ0VfR1VBUkRfSUQpKTtcclxuICAgICAgICAgICAgICAgIGlmICghbWVudXMubGVuZ3RoKSByZXR1cm4geyAkaW1wbGljaXQ6IFtdLCBzdGF0dXM6IGZhbHNlIH07XHJcbiAgICAgICAgICAgICAgICByZXR1cm4ge1xyXG4gICAgICAgICAgICAgICAgICAgICRpbXBsaWNpdDogdHlwZW9mIHRhZ3MgPT09ICdzdHJpbmcnID8gbWVudXNbMF0gOiBtZW51cyxcclxuICAgICAgICAgICAgICAgICAgICBzdGF0dXM6ICEhbWVudXMubGVuZ3RoLFxyXG4gICAgICAgICAgICAgICAgfTtcclxuICAgICAgICAgICAgfSksXHJcbiAgICAgICAgKTtcclxuICAgIH1cclxufVxyXG4iXX0=
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.directive.js","sourceRoot":"","sources":["../../../../library/core/directives/auth.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, Input, OnDestroy, TemplateRef, ViewContainerRef } from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { map, takeUntil } from 'rxjs/operators';\r\nimport { RkMenuService } from '@reskin/core/services';\r\n\r\n/**\r\n * 权限指令上下文\r\n */\r\nexport interface RkAuthContext<T = any> {\r\n    /** 隐式上下文，包含匹配的菜单数据 */\r\n    $implicit: T;\r\n    /** 权限验证状态 */\r\n    status: boolean;\r\n}\r\n\r\n/**\r\n * 权限指令\r\n * 根据权限标识符控制模板的显示与隐藏\r\n *\r\n * @example\r\n * ```html\r\n * <!-- 基础用法 -->\r\n * <div *rkAuth=\"'user:edit'\">有编辑权限才显示</div>\r\n *\r\n * <!-- 多个权限（或关系） -->\r\n * <div *rkAuth=\"['user:edit', 'user:delete']\">有任一权限即显示</div>\r\n *\r\n * <!-- 使用 then/else 模板 -->\r\n * <ng-container *rkAuth=\"'user:edit'; then hasAuth; else noAuth\"></ng-container>\r\n * <ng-template #hasAuth let-menu>\r\n *   <p>有权限，菜单信息：{{ menu | json }}</p>\r\n * </ng-template>\r\n * <ng-template #noAuth>\r\n *   <p>无权限</p>\r\n * </ng-template>\r\n * ```\r\n */\r\n@Directive({\r\n    selector: '[rkAuth]',\r\n})\r\nexport class RkAuthDirective implements OnDestroy {\r\n    private condition: string | string[] = '';\r\n    private thenTemplateRef?: TemplateRef<any>;\r\n    private elseTemplateRef?: TemplateRef<any>;\r\n    private readonly destroy$ = new Subject<void>();\r\n\r\n    /**\r\n     * 权限标识符\r\n     * 传递多个标识符时，匹配按\"或\"运算\r\n     */\r\n    @Input()\r\n    set rkAuth(condition: string | string[]) {\r\n        this.condition = condition;\r\n        this.updateView();\r\n    }\r\n\r\n    /**\r\n     * 有权限时显示的模板\r\n     */\r\n    @Input()\r\n    set rkAuthThen(templateRef: TemplateRef<any>) {\r\n        this.thenTemplateRef = templateRef;\r\n        this.updateView();\r\n    }\r\n\r\n    /**\r\n     * 无权限时显示的模板\r\n     */\r\n    @Input()\r\n    set rkAuthElse(templateRef: TemplateRef<any>) {\r\n        this.elseTemplateRef = templateRef;\r\n        this.updateView();\r\n    }\r\n\r\n    constructor(\r\n        private menuService: RkMenuService,\r\n        private viewContainerRef: ViewContainerRef,\r\n        public templateRef: TemplateRef<any>,\r\n    ) {\r\n        this.thenTemplateRef = templateRef;\r\n    }\r\n\r\n    ngOnDestroy(): void {\r\n        this.destroy$.next();\r\n        this.destroy$.complete();\r\n    }\r\n\r\n    /**\r\n     * 更新视图显示\r\n     */\r\n    private updateView(): void {\r\n        if (!this.condition) {\r\n            this.viewContainerRef.clear();\r\n            return;\r\n        }\r\n\r\n        this.checkAuth(this.condition)\r\n            .pipe(takeUntil(this.destroy$))\r\n            .subscribe((context) => {\r\n                this.renderView(context);\r\n            });\r\n    }\r\n\r\n    /**\r\n     * 渲染视图\r\n     */\r\n    private renderView(context: RkAuthContext): void {\r\n        this.viewContainerRef.clear();\r\n\r\n        if (context.status && this.thenTemplateRef) {\r\n            this.viewContainerRef.createEmbeddedView(this.thenTemplateRef, context);\r\n        } else if (!context.status && this.elseTemplateRef) {\r\n            this.viewContainerRef.createEmbeddedView(this.elseTemplateRef, context);\r\n        }\r\n    }\r\n\r\n    /**\r\n     * 检查权限\r\n     */\r\n    private checkAuth(tags: string | string[]) {\r\n        const tagList: string[] = Array.isArray(tags) ? tags : [tags];\r\n\r\n        return this.menuService.requestData().pipe(\r\n            map((response) => {\r\n                const matchedMenus = response.data.filter((menu) => tagList.includes(menu.SYSTEM_RESOURCE_GUARD_ID));\r\n\r\n                const context: RkAuthContext = {\r\n                    $implicit: Array.isArray(tags) ? matchedMenus : matchedMenus[0] || null,\r\n                    status: matchedMenus.length > 0,\r\n                };\r\n\r\n                return context;\r\n            }),\r\n        );\r\n    }\r\n}\r\n"]}
@@ -1,31 +1,100 @@
1
1
  import { Directive, Input, Output, EventEmitter } from '@angular/core';
2
+ import { Subject } from 'rxjs';
3
+ import { takeUntil } from 'rxjs/operators';
2
4
  import * as i0 from "@angular/core";
3
5
  import * as i1 from "@reskin/core/services";
6
+ /**
7
+ * 样式按需加载指令
8
+ * 在指令初始化时加载指定的样式文件,销毁时自动卸载
9
+ *
10
+ * @example
11
+ * ```html
12
+ * <!-- 加载单个样式 -->
13
+ * <div [rkLoadStyles]="'assets/styles/theme.css'">内容</div>
14
+ *
15
+ * <!-- 加载多个样式 -->
16
+ * <div [rkLoadStyles]="['assets/styles/theme.css', 'assets/styles/components.css']">内容</div>
17
+ *
18
+ * <!-- 监听加载完成事件 -->
19
+ * <div [rkLoadStyles]="styleUrls" (stylesLoaded)="onLoaded($event)">内容</div>
20
+ * ```
21
+ */
4
22
  export class RkLoadStylesDirective {
5
- constructor(styleLoader) {
6
- this.styleLoader = styleLoader;
23
+ constructor(styleLoaderService) {
24
+ this.styleLoaderService = styleLoaderService;
25
+ this.destroy$ = new Subject();
26
+ this.styleUrls = [];
27
+ /**
28
+ * 样式文件路径
29
+ * 可以是单个路径字符串或路径数组
30
+ */
7
31
  this.hrefs = [];
8
- // (可选) 输出加载完成事件,以便父组件可以响应
32
+ /**
33
+ * 样式加载完成事件
34
+ * 返回每个样式文件的加载结果
35
+ */
9
36
  this.stylesLoaded = new EventEmitter();
10
- this.styleUrls = [];
11
37
  }
12
38
  ngOnInit() {
13
- this.styleUrls = Array.isArray(this.hrefs) ? this.hrefs : [this.hrefs];
14
- if (this.styleUrls.length > 0) {
15
- this.subscription = this.styleLoader.load(...this.styleUrls).subscribe((results) => {
16
- this.stylesLoaded.emit(results);
17
- });
18
- }
39
+ this.loadStyles();
19
40
  }
20
41
  ngOnDestroy() {
21
- if (this.subscription) {
22
- this.subscription.unsubscribe();
42
+ this.destroy$.next();
43
+ this.destroy$.complete();
44
+ this.unloadStyles();
45
+ }
46
+ /**
47
+ * 加载样式文件
48
+ */
49
+ loadStyles() {
50
+ this.styleUrls = this.normalizeHrefs(this.hrefs);
51
+ if (this.styleUrls.length === 0) {
52
+ return;
23
53
  }
24
- // 指令销毁时,自动调用 unload,服务内部会处理引用计数
54
+ this.styleLoaderService
55
+ .load(...this.styleUrls)
56
+ .pipe(takeUntil(this.destroy$))
57
+ .subscribe({
58
+ next: (results) => {
59
+ this.stylesLoaded.emit(results);
60
+ this.logLoadResults(results);
61
+ },
62
+ error: (error) => {
63
+ console.error('[RkLoadStyles] 样式加载失败:', error);
64
+ },
65
+ });
66
+ }
67
+ /**
68
+ * 卸载样式文件
69
+ */
70
+ unloadStyles() {
25
71
  if (this.styleUrls.length > 0) {
26
- this.styleLoader.unload(...this.styleUrls);
72
+ this.styleLoaderService.unload(...this.styleUrls);
27
73
  }
28
74
  }
75
+ /**
76
+ * 标准化 hrefs 输入为数组
77
+ */
78
+ normalizeHrefs(hrefs) {
79
+ if (!hrefs) {
80
+ return [];
81
+ }
82
+ return Array.isArray(hrefs) ? hrefs : [hrefs];
83
+ }
84
+ /**
85
+ * 记录加载结果(仅在开发模式)
86
+ */
87
+ logLoadResults(results) {
88
+ // 开发模式下记录加载结果
89
+ results.forEach((result) => {
90
+ if (result.success) {
91
+ console.log(`[RkLoadStyles] 样式加载成功: ${result.href}`);
92
+ }
93
+ else {
94
+ console.error(`[RkLoadStyles] 样式加载失败: ${result.href}`, result.error);
95
+ }
96
+ });
97
+ }
29
98
  }
30
99
  RkLoadStylesDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: RkLoadStylesDirective, deps: [{ token: i1.RkStyleLoaderService }], target: i0.ɵɵFactoryTarget.Directive });
31
100
  RkLoadStylesDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "12.2.17", type: RkLoadStylesDirective, selector: "[rkLoadStyles]", inputs: { hrefs: ["rkLoadStyles", "hrefs"] }, outputs: { stylesLoaded: "stylesLoaded" }, ngImport: i0 });
@@ -40,4 +109,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImpo
40
109
  }], stylesLoaded: [{
41
110
  type: Output
42
111
  }] } });
43
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC5zdHlsZXMuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vbGlicmFyeS9jb3JlL2RpcmVjdGl2ZXMvbG9hZC5zdHlsZXMuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFxQixNQUFNLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFPMUYsTUFBTSxPQUFPLHFCQUFxQjtJQVM5QixZQUFvQixXQUFpQztRQUFqQyxnQkFBVyxHQUFYLFdBQVcsQ0FBc0I7UUFSOUIsVUFBSyxHQUFzQixFQUFFLENBQUM7UUFFckQsMEJBQTBCO1FBQ2hCLGlCQUFZLEdBQUcsSUFBSSxZQUFZLEVBQWlCLENBQUM7UUFHbkQsY0FBUyxHQUFhLEVBQUUsQ0FBQztJQUV1QixDQUFDO0lBRXpELFFBQVE7UUFDSixJQUFJLENBQUMsU0FBUyxHQUFHLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUN2RSxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtZQUMzQixJQUFJLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFO2dCQUMvRSxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQztZQUNwQyxDQUFDLENBQUMsQ0FBQztTQUNOO0lBQ0wsQ0FBQztJQUVELFdBQVc7UUFDUCxJQUFJLElBQUksQ0FBQyxZQUFZLEVBQUU7WUFDbkIsSUFBSSxDQUFDLFlBQVksQ0FBQyxXQUFXLEVBQUUsQ0FBQztTQUNuQztRQUNELGdDQUFnQztRQUNoQyxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtZQUMzQixJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQztTQUM5QztJQUNMLENBQUM7O21IQTVCUSxxQkFBcUI7dUdBQXJCLHFCQUFxQjs0RkFBckIscUJBQXFCO2tCQUhqQyxTQUFTO21CQUFDO29CQUNQLFFBQVEsRUFBRSxnQkFBZ0I7aUJBQzdCOzJHQUUwQixLQUFLO3NCQUEzQixLQUFLO3VCQUFDLGNBQWM7Z0JBR1gsWUFBWTtzQkFBckIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgSW5wdXQsIE9uSW5pdCwgT25EZXN0cm95LCBPdXRwdXQsIEV2ZW50RW1pdHRlciB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBTdWJzY3JpcHRpb24gfSBmcm9tICdyeGpzJztcclxuaW1wb3J0IHsgSUxvYWRSZXN1bHQsIFJrU3R5bGVMb2FkZXJTZXJ2aWNlIH0gZnJvbSAnQHJlc2tpbi9jb3JlL3NlcnZpY2VzJztcclxuXHJcbkBEaXJlY3RpdmUoe1xyXG4gICAgc2VsZWN0b3I6ICdbcmtMb2FkU3R5bGVzXScsXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBSa0xvYWRTdHlsZXNEaXJlY3RpdmUgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSB7XHJcbiAgICBASW5wdXQoJ3JrTG9hZFN0eWxlcycpIGhyZWZzOiBzdHJpbmdbXSB8IHN0cmluZyA9IFtdO1xyXG5cclxuICAgIC8vICjlj6/pgIkpIOi+k+WHuuWKoOi9veWujOaIkOS6i+S7tu+8jOS7peS+v+eItue7hOS7tuWPr+S7peWTjeW6lFxyXG4gICAgQE91dHB1dCgpIHN0eWxlc0xvYWRlZCA9IG5ldyBFdmVudEVtaXR0ZXI8SUxvYWRSZXN1bHRbXT4oKTtcclxuXHJcbiAgICBwcml2YXRlIHN1YnNjcmlwdGlvbjogU3Vic2NyaXB0aW9uIHwgdW5kZWZpbmVkO1xyXG4gICAgcHJpdmF0ZSBzdHlsZVVybHM6IHN0cmluZ1tdID0gW107XHJcblxyXG4gICAgY29uc3RydWN0b3IocHJpdmF0ZSBzdHlsZUxvYWRlcjogUmtTdHlsZUxvYWRlclNlcnZpY2UpIHt9XHJcblxyXG4gICAgbmdPbkluaXQoKTogdm9pZCB7XHJcbiAgICAgICAgdGhpcy5zdHlsZVVybHMgPSBBcnJheS5pc0FycmF5KHRoaXMuaHJlZnMpID8gdGhpcy5ocmVmcyA6IFt0aGlzLmhyZWZzXTtcclxuICAgICAgICBpZiAodGhpcy5zdHlsZVVybHMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgICAgICB0aGlzLnN1YnNjcmlwdGlvbiA9IHRoaXMuc3R5bGVMb2FkZXIubG9hZCguLi50aGlzLnN0eWxlVXJscykuc3Vic2NyaWJlKChyZXN1bHRzKSA9PiB7XHJcbiAgICAgICAgICAgICAgICB0aGlzLnN0eWxlc0xvYWRlZC5lbWl0KHJlc3VsdHMpO1xyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XHJcbiAgICAgICAgaWYgKHRoaXMuc3Vic2NyaXB0aW9uKSB7XHJcbiAgICAgICAgICAgIHRoaXMuc3Vic2NyaXB0aW9uLnVuc3Vic2NyaWJlKCk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIC8vIOaMh+S7pOmUgOavgeaXtu+8jOiHquWKqOiwg+eUqCB1bmxvYWTvvIzmnI3liqHlhoXpg6jkvJrlpITnkIblvJXnlKjorqHmlbBcclxuICAgICAgICBpZiAodGhpcy5zdHlsZVVybHMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgICAgICB0aGlzLnN0eWxlTG9hZGVyLnVubG9hZCguLi50aGlzLnN0eWxlVXJscyk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG59XHJcbiJdfQ==
112
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9hZC5zdHlsZXMuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vbGlicmFyeS9jb3JlL2RpcmVjdGl2ZXMvbG9hZC5zdHlsZXMuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFxQixNQUFNLEVBQUUsWUFBWSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzFGLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDL0IsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdCQUFnQixDQUFDOzs7QUFHM0M7Ozs7Ozs7Ozs7Ozs7OztHQWVHO0FBSUgsTUFBTSxPQUFPLHFCQUFxQjtJQWdCOUIsWUFBb0Isa0JBQXdDO1FBQXhDLHVCQUFrQixHQUFsQixrQkFBa0IsQ0FBc0I7UUFmM0MsYUFBUSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFDeEMsY0FBUyxHQUFhLEVBQUUsQ0FBQztRQUVqQzs7O1dBR0c7UUFDb0IsVUFBSyxHQUFzQixFQUFFLENBQUM7UUFFckQ7OztXQUdHO1FBQ08saUJBQVksR0FBRyxJQUFJLFlBQVksRUFBaUIsQ0FBQztJQUVJLENBQUM7SUFFaEUsUUFBUTtRQUNKLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztJQUN0QixDQUFDO0lBRUQsV0FBVztRQUNQLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDckIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUN6QixJQUFJLENBQUMsWUFBWSxFQUFFLENBQUM7SUFDeEIsQ0FBQztJQUVEOztPQUVHO0lBQ0ssVUFBVTtRQUNkLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFFakQsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7WUFDN0IsT0FBTztTQUNWO1FBRUQsSUFBSSxDQUFDLGtCQUFrQjthQUNsQixJQUFJLENBQUMsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDO2FBQ3ZCLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO2FBQzlCLFNBQVMsQ0FBQztZQUNQLElBQUksRUFBRSxDQUFDLE9BQU8sRUFBRSxFQUFFO2dCQUNkLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO2dCQUNoQyxJQUFJLENBQUMsY0FBYyxDQUFDLE9BQU8sQ0FBQyxDQUFDO1lBQ2pDLENBQUM7WUFDRCxLQUFLLEVBQUUsQ0FBQyxLQUFLLEVBQUUsRUFBRTtnQkFDYixPQUFPLENBQUMsS0FBSyxDQUFDLHdCQUF3QixFQUFFLEtBQUssQ0FBQyxDQUFDO1lBQ25ELENBQUM7U0FDSixDQUFDLENBQUM7SUFDWCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxZQUFZO1FBQ2hCLElBQUksSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFO1lBQzNCLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxNQUFNLENBQUMsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7U0FDckQ7SUFDTCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxjQUFjLENBQUMsS0FBd0I7UUFDM0MsSUFBSSxDQUFDLEtBQUssRUFBRTtZQUNSLE9BQU8sRUFBRSxDQUFDO1NBQ2I7UUFDRCxPQUFPLEtBQUssQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUNsRCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxjQUFjLENBQUMsT0FBc0I7UUFDekMsY0FBYztRQUNkLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQyxNQUFNLEVBQUUsRUFBRTtZQUN2QixJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUU7Z0JBQ2hCLE9BQU8sQ0FBQyxHQUFHLENBQUMsMEJBQTBCLE1BQU0sQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDO2FBQ3hEO2lCQUFNO2dCQUNILE9BQU8sQ0FBQyxLQUFLLENBQUMsMEJBQTBCLE1BQU0sQ0FBQyxJQUFJLEVBQUUsRUFBRSxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7YUFDeEU7UUFDTCxDQUFDLENBQUMsQ0FBQztJQUNQLENBQUM7O21IQW5GUSxxQkFBcUI7dUdBQXJCLHFCQUFxQjs0RkFBckIscUJBQXFCO2tCQUhqQyxTQUFTO21CQUFDO29CQUNQLFFBQVEsRUFBRSxnQkFBZ0I7aUJBQzdCOzJHQVMwQixLQUFLO3NCQUEzQixLQUFLO3VCQUFDLGNBQWM7Z0JBTVgsWUFBWTtzQkFBckIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgSW5wdXQsIE9uSW5pdCwgT25EZXN0cm95LCBPdXRwdXQsIEV2ZW50RW1pdHRlciB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSAncnhqcyc7XHJcbmltcG9ydCB7IHRha2VVbnRpbCB9IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcclxuaW1wb3J0IHsgSUxvYWRSZXN1bHQsIFJrU3R5bGVMb2FkZXJTZXJ2aWNlIH0gZnJvbSAnQHJlc2tpbi9jb3JlL3NlcnZpY2VzJztcclxuXHJcbi8qKlxyXG4gKiDmoLflvI/mjInpnIDliqDovb3mjIfku6RcclxuICog5Zyo5oyH5Luk5Yid5aeL5YyW5pe25Yqg6L295oyH5a6a55qE5qC35byP5paH5Lu277yM6ZSA5q+B5pe26Ieq5Yqo5Y246L29XHJcbiAqXHJcbiAqIEBleGFtcGxlXHJcbiAqIGBgYGh0bWxcclxuICogPCEtLSDliqDovb3ljZXkuKrmoLflvI8gLS0+XHJcbiAqIDxkaXYgW3JrTG9hZFN0eWxlc109XCInYXNzZXRzL3N0eWxlcy90aGVtZS5jc3MnXCI+5YaF5a65PC9kaXY+XHJcbiAqXHJcbiAqIDwhLS0g5Yqg6L295aSa5Liq5qC35byPIC0tPlxyXG4gKiA8ZGl2IFtya0xvYWRTdHlsZXNdPVwiWydhc3NldHMvc3R5bGVzL3RoZW1lLmNzcycsICdhc3NldHMvc3R5bGVzL2NvbXBvbmVudHMuY3NzJ11cIj7lhoXlrrk8L2Rpdj5cclxuICpcclxuICogPCEtLSDnm5HlkKzliqDovb3lrozmiJDkuovku7YgLS0+XHJcbiAqIDxkaXYgW3JrTG9hZFN0eWxlc109XCJzdHlsZVVybHNcIiAoc3R5bGVzTG9hZGVkKT1cIm9uTG9hZGVkKCRldmVudClcIj7lhoXlrrk8L2Rpdj5cclxuICogYGBgXHJcbiAqL1xyXG5ARGlyZWN0aXZlKHtcclxuICAgIHNlbGVjdG9yOiAnW3JrTG9hZFN0eWxlc10nLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgUmtMb2FkU3R5bGVzRGlyZWN0aXZlIGltcGxlbWVudHMgT25Jbml0LCBPbkRlc3Ryb3kge1xyXG4gICAgcHJpdmF0ZSByZWFkb25seSBkZXN0cm95JCA9IG5ldyBTdWJqZWN0PHZvaWQ+KCk7XHJcbiAgICBwcml2YXRlIHN0eWxlVXJsczogc3RyaW5nW10gPSBbXTtcclxuXHJcbiAgICAvKipcclxuICAgICAqIOagt+W8j+aWh+S7tui3r+W+hFxyXG4gICAgICog5Y+v5Lul5piv5Y2V5Liq6Lev5b6E5a2X56ym5Liy5oiW6Lev5b6E5pWw57uEXHJcbiAgICAgKi9cclxuICAgIEBJbnB1dCgncmtMb2FkU3R5bGVzJykgaHJlZnM6IHN0cmluZyB8IHN0cmluZ1tdID0gW107XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiDmoLflvI/liqDovb3lrozmiJDkuovku7ZcclxuICAgICAqIOi/lOWbnuavj+S4quagt+W8j+aWh+S7tueahOWKoOi9vee7k+aenFxyXG4gICAgICovXHJcbiAgICBAT3V0cHV0KCkgc3R5bGVzTG9hZGVkID0gbmV3IEV2ZW50RW1pdHRlcjxJTG9hZFJlc3VsdFtdPigpO1xyXG5cclxuICAgIGNvbnN0cnVjdG9yKHByaXZhdGUgc3R5bGVMb2FkZXJTZXJ2aWNlOiBSa1N0eWxlTG9hZGVyU2VydmljZSkge31cclxuXHJcbiAgICBuZ09uSW5pdCgpOiB2b2lkIHtcclxuICAgICAgICB0aGlzLmxvYWRTdHlsZXMoKTtcclxuICAgIH1cclxuXHJcbiAgICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcclxuICAgICAgICB0aGlzLmRlc3Ryb3kkLm5leHQoKTtcclxuICAgICAgICB0aGlzLmRlc3Ryb3kkLmNvbXBsZXRlKCk7XHJcbiAgICAgICAgdGhpcy51bmxvYWRTdHlsZXMoKTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIOWKoOi9veagt+W8j+aWh+S7tlxyXG4gICAgICovXHJcbiAgICBwcml2YXRlIGxvYWRTdHlsZXMoKTogdm9pZCB7XHJcbiAgICAgICAgdGhpcy5zdHlsZVVybHMgPSB0aGlzLm5vcm1hbGl6ZUhyZWZzKHRoaXMuaHJlZnMpO1xyXG5cclxuICAgICAgICBpZiAodGhpcy5zdHlsZVVybHMubGVuZ3RoID09PSAwKSB7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuc3R5bGVMb2FkZXJTZXJ2aWNlXHJcbiAgICAgICAgICAgIC5sb2FkKC4uLnRoaXMuc3R5bGVVcmxzKVxyXG4gICAgICAgICAgICAucGlwZSh0YWtlVW50aWwodGhpcy5kZXN0cm95JCkpXHJcbiAgICAgICAgICAgIC5zdWJzY3JpYmUoe1xyXG4gICAgICAgICAgICAgICAgbmV4dDogKHJlc3VsdHMpID0+IHtcclxuICAgICAgICAgICAgICAgICAgICB0aGlzLnN0eWxlc0xvYWRlZC5lbWl0KHJlc3VsdHMpO1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMubG9nTG9hZFJlc3VsdHMocmVzdWx0cyk7XHJcbiAgICAgICAgICAgICAgICB9LFxyXG4gICAgICAgICAgICAgICAgZXJyb3I6IChlcnJvcikgPT4ge1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoJ1tSa0xvYWRTdHlsZXNdIOagt+W8j+WKoOi9veWksei0pTonLCBlcnJvcik7XHJcbiAgICAgICAgICAgICAgICB9LFxyXG4gICAgICAgICAgICB9KTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIOWNuOi9veagt+W8j+aWh+S7tlxyXG4gICAgICovXHJcbiAgICBwcml2YXRlIHVubG9hZFN0eWxlcygpOiB2b2lkIHtcclxuICAgICAgICBpZiAodGhpcy5zdHlsZVVybHMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgICAgICB0aGlzLnN0eWxlTG9hZGVyU2VydmljZS51bmxvYWQoLi4udGhpcy5zdHlsZVVybHMpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIOagh+WHhuWMliBocmVmcyDovpPlhaXkuLrmlbDnu4RcclxuICAgICAqL1xyXG4gICAgcHJpdmF0ZSBub3JtYWxpemVIcmVmcyhocmVmczogc3RyaW5nIHwgc3RyaW5nW10pOiBzdHJpbmdbXSB7XHJcbiAgICAgICAgaWYgKCFocmVmcykge1xyXG4gICAgICAgICAgICByZXR1cm4gW107XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHJldHVybiBBcnJheS5pc0FycmF5KGhyZWZzKSA/IGhyZWZzIDogW2hyZWZzXTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIOiusOW9leWKoOi9vee7k+aenO+8iOS7heWcqOW8gOWPkeaooeW8j++8iVxyXG4gICAgICovXHJcbiAgICBwcml2YXRlIGxvZ0xvYWRSZXN1bHRzKHJlc3VsdHM6IElMb2FkUmVzdWx0W10pOiB2b2lkIHtcclxuICAgICAgICAvLyDlvIDlj5HmqKHlvI/kuIvorrDlvZXliqDovb3nu5PmnpxcclxuICAgICAgICByZXN1bHRzLmZvckVhY2goKHJlc3VsdCkgPT4ge1xyXG4gICAgICAgICAgICBpZiAocmVzdWx0LnN1Y2Nlc3MpIHtcclxuICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKGBbUmtMb2FkU3R5bGVzXSDmoLflvI/liqDovb3miJDlip86ICR7cmVzdWx0LmhyZWZ9YCk7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKGBbUmtMb2FkU3R5bGVzXSDmoLflvI/liqDovb3lpLHotKU6ICR7cmVzdWx0LmhyZWZ9YCwgcmVzdWx0LmVycm9yKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgIH0pO1xyXG4gICAgfVxyXG59XHJcbiJdfQ==
@@ -1,91 +1,149 @@
1
1
  import { Directive, Input, TemplateRef } from '@angular/core';
2
2
  import * as i0 from "@angular/core";
3
+ /**
4
+ * 字符串模板输出上下文
5
+ */
6
+ export class RkStringTemplateOutletContext {
7
+ constructor() {
8
+ /** 隐式上下文值 */
9
+ this.$implicit = null;
10
+ }
11
+ }
12
+ /**
13
+ * 字符串模板输出指令
14
+ * 支持动态渲染字符串或 TemplateRef
15
+ *
16
+ * @example
17
+ * ```html
18
+ * <!-- 渲染字符串 -->
19
+ * <ng-container *rkStringTemplateOutlet="'Hello World'">
20
+ * <ng-template>默认模板</ng-template>
21
+ * </ng-container>
22
+ *
23
+ * <!-- 渲染 TemplateRef -->
24
+ * <ng-container *rkStringTemplateOutlet="customTemplate; context: { $implicit: data }">
25
+ * </ng-container>
26
+ * <ng-template #customTemplate let-item>
27
+ * <div>{{ item }}</div>
28
+ * </ng-template>
29
+ *
30
+ * <!-- 动态切换 -->
31
+ * <ng-container *rkStringTemplateOutlet="isTemplate ? templateRef : 'Static Text'">
32
+ * </ng-container>
33
+ * ```
34
+ */
3
35
  export class RkStringTemplateOutletDirective {
4
- constructor(viewContainer, templateRef) {
5
- this.viewContainer = viewContainer;
36
+ constructor(viewContainerRef, templateRef) {
37
+ this.viewContainerRef = viewContainerRef;
6
38
  this.templateRef = templateRef;
7
39
  this.embeddedViewRef = null;
8
40
  this.context = new RkStringTemplateOutletContext();
9
- this.rkStringTemplateOutletContext = null;
41
+ /**
42
+ * 输出内容
43
+ * 可以是字符串或 TemplateRef
44
+ */
10
45
  this.rkStringTemplateOutlet = null;
46
+ /**
47
+ * 模板上下文
48
+ * 当 rkStringTemplateOutlet 是 TemplateRef 时使用
49
+ */
50
+ this.rkStringTemplateOutletContext = null;
11
51
  }
52
+ /**
53
+ * 类型守卫
54
+ * 用于 Angular 模板类型检查
55
+ */
12
56
  static rkTemplateContextGuard(_dir, _ctx) {
13
57
  return true;
14
58
  }
15
- recreateView() {
16
- this.viewContainer.clear();
17
- const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;
18
- const templateRef = (isTemplateRef ? this.rkStringTemplateOutlet : this.templateRef);
19
- this.embeddedViewRef = this.viewContainer.createEmbeddedView(templateRef, isTemplateRef ? this.rkStringTemplateOutletContext : this.context);
20
- }
21
- updateContext() {
22
- const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;
23
- const newCtx = isTemplateRef ? this.rkStringTemplateOutletContext : this.context;
24
- const oldCtx = this.embeddedViewRef.context;
25
- if (newCtx) {
26
- for (const propName of Object.keys(newCtx)) {
27
- oldCtx[propName] = newCtx[propName];
28
- }
29
- }
30
- }
31
59
  ngOnChanges(changes) {
32
- const { rkStringTemplateOutletContext, rkStringTemplateOutlet } = changes;
33
- const shouldRecreateView = () => {
34
- let shouldOutletRecreate = false;
35
- if (rkStringTemplateOutlet) {
36
- if (rkStringTemplateOutlet.firstChange) {
37
- shouldOutletRecreate = true;
38
- }
39
- else {
40
- const isPreviousOutletTemplate = rkStringTemplateOutlet.previousValue instanceof TemplateRef;
41
- const isCurrentOutletTemplate = rkStringTemplateOutlet.currentValue instanceof TemplateRef;
42
- shouldOutletRecreate = isPreviousOutletTemplate || isCurrentOutletTemplate;
43
- }
44
- }
45
- const hasContextShapeChanged = (ctxChange) => {
46
- const prevCtxKeys = Object.keys(ctxChange.previousValue || {});
47
- const currCtxKeys = Object.keys(ctxChange.currentValue || {});
48
- if (prevCtxKeys.length === currCtxKeys.length) {
49
- for (const propName of currCtxKeys) {
50
- if (prevCtxKeys.indexOf(propName) === -1) {
51
- return true;
52
- }
53
- }
54
- return false;
55
- }
56
- else {
57
- return true;
58
- }
59
- };
60
- const shouldContextRecreate = rkStringTemplateOutletContext && hasContextShapeChanged(rkStringTemplateOutletContext);
61
- return shouldContextRecreate || shouldOutletRecreate;
62
- };
60
+ const { rkStringTemplateOutlet, rkStringTemplateOutletContext } = changes;
61
+ // 更新隐式上下文
63
62
  if (rkStringTemplateOutlet) {
64
63
  this.context.$implicit = rkStringTemplateOutlet.currentValue;
65
64
  }
66
- const recreateView = shouldRecreateView();
67
- if (recreateView) {
68
- /** recreate view when context shape or outlet change **/
65
+ // 判断是否需要重建视图
66
+ if (this.shouldRecreateView(rkStringTemplateOutlet, rkStringTemplateOutletContext)) {
69
67
  this.recreateView();
70
68
  }
71
69
  else {
72
- /** update context **/
73
70
  this.updateContext();
74
71
  }
75
72
  }
73
+ /**
74
+ * 判断是否需要重建视图
75
+ */
76
+ shouldRecreateView(outletChange, contextChange) {
77
+ // 首次变更需要创建视图
78
+ if (outletChange === null || outletChange === void 0 ? void 0 : outletChange.firstChange) {
79
+ return true;
80
+ }
81
+ // outlet 在 TemplateRef 和非 TemplateRef 之间切换时需要重建
82
+ if (outletChange && this.isOutletTypeChanged(outletChange)) {
83
+ return true;
84
+ }
85
+ // 上下文结构变化时需要重建
86
+ if (contextChange && this.isContextShapeChanged(contextChange)) {
87
+ return true;
88
+ }
89
+ return false;
90
+ }
91
+ /**
92
+ * 检查 outlet 类型是否变化
93
+ */
94
+ isOutletTypeChanged(change) {
95
+ const isPreviousTemplate = change.previousValue instanceof TemplateRef;
96
+ const isCurrentTemplate = change.currentValue instanceof TemplateRef;
97
+ return isPreviousTemplate !== isCurrentTemplate;
98
+ }
99
+ /**
100
+ * 检查上下文结构是否变化
101
+ */
102
+ isContextShapeChanged(change) {
103
+ const prevKeys = Object.keys(change.previousValue || {});
104
+ const currKeys = Object.keys(change.currentValue || {});
105
+ if (prevKeys.length !== currKeys.length) {
106
+ return true;
107
+ }
108
+ return currKeys.some((key) => !prevKeys.includes(key));
109
+ }
110
+ /**
111
+ * 重建视图
112
+ */
113
+ recreateView() {
114
+ this.viewContainerRef.clear();
115
+ const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;
116
+ const templateRef = isTemplateRef ? this.rkStringTemplateOutlet : this.templateRef;
117
+ const context = isTemplateRef ? this.rkStringTemplateOutletContext : this.context;
118
+ this.embeddedViewRef = this.viewContainerRef.createEmbeddedView(templateRef, context || this.context);
119
+ }
120
+ /**
121
+ * 更新视图上下文
122
+ */
123
+ updateContext() {
124
+ if (!this.embeddedViewRef) {
125
+ return;
126
+ }
127
+ const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;
128
+ const newContext = isTemplateRef ? this.rkStringTemplateOutletContext : this.context;
129
+ const viewContext = this.embeddedViewRef.context;
130
+ if (newContext) {
131
+ Object.keys(newContext).forEach((key) => {
132
+ viewContext[key] = newContext[key];
133
+ });
134
+ }
135
+ }
76
136
  }
77
137
  RkStringTemplateOutletDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: RkStringTemplateOutletDirective, deps: [{ token: i0.ViewContainerRef }, { token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });
78
- RkStringTemplateOutletDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "12.2.17", type: RkStringTemplateOutletDirective, selector: "[rkStringTemplateOutlet]", inputs: { rkStringTemplateOutletContext: "rkStringTemplateOutletContext", rkStringTemplateOutlet: "rkStringTemplateOutlet" }, usesOnChanges: true, ngImport: i0 });
138
+ RkStringTemplateOutletDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "12.2.17", type: RkStringTemplateOutletDirective, selector: "[rkStringTemplateOutlet]", inputs: { rkStringTemplateOutlet: "rkStringTemplateOutlet", rkStringTemplateOutletContext: "rkStringTemplateOutletContext" }, usesOnChanges: true, ngImport: i0 });
79
139
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: RkStringTemplateOutletDirective, decorators: [{
80
140
  type: Directive,
81
141
  args: [{
82
142
  selector: '[rkStringTemplateOutlet]',
83
143
  }]
84
- }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.TemplateRef }]; }, propDecorators: { rkStringTemplateOutletContext: [{
144
+ }], ctorParameters: function () { return [{ type: i0.ViewContainerRef }, { type: i0.TemplateRef }]; }, propDecorators: { rkStringTemplateOutlet: [{
85
145
  type: Input
86
- }], rkStringTemplateOutlet: [{
146
+ }], rkStringTemplateOutletContext: [{
87
147
  type: Input
88
148
  }] } });
89
- export class RkStringTemplateOutletContext {
90
- }
91
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"string.template.outlet.directive.js","sourceRoot":"","sources":["../../../../library/core/directives/string.template.outlet.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, EmbeddedViewRef, Input, OnChanges, SimpleChange, SimpleChanges, TemplateRef, ViewContainerRef } from '@angular/core';\r\n\r\n@Directive({\r\n    selector: '[rkStringTemplateOutlet]',\r\n})\r\nexport class RkStringTemplateOutletDirective<_T = unknown> implements OnChanges {\r\n    private embeddedViewRef: EmbeddedViewRef<any> | null = null;\r\n    private context = new RkStringTemplateOutletContext();\r\n    @Input() rkStringTemplateOutletContext: any | null = null;\r\n    @Input() rkStringTemplateOutlet: any | TemplateRef<any> = null;\r\n\r\n    static rkTemplateContextGuard<T>(_dir: RkStringTemplateOutletDirective<T>, _ctx: any): _ctx is RkStringTemplateOutletContext {\r\n        return true;\r\n    }\r\n\r\n    private recreateView(): void {\r\n        this.viewContainer.clear();\r\n        const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;\r\n        const templateRef = (isTemplateRef ? this.rkStringTemplateOutlet : this.templateRef) as any;\r\n        this.embeddedViewRef = this.viewContainer.createEmbeddedView(\r\n            templateRef,\r\n            isTemplateRef ? this.rkStringTemplateOutletContext : this.context,\r\n        );\r\n    }\r\n\r\n    private updateContext(): void {\r\n        const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;\r\n        const newCtx = isTemplateRef ? this.rkStringTemplateOutletContext : this.context;\r\n        const oldCtx = this.embeddedViewRef!.context as any;\r\n        if (newCtx) {\r\n            for (const propName of Object.keys(newCtx)) {\r\n                oldCtx[propName] = newCtx[propName];\r\n            }\r\n        }\r\n    }\r\n\r\n    constructor(\r\n        private viewContainer: ViewContainerRef,\r\n        private templateRef: TemplateRef<any>,\r\n    ) {}\r\n\r\n    ngOnChanges(changes: SimpleChanges): void {\r\n        const { rkStringTemplateOutletContext, rkStringTemplateOutlet } = changes;\r\n        const shouldRecreateView = (): boolean => {\r\n            let shouldOutletRecreate = false;\r\n            if (rkStringTemplateOutlet) {\r\n                if (rkStringTemplateOutlet.firstChange) {\r\n                    shouldOutletRecreate = true;\r\n                } else {\r\n                    const isPreviousOutletTemplate = rkStringTemplateOutlet.previousValue instanceof TemplateRef;\r\n                    const isCurrentOutletTemplate = rkStringTemplateOutlet.currentValue instanceof TemplateRef;\r\n                    shouldOutletRecreate = isPreviousOutletTemplate || isCurrentOutletTemplate;\r\n                }\r\n            }\r\n            const hasContextShapeChanged = (ctxChange: SimpleChange): boolean => {\r\n                const prevCtxKeys = Object.keys(ctxChange.previousValue || {});\r\n                const currCtxKeys = Object.keys(ctxChange.currentValue || {});\r\n                if (prevCtxKeys.length === currCtxKeys.length) {\r\n                    for (const propName of currCtxKeys) {\r\n                        if (prevCtxKeys.indexOf(propName) === -1) {\r\n                            return true;\r\n                        }\r\n                    }\r\n                    return false;\r\n                } else {\r\n                    return true;\r\n                }\r\n            };\r\n            const shouldContextRecreate = rkStringTemplateOutletContext && hasContextShapeChanged(rkStringTemplateOutletContext);\r\n            return shouldContextRecreate || shouldOutletRecreate;\r\n        };\r\n\r\n        if (rkStringTemplateOutlet) {\r\n            this.context.$implicit = rkStringTemplateOutlet.currentValue;\r\n        }\r\n\r\n        const recreateView = shouldRecreateView();\r\n        if (recreateView) {\r\n            /** recreate view when context shape or outlet change **/\r\n            this.recreateView();\r\n        } else {\r\n            /** update context **/\r\n            this.updateContext();\r\n        }\r\n    }\r\n}\r\n\r\nexport class RkStringTemplateOutletContext {\r\n    public $implicit: any;\r\n}\r\n"]}
149
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"string.template.outlet.directive.js","sourceRoot":"","sources":["../../../../library/core/directives/string.template.outlet.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, EmbeddedViewRef, Input, OnChanges, SimpleChange, SimpleChanges, TemplateRef, ViewContainerRef } from '@angular/core';\r\n\r\n/**\r\n * 字符串模板输出上下文\r\n */\r\nexport class RkStringTemplateOutletContext<T = any> {\r\n    /** 隐式上下文值 */\r\n    public $implicit: T | null = null;\r\n}\r\n\r\n/**\r\n * 字符串模板输出指令\r\n * 支持动态渲染字符串或 TemplateRef\r\n *\r\n * @example\r\n * ```html\r\n * <!-- 渲染字符串 -->\r\n * <ng-container *rkStringTemplateOutlet=\"'Hello World'\">\r\n *   <ng-template>默认模板</ng-template>\r\n * </ng-container>\r\n *\r\n * <!-- 渲染 TemplateRef -->\r\n * <ng-container *rkStringTemplateOutlet=\"customTemplate; context: { $implicit: data }\">\r\n * </ng-container>\r\n * <ng-template #customTemplate let-item>\r\n *   <div>{{ item }}</div>\r\n * </ng-template>\r\n *\r\n * <!-- 动态切换 -->\r\n * <ng-container *rkStringTemplateOutlet=\"isTemplate ? templateRef : 'Static Text'\">\r\n * </ng-container>\r\n * ```\r\n */\r\n@Directive({\r\n    selector: '[rkStringTemplateOutlet]',\r\n})\r\nexport class RkStringTemplateOutletDirective<T = unknown> implements OnChanges {\r\n    private embeddedViewRef: EmbeddedViewRef<RkStringTemplateOutletContext<T>> | null = null;\r\n    private context = new RkStringTemplateOutletContext<T>();\r\n\r\n    /**\r\n     * 输出内容\r\n     * 可以是字符串或 TemplateRef\r\n     */\r\n    @Input() rkStringTemplateOutlet: T | TemplateRef<RkStringTemplateOutletContext<T>> | null = null;\r\n\r\n    /**\r\n     * 模板上下文\r\n     * 当 rkStringTemplateOutlet 是 TemplateRef 时使用\r\n     */\r\n    @Input() rkStringTemplateOutletContext: RkStringTemplateOutletContext<T> | null = null;\r\n\r\n    /**\r\n     * 类型守卫\r\n     * 用于 Angular 模板类型检查\r\n     */\r\n    static rkTemplateContextGuard<T>(\r\n        _dir: RkStringTemplateOutletDirective<T>,\r\n        _ctx: any,\r\n    ): _ctx is RkStringTemplateOutletContext<T> {\r\n        return true;\r\n    }\r\n\r\n    constructor(\r\n        private viewContainerRef: ViewContainerRef,\r\n        private templateRef: TemplateRef<RkStringTemplateOutletContext<T>>,\r\n    ) {}\r\n\r\n    ngOnChanges(changes: SimpleChanges): void {\r\n        const { rkStringTemplateOutlet, rkStringTemplateOutletContext } = changes;\r\n\r\n        // 更新隐式上下文\r\n        if (rkStringTemplateOutlet) {\r\n            this.context.$implicit = rkStringTemplateOutlet.currentValue;\r\n        }\r\n\r\n        // 判断是否需要重建视图\r\n        if (this.shouldRecreateView(rkStringTemplateOutlet, rkStringTemplateOutletContext)) {\r\n            this.recreateView();\r\n        } else {\r\n            this.updateContext();\r\n        }\r\n    }\r\n\r\n    /**\r\n     * 判断是否需要重建视图\r\n     */\r\n    private shouldRecreateView(outletChange?: SimpleChange, contextChange?: SimpleChange): boolean {\r\n        // 首次变更需要创建视图\r\n        if (outletChange?.firstChange) {\r\n            return true;\r\n        }\r\n\r\n        // outlet 在 TemplateRef 和非 TemplateRef 之间切换时需要重建\r\n        if (outletChange && this.isOutletTypeChanged(outletChange)) {\r\n            return true;\r\n        }\r\n\r\n        // 上下文结构变化时需要重建\r\n        if (contextChange && this.isContextShapeChanged(contextChange)) {\r\n            return true;\r\n        }\r\n\r\n        return false;\r\n    }\r\n\r\n    /**\r\n     * 检查 outlet 类型是否变化\r\n     */\r\n    private isOutletTypeChanged(change: SimpleChange): boolean {\r\n        const isPreviousTemplate = change.previousValue instanceof TemplateRef;\r\n        const isCurrentTemplate = change.currentValue instanceof TemplateRef;\r\n        return isPreviousTemplate !== isCurrentTemplate;\r\n    }\r\n\r\n    /**\r\n     * 检查上下文结构是否变化\r\n     */\r\n    private isContextShapeChanged(change: SimpleChange): boolean {\r\n        const prevKeys = Object.keys(change.previousValue || {});\r\n        const currKeys = Object.keys(change.currentValue || {});\r\n\r\n        if (prevKeys.length !== currKeys.length) {\r\n            return true;\r\n        }\r\n\r\n        return currKeys.some((key) => !prevKeys.includes(key));\r\n    }\r\n\r\n    /**\r\n     * 重建视图\r\n     */\r\n    private recreateView(): void {\r\n        this.viewContainerRef.clear();\r\n\r\n        const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;\r\n        const templateRef = isTemplateRef ? this.rkStringTemplateOutlet : this.templateRef;\r\n        const context = isTemplateRef ? this.rkStringTemplateOutletContext : this.context;\r\n\r\n        this.embeddedViewRef = this.viewContainerRef.createEmbeddedView(\r\n            templateRef as TemplateRef<RkStringTemplateOutletContext<T>>,\r\n            context || this.context,\r\n        );\r\n    }\r\n\r\n    /**\r\n     * 更新视图上下文\r\n     */\r\n    private updateContext(): void {\r\n        if (!this.embeddedViewRef) {\r\n            return;\r\n        }\r\n\r\n        const isTemplateRef = this.rkStringTemplateOutlet instanceof TemplateRef;\r\n        const newContext = isTemplateRef ? this.rkStringTemplateOutletContext : this.context;\r\n        const viewContext = this.embeddedViewRef.context;\r\n\r\n        if (newContext) {\r\n            Object.keys(newContext).forEach((key) => {\r\n                (viewContext as any)[key] = (newContext as any)[key];\r\n            });\r\n        }\r\n    }\r\n}\r\n"]}