@yibozhang/pro-table 0.0.1

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 (63) hide show
  1. package/README.md +24 -0
  2. package/bundles/yibozhang-pro-table.umd.js +3110 -0
  3. package/bundles/yibozhang-pro-table.umd.js.map +1 -0
  4. package/bundles/yibozhang-pro-table.umd.min.js +2 -0
  5. package/bundles/yibozhang-pro-table.umd.min.js.map +1 -0
  6. package/esm2015/lib/components/colmuns-setting/colmuns-setting.component.js +158 -0
  7. package/esm2015/lib/components/dynamic-search-field/dynamic-search-field.component.js +101 -0
  8. package/esm2015/lib/constants.js +89 -0
  9. package/esm2015/lib/page-container/page-container.component.js +21 -0
  10. package/esm2015/lib/page-container/page-container.module.js +25 -0
  11. package/esm2015/lib/page-public/antd-form.js +734 -0
  12. package/esm2015/lib/plate-input/plate-input.component.js +155 -0
  13. package/esm2015/lib/plate-input/plate-input.module.js +26 -0
  14. package/esm2015/lib/plate-input/plate-prefix-load.service.js +41 -0
  15. package/esm2015/lib/pro-table.component.js +983 -0
  16. package/esm2015/lib/pro-table.module.js +73 -0
  17. package/esm2015/lib/table-search-bar/table-search-bar-module.js +27 -0
  18. package/esm2015/lib/table-search-bar/table-search-bar.component.js +51 -0
  19. package/esm2015/lib/tokens.js +4 -0
  20. package/esm2015/lib/type.js +2 -0
  21. package/esm2015/public-api.js +12 -0
  22. package/esm2015/yibozhang-pro-table.js +13 -0
  23. package/fesm2015/yibozhang-pro-table.js +2448 -0
  24. package/fesm2015/yibozhang-pro-table.js.map +1 -0
  25. package/lib/components/colmuns-setting/colmuns-setting.component.d.ts +35 -0
  26. package/lib/components/colmuns-setting/colmuns-setting.component.d.ts.map +1 -0
  27. package/lib/components/dynamic-search-field/dynamic-search-field.component.d.ts +22 -0
  28. package/lib/components/dynamic-search-field/dynamic-search-field.component.d.ts.map +1 -0
  29. package/lib/constants.d.ts +20 -0
  30. package/lib/constants.d.ts.map +1 -0
  31. package/lib/page-container/page-container.component.d.ts +9 -0
  32. package/lib/page-container/page-container.component.d.ts.map +1 -0
  33. package/lib/page-container/page-container.module.d.ts +3 -0
  34. package/lib/page-container/page-container.module.d.ts.map +1 -0
  35. package/lib/page-public/antd-form.d.ts +99 -0
  36. package/lib/page-public/antd-form.d.ts.map +1 -0
  37. package/lib/plate-input/plate-input.component.d.ts +31 -0
  38. package/lib/plate-input/plate-input.component.d.ts.map +1 -0
  39. package/lib/plate-input/plate-input.module.d.ts +3 -0
  40. package/lib/plate-input/plate-input.module.d.ts.map +1 -0
  41. package/lib/plate-input/plate-prefix-load.service.d.ts +13 -0
  42. package/lib/plate-input/plate-prefix-load.service.d.ts.map +1 -0
  43. package/lib/pro-table.component.d.ts +122 -0
  44. package/lib/pro-table.component.d.ts.map +1 -0
  45. package/lib/pro-table.module.d.ts +3 -0
  46. package/lib/pro-table.module.d.ts.map +1 -0
  47. package/lib/table-search-bar/table-search-bar-module.d.ts +3 -0
  48. package/lib/table-search-bar/table-search-bar-module.d.ts.map +1 -0
  49. package/lib/table-search-bar/table-search-bar.component.d.ts +15 -0
  50. package/lib/table-search-bar/table-search-bar.component.d.ts.map +1 -0
  51. package/lib/tokens.d.ts +36 -0
  52. package/lib/tokens.d.ts.map +1 -0
  53. package/lib/type.d.ts +226 -0
  54. package/lib/type.d.ts.map +1 -0
  55. package/package.json +40 -0
  56. package/public-api.d.ts +9 -0
  57. package/public-api.d.ts.map +1 -0
  58. package/src/lib/styles/custom-antd.less +114 -0
  59. package/src/lib/styles/margin.css +90 -0
  60. package/src/lib/styles/theme.less +42 -0
  61. package/yibozhang-pro-table.d.ts +13 -0
  62. package/yibozhang-pro-table.d.ts.map +1 -0
  63. package/yibozhang-pro-table.metadata.json +1 -0
@@ -0,0 +1,155 @@
1
+ import { Component, forwardRef, ChangeDetectorRef, ElementRef, Inject, Optional } from "@angular/core";
2
+ import { NG_VALUE_ACCESSOR } from "@angular/forms";
3
+ import { PLATE_PREFIX_LOAD_SERVICE } from "../tokens";
4
+ export class PlateInputComponent {
5
+ constructor(cdr, elementRef, platePrefixLoadService) {
6
+ this.cdr = cdr;
7
+ this.elementRef = elementRef;
8
+ this.platePrefixLoadService = platePrefixLoadService;
9
+ this.nzPopoverVisible = false;
10
+ this.platePrefix = "";
11
+ this.plateSuffix = "";
12
+ this.platePrefixOptions = [];
13
+ this.disabled = false;
14
+ this.onChange = (value) => { };
15
+ this.onTouched = () => { };
16
+ }
17
+ ngOnInit() {
18
+ if (this.platePrefixLoadService) {
19
+ // 使用注入的全局服务加载车牌前缀选项
20
+ this.platePrefixLoadService
21
+ .loadPrefixOptions()
22
+ .then((data) => {
23
+ this.platePrefixOptions = data;
24
+ })
25
+ .catch(() => {
26
+ this.platePrefixOptions = [];
27
+ });
28
+ }
29
+ else {
30
+ // 如果没有注入服务,则保持空数组
31
+ this.platePrefixOptions = [];
32
+ }
33
+ }
34
+ handlePlatePrefix(value) {
35
+ this.platePrefix = value;
36
+ this.emitValue();
37
+ this.nzPopoverVisible = false;
38
+ this.removeClickOutsideListener();
39
+ }
40
+ writeValue(value) {
41
+ if (value) {
42
+ this.platePrefix = value.split("")[0];
43
+ this.plateSuffix = value
44
+ .split("")
45
+ .filter((item, index) => index > 0)
46
+ .join("");
47
+ }
48
+ else {
49
+ this.platePrefix = "";
50
+ this.plateSuffix = "";
51
+ }
52
+ }
53
+ registerOnChange(fn) {
54
+ this.onChange = fn;
55
+ }
56
+ registerOnTouched(fn) {
57
+ this.onTouched = fn;
58
+ }
59
+ setDisabledState(isDisabled) {
60
+ const previousDisabled = this.disabled;
61
+ this.disabled = isDisabled;
62
+ // 如果禁用状态改变,关闭 popover 并移除监听器
63
+ if (previousDisabled !== isDisabled) {
64
+ this.nzPopoverVisible = false;
65
+ this.removeClickOutsideListener();
66
+ // 强制变更检测
67
+ this.cdr.detectChanges();
68
+ }
69
+ }
70
+ handlePrefixClick(event) {
71
+ if (this.disabled) {
72
+ event.preventDefault();
73
+ event.stopPropagation();
74
+ this.nzPopoverVisible = false;
75
+ return;
76
+ }
77
+ // 手动切换 popover 的显示/隐藏状态
78
+ this.nzPopoverVisible = !this.nzPopoverVisible;
79
+ // 如果打开了 popover,添加文档点击监听器
80
+ if (this.nzPopoverVisible) {
81
+ setTimeout(() => {
82
+ this.attachClickOutsideListener();
83
+ }, 0);
84
+ }
85
+ else {
86
+ this.removeClickOutsideListener();
87
+ }
88
+ event.stopPropagation(); // 阻止事件冒泡
89
+ }
90
+ attachClickOutsideListener() {
91
+ if (this.clickOutsideHandler) {
92
+ return;
93
+ }
94
+ this.clickOutsideHandler = (event) => {
95
+ const nativeElement = this.elementRef.nativeElement;
96
+ const target = event.target;
97
+ // 检查是否点击在 popover 内容区域内
98
+ const popoverContent = document.querySelector('.plate-input-popover');
99
+ const clickedInPopover = popoverContent && popoverContent.contains(target);
100
+ // 检查是否点击在组件内部(前缀按钮)
101
+ const clickedInComponent = nativeElement.contains(target);
102
+ // 如果点击既不在组件内部,也不在 popover 内容区域,则关闭 popover
103
+ if (!clickedInComponent && !clickedInPopover) {
104
+ this.nzPopoverVisible = false;
105
+ this.removeClickOutsideListener();
106
+ this.cdr.detectChanges();
107
+ }
108
+ };
109
+ // 使用捕获阶段,确保在其他事件处理之前执行
110
+ document.addEventListener('click', this.clickOutsideHandler, true);
111
+ }
112
+ removeClickOutsideListener() {
113
+ if (this.clickOutsideHandler) {
114
+ document.removeEventListener('click', this.clickOutsideHandler, true);
115
+ this.clickOutsideHandler = undefined;
116
+ }
117
+ }
118
+ ngOnDestroy() {
119
+ this.removeClickOutsideListener();
120
+ }
121
+ emitValue() {
122
+ const fullPlate = this.platePrefix + this.plateSuffix;
123
+ this.onChange(fullPlate);
124
+ }
125
+ onBlur() {
126
+ this.onTouched();
127
+ }
128
+ handleAntdPlateSuffixBlur(event) {
129
+ if (this.disabled)
130
+ return;
131
+ const value = event.target.value.toUpperCase();
132
+ this.plateSuffix = value;
133
+ this.emitValue();
134
+ }
135
+ }
136
+ PlateInputComponent.decorators = [
137
+ { type: Component, args: [{
138
+ selector: "app-plate-input",
139
+ template: "<div\r\n class=\"plate-antd-input-container\"\r\n [class.plate-antd-input-disabled]=\"disabled\"\r\n>\r\n <div\r\n class=\"plate-input-prefix\"\r\n [class.plate-input-prefix-disabled]=\"disabled\"\r\n nz-popover\r\n nzPopoverPlacement=\"bottom\"\r\n nzPopoverOverlayClassName=\"plate-input-popover\"\r\n [(nzPopoverVisible)]=\"nzPopoverVisible\"\r\n [nzPopoverTrigger]=\"'manual'\"\r\n [nzPopoverContent]=\"contentTemplate\"\r\n (click)=\"handlePrefixClick($event)\"\r\n >\r\n {{ platePrefix | uppercase }}\r\n <i\r\n class=\"plate-input-icon-down\"\r\n [nzRotate]=\"nzPopoverVisible ? 180 : 0\"\r\n nz-icon\r\n nzType=\"down\"\r\n nzTheme=\"outline\"\r\n ></i>\r\n </div>\r\n <ng-template #contentTemplate>\r\n <div nz-row style=\"height: 100%\">\r\n <div\r\n nz-col\r\n nzSpan=\"4\"\r\n *ngFor=\"let item of platePrefixOptions\"\r\n class=\"plate-input-popover-item\"\r\n (click)=\"!disabled && handlePlatePrefix(item.value)\"\r\n [ngClass]=\"{\r\n 'plate-input-popover-item-active': item.value === platePrefix,\r\n 'plate-input-popover-item-default': item.value !== platePrefix\r\n }\"\r\n >\r\n <div class=\"plate-input-popover-item-label\">\r\n {{ item.label }}\r\n </div>\r\n </div>\r\n </div>\r\n </ng-template>\r\n <input\r\n class=\"plate-input-suffix\"\r\n type=\"text\"\r\n nz-input\r\n style=\"width: 75%\"\r\n [value]=\"plateSuffix | uppercase\"\r\n [disabled]=\"disabled\"\r\n (blur)=\"!disabled && handleAntdPlateSuffixBlur($event)\"\r\n />\r\n</div>\r\n",
140
+ providers: [
141
+ {
142
+ provide: NG_VALUE_ACCESSOR,
143
+ useExisting: forwardRef(() => PlateInputComponent),
144
+ multi: true,
145
+ },
146
+ ],
147
+ styles: ["::ng-deep .plate-input-popover{box-sizing:border-box;padding:5px 0}::ng-deep .plate-input-popover .ant-popover-inner-content{height:264px;padding:0;width:387px}::ng-deep .plate-input-popover .ant-popover-inner-content .plate-input-popover-item{box-sizing:border-box;cursor:pointer;height:42px;padding:6px 12px}::ng-deep .plate-input-popover .ant-popover-inner-content .plate-input-popover-item .plate-input-popover-item-label{align-items:center;border:1px solid #ccc;display:flex;height:100%;justify-content:center;width:100%}::ng-deep .plate-input-popover .ant-popover-inner-content .plate-input-popover-item-default:hover{background-color:#eaeaea}::ng-deep .plate-input-popover .ant-popover-inner-content .plate-input-popover-item-active{background-color:#096dd9;color:#fff}::ng-deep .plate-input-prefix:hover:after{background-color:transparent;content:\"\";height:100%;position:absolute;right:0;top:0;width:1px;z-index:2}::ng-deep .plate-antd-input-container{display:flex;flex-wrap:nowrap;width:100%}::ng-deep .plate-antd-input-container .plate-input-prefix{align-items:center;border:1px solid #d9d9d9;border-radius:2px;cursor:pointer;display:flex;justify-content:center;min-height:32px;position:relative;text-align:center;width:60px}::ng-deep .plate-antd-input-container .plate-input-prefix .plate-input-icon-down{font-size:10px;position:absolute;right:5px;top:50%;transform:translateY(-50%)}::ng-deep .plate-antd-input-container .plate-input-prefix:hover:not(.plate-input-prefix-disabled){border-color:#40a9ff}::ng-deep .plate-antd-input-container .plate-input-prefix-disabled{background-color:#f5f5f5;border-color:#d9d9d9;color:rgba(0,0,0,.25);cursor:not-allowed}::ng-deep .plate-antd-input-container .plate-input-suffix{width:calc(100% - 60px)}"]
148
+ },] }
149
+ ];
150
+ PlateInputComponent.ctorParameters = () => [
151
+ { type: ChangeDetectorRef },
152
+ { type: ElementRef },
153
+ { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [PLATE_PREFIX_LOAD_SERVICE,] }] }
154
+ ];
155
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plate-input.component.js","sourceRoot":"D:/projects/vps-front/Front/DasPMSWeb/projects/pro-table/src/","sources":["lib/plate-input/plate-input.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAqB,UAAU,EAAE,iBAAiB,EAAE,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAC1H,OAAO,EAAwB,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACzE,OAAO,EAAE,yBAAyB,EAA6C,MAAM,WAAW,CAAC;AAcjG,MAAM,OAAO,mBAAmB;IAW9B,YACU,GAAsB,EACtB,UAAsB,EACyB,sBAA+C;QAF9F,QAAG,GAAH,GAAG,CAAmB;QACtB,eAAU,GAAV,UAAU,CAAY;QACyB,2BAAsB,GAAtB,sBAAsB,CAAyB;QAbxG,qBAAgB,GAAY,KAAK,CAAC;QAClC,gBAAW,GAAW,EAAE,CAAC;QACzB,gBAAW,GAAW,EAAE,CAAC;QACzB,uBAAkB,GAA6B,EAAE,CAAC;QAClD,aAAQ,GAAY,KAAK,CAAC;QAClB,aAAQ,GAAG,CAAC,KAAa,EAAE,EAAE,GAAE,CAAC,CAAC;QACjC,cAAS,GAAG,GAAG,EAAE,GAAE,CAAC,CAAC;IAQ1B,CAAC;IAEJ,QAAQ;QACN,IAAI,IAAI,CAAC,sBAAsB,EAAE;YAC/B,oBAAoB;YACpB,IAAI,CAAC,sBAAsB;iBACxB,iBAAiB,EAAE;iBACnB,IAAI,CAAC,CAAC,IAA8B,EAAE,EAAE;gBACvC,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;YACjC,CAAC,CAAC;iBACD,KAAK,CAAC,GAAG,EAAE;gBACV,IAAI,CAAC,kBAAkB,GAAG,EAAE,CAAC;YAC/B,CAAC,CAAC,CAAC;SACN;aAAM;YACL,kBAAkB;YAClB,IAAI,CAAC,kBAAkB,GAAG,EAAE,CAAC;SAC9B;IACH,CAAC;IAED,iBAAiB,CAAC,KAAa;QAC7B,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;QAC9B,IAAI,CAAC,0BAA0B,EAAE,CAAC;IACpC,CAAC;IAED,UAAU,CAAC,KAAa;QACtB,IAAI,KAAK,EAAE;YACT,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YACtC,IAAI,CAAC,WAAW,GAAG,KAAK;iBACrB,KAAK,CAAC,EAAE,CAAC;iBACT,MAAM,CAAC,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,GAAG,CAAC,CAAC;iBAClC,IAAI,CAAC,EAAE,CAAC,CAAC;SACb;aAAM;YACL,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;YACtB,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;SACvB;IACH,CAAC;IAED,gBAAgB,CAAC,EAA2B;QAC1C,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,EAAc;QAC9B,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAED,gBAAgB,CAAC,UAAmB;QAClC,MAAM,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC;QACvC,IAAI,CAAC,QAAQ,GAAG,UAAU,CAAC;QAE3B,6BAA6B;QAC7B,IAAI,gBAAgB,KAAK,UAAU,EAAE;YACnC,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;YAC9B,IAAI,CAAC,0BAA0B,EAAE,CAAC;YAClC,SAAS;YACT,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;SAC1B;IACH,CAAC;IAED,iBAAiB,CAAC,KAAY;QAC5B,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,KAAK,CAAC,cAAc,EAAE,CAAC;YACvB,KAAK,CAAC,eAAe,EAAE,CAAC;YACxB,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;YAC9B,OAAO;SACR;QACD,wBAAwB;QACxB,IAAI,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC;QAE/C,0BAA0B;QAC1B,IAAI,IAAI,CAAC,gBAAgB,EAAE;YACzB,UAAU,CAAC,GAAG,EAAE;gBACd,IAAI,CAAC,0BAA0B,EAAE,CAAC;YACpC,CAAC,EAAE,CAAC,CAAC,CAAC;SACP;aAAM;YACL,IAAI,CAAC,0BAA0B,EAAE,CAAC;SACnC;QAED,KAAK,CAAC,eAAe,EAAE,CAAC,CAAC,SAAS;IACpC,CAAC;IAEO,0BAA0B;QAChC,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAC5B,OAAO;SACR;QAED,IAAI,CAAC,mBAAmB,GAAG,CAAC,KAAiB,EAAE,EAAE;YAC/C,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC;YACpD,MAAM,MAAM,GAAG,KAAK,CAAC,MAAqB,CAAC;YAE3C,wBAAwB;YACxB,MAAM,cAAc,GAAG,QAAQ,CAAC,aAAa,CAAC,sBAAsB,CAAC,CAAC;YACtE,MAAM,gBAAgB,GAAG,cAAc,IAAI,cAAc,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YAE3E,oBAAoB;YACpB,MAAM,kBAAkB,GAAG,aAAa,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YAE1D,2CAA2C;YAC3C,IAAI,CAAC,kBAAkB,IAAI,CAAC,gBAAgB,EAAE;gBAC5C,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC;gBAC9B,IAAI,CAAC,0BAA0B,EAAE,CAAC;gBAClC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;aAC1B;QACH,CAAC,CAAC;QAEF,uBAAuB;QACvB,QAAQ,CAAC,gBAAgB,CAAC,OAAO,EAAE,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;IACrE,CAAC;IAEO,0BAA0B;QAChC,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAC5B,QAAQ,CAAC,mBAAmB,CAAC,OAAO,EAAE,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,CAAC;YACtE,IAAI,CAAC,mBAAmB,GAAG,SAAS,CAAC;SACtC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,0BAA0B,EAAE,CAAC;IACpC,CAAC;IAEO,SAAS;QACf,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACtD,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;IAC3B,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,yBAAyB,CAAC,KAAU;QAClC,IAAI,IAAI,CAAC,QAAQ;YAAE,OAAO;QAC1B,MAAM,KAAK,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;QAC/C,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;;;YAlKF,SAAS,SAAC;gBACT,QAAQ,EAAE,iBAAiB;gBAC3B,mpDAA2C;gBAE3C,SAAS,EAAE;oBACT;wBACE,OAAO,EAAE,iBAAiB;wBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,mBAAmB,CAAC;wBAClD,KAAK,EAAE,IAAI;qBACZ;iBACF;;aACF;;;YAfkD,iBAAiB;YAAE,UAAU;4CA8B3E,QAAQ,YAAI,MAAM,SAAC,yBAAyB","sourcesContent":["import { Component, OnInit, OnDestroy, forwardRef, ChangeDetectorRef, ElementRef, Inject, Optional } from \"@angular/core\";\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from \"@angular/forms\";\r\nimport { PLATE_PREFIX_LOAD_SERVICE, PlatePrefixLoadService, PlatePrefixOption } from \"../tokens\";\r\n\r\n@Component({\r\n  selector: \"app-plate-input\",\r\n  templateUrl: \"./plate-input.component.html\",\r\n  styleUrls: [\"./plate-input.component.less\"],\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => PlateInputComponent),\r\n      multi: true,\r\n    },\r\n  ],\r\n})\r\nexport class PlateInputComponent implements OnInit, OnDestroy, ControlValueAccessor {\r\n  nzPopoverVisible: boolean = false;\r\n  platePrefix: string = \"\";\r\n  plateSuffix: string = \"\";\r\n  platePrefixOptions: Array<PlatePrefixOption> = [];\r\n  disabled: boolean = false;\r\n  private onChange = (value: string) => {};\r\n  private onTouched = () => {};\r\n\r\n  private clickOutsideHandler?: (event: MouseEvent) => void;\r\n\r\n  constructor(\r\n    private cdr: ChangeDetectorRef,\r\n    private elementRef: ElementRef,\r\n    @Optional() @Inject(PLATE_PREFIX_LOAD_SERVICE) private platePrefixLoadService?: PlatePrefixLoadService\r\n  ) {}\r\n\r\n  ngOnInit(): void {\r\n    if (this.platePrefixLoadService) {\r\n      // 使用注入的全局服务加载车牌前缀选项\r\n      this.platePrefixLoadService\r\n        .loadPrefixOptions()\r\n        .then((data: Array<PlatePrefixOption>) => {\r\n          this.platePrefixOptions = data;\r\n        })\r\n        .catch(() => {\r\n          this.platePrefixOptions = [];\r\n        });\r\n    } else {\r\n      // 如果没有注入服务，则保持空数组\r\n      this.platePrefixOptions = [];\r\n    }\r\n  }\r\n\r\n  handlePlatePrefix(value: string): void {\r\n    this.platePrefix = value;\r\n    this.emitValue();\r\n    this.nzPopoverVisible = false;\r\n    this.removeClickOutsideListener();\r\n  }\r\n\r\n  writeValue(value: string): void {\r\n    if (value) {\r\n      this.platePrefix = value.split(\"\")[0];\r\n      this.plateSuffix = value\r\n        .split(\"\")\r\n        .filter((item, index) => index > 0)\r\n        .join(\"\");\r\n    } else {\r\n      this.platePrefix = \"\";\r\n      this.plateSuffix = \"\";\r\n    }\r\n  }\r\n\r\n  registerOnChange(fn: (value: string) => void): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: () => void): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    const previousDisabled = this.disabled;\r\n    this.disabled = isDisabled;\r\n    \r\n    // 如果禁用状态改变，关闭 popover 并移除监听器\r\n    if (previousDisabled !== isDisabled) {\r\n      this.nzPopoverVisible = false;\r\n      this.removeClickOutsideListener();\r\n      // 强制变更检测\r\n      this.cdr.detectChanges();\r\n    }\r\n  }\r\n\r\n  handlePrefixClick(event: Event): void {\r\n    if (this.disabled) {\r\n      event.preventDefault();\r\n      event.stopPropagation();\r\n      this.nzPopoverVisible = false;\r\n      return;\r\n    }\r\n    // 手动切换 popover 的显示/隐藏状态\r\n    this.nzPopoverVisible = !this.nzPopoverVisible;\r\n    \r\n    // 如果打开了 popover，添加文档点击监听器\r\n    if (this.nzPopoverVisible) {\r\n      setTimeout(() => {\r\n        this.attachClickOutsideListener();\r\n      }, 0);\r\n    } else {\r\n      this.removeClickOutsideListener();\r\n    }\r\n    \r\n    event.stopPropagation(); // 阻止事件冒泡\r\n  }\r\n\r\n  private attachClickOutsideListener(): void {\r\n    if (this.clickOutsideHandler) {\r\n      return;\r\n    }\r\n    \r\n    this.clickOutsideHandler = (event: MouseEvent) => {\r\n      const nativeElement = this.elementRef.nativeElement;\r\n      const target = event.target as HTMLElement;\r\n      \r\n      // 检查是否点击在 popover 内容区域内\r\n      const popoverContent = document.querySelector('.plate-input-popover');\r\n      const clickedInPopover = popoverContent && popoverContent.contains(target);\r\n      \r\n      // 检查是否点击在组件内部（前缀按钮）\r\n      const clickedInComponent = nativeElement.contains(target);\r\n      \r\n      // 如果点击既不在组件内部，也不在 popover 内容区域，则关闭 popover\r\n      if (!clickedInComponent && !clickedInPopover) {\r\n        this.nzPopoverVisible = false;\r\n        this.removeClickOutsideListener();\r\n        this.cdr.detectChanges();\r\n      }\r\n    };\r\n    \r\n    // 使用捕获阶段，确保在其他事件处理之前执行\r\n    document.addEventListener('click', this.clickOutsideHandler, true);\r\n  }\r\n\r\n  private removeClickOutsideListener(): void {\r\n    if (this.clickOutsideHandler) {\r\n      document.removeEventListener('click', this.clickOutsideHandler, true);\r\n      this.clickOutsideHandler = undefined;\r\n    }\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.removeClickOutsideListener();\r\n  }\r\n\r\n  private emitValue(): void {\r\n    const fullPlate = this.platePrefix + this.plateSuffix;\r\n    this.onChange(fullPlate);\r\n  }\r\n\r\n  onBlur(): void {\r\n    this.onTouched();\r\n  }\r\n\r\n  handleAntdPlateSuffixBlur(event: any): void {\r\n    if (this.disabled) return;\r\n    const value = event.target.value.toUpperCase();\r\n    this.plateSuffix = value;\r\n    this.emitValue();\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,26 @@
1
+ import { CommonModule } from "@angular/common";
2
+ import { NgModule } from "@angular/core";
3
+ import { FormsModule } from "@angular/forms";
4
+ import { NzGridModule } from "ng-zorro-antd/grid";
5
+ import { NzIconModule } from "ng-zorro-antd/icon";
6
+ import { NzInputModule } from "ng-zorro-antd/input";
7
+ import { NzPopoverModule } from "ng-zorro-antd/popover";
8
+ import { PlateInputComponent } from "./plate-input.component";
9
+ export class PlateInputModule {
10
+ }
11
+ PlateInputModule.decorators = [
12
+ { type: NgModule, args: [{
13
+ imports: [
14
+ CommonModule,
15
+ FormsModule,
16
+ NzGridModule,
17
+ NzInputModule,
18
+ NzPopoverModule,
19
+ NzIconModule,
20
+ ],
21
+ declarations: [PlateInputComponent],
22
+ providers: [],
23
+ exports: [PlateInputComponent],
24
+ },] }
25
+ ];
26
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGxhdGUtaW5wdXQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IkQ6L3Byb2plY3RzL3Zwcy1mcm9udC9Gcm9udC9EYXNQTVNXZWIvcHJvamVjdHMvcHJvLXRhYmxlL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9wbGF0ZS1pbnB1dC9wbGF0ZS1pbnB1dC5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQzdDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUNsRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFDbEQsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3BELE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUN4RCxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQWU5RCxNQUFNLE9BQU8sZ0JBQWdCOzs7WUFiNUIsUUFBUSxTQUFDO2dCQUNSLE9BQU8sRUFBRTtvQkFDUCxZQUFZO29CQUNaLFdBQVc7b0JBQ1gsWUFBWTtvQkFDWixhQUFhO29CQUNiLGVBQWU7b0JBQ2YsWUFBWTtpQkFDYjtnQkFDRCxZQUFZLEVBQUUsQ0FBQyxtQkFBbUIsQ0FBQztnQkFDbkMsU0FBUyxFQUFFLEVBQUU7Z0JBQ2IsT0FBTyxFQUFFLENBQUMsbUJBQW1CLENBQUM7YUFDL0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tIFwiQGFuZ3VsYXIvY29tbW9uXCI7XHJcbmltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcclxuaW1wb3J0IHsgRm9ybXNNb2R1bGUgfSBmcm9tIFwiQGFuZ3VsYXIvZm9ybXNcIjtcclxuaW1wb3J0IHsgTnpHcmlkTW9kdWxlIH0gZnJvbSBcIm5nLXpvcnJvLWFudGQvZ3JpZFwiO1xyXG5pbXBvcnQgeyBOekljb25Nb2R1bGUgfSBmcm9tIFwibmctem9ycm8tYW50ZC9pY29uXCI7XHJcbmltcG9ydCB7IE56SW5wdXRNb2R1bGUgfSBmcm9tIFwibmctem9ycm8tYW50ZC9pbnB1dFwiO1xyXG5pbXBvcnQgeyBOelBvcG92ZXJNb2R1bGUgfSBmcm9tIFwibmctem9ycm8tYW50ZC9wb3BvdmVyXCI7XHJcbmltcG9ydCB7IFBsYXRlSW5wdXRDb21wb25lbnQgfSBmcm9tIFwiLi9wbGF0ZS1pbnB1dC5jb21wb25lbnRcIjtcclxuXHJcbkBOZ01vZHVsZSh7XHJcbiAgaW1wb3J0czogW1xyXG4gICAgQ29tbW9uTW9kdWxlLFxyXG4gICAgRm9ybXNNb2R1bGUsXHJcbiAgICBOekdyaWRNb2R1bGUsXHJcbiAgICBOeklucHV0TW9kdWxlLFxyXG4gICAgTnpQb3BvdmVyTW9kdWxlLFxyXG4gICAgTnpJY29uTW9kdWxlLFxyXG4gIF0sXHJcbiAgZGVjbGFyYXRpb25zOiBbUGxhdGVJbnB1dENvbXBvbmVudF0sXHJcbiAgcHJvdmlkZXJzOiBbXSxcclxuICBleHBvcnRzOiBbUGxhdGVJbnB1dENvbXBvbmVudF0sXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBQbGF0ZUlucHV0TW9kdWxlIHt9XHJcbiJdfQ==
@@ -0,0 +1,41 @@
1
+ import { Injectable } from "@angular/core";
2
+ import { HttpClient } from "@angular/common/http";
3
+ /**
4
+ * 默认的车牌前缀加载服务实现
5
+ * 使用 HttpClient 调用 API 获取车牌前缀选项
6
+ */
7
+ export class DefaultPlatePrefixLoadService {
8
+ constructor(http) {
9
+ this.http = http;
10
+ this.defaultApiUrl = "/PlateFirstWord/GetPlateFirstWord";
11
+ }
12
+ loadPrefixOptions() {
13
+ return this.http
14
+ .post(this.defaultApiUrl, { dictCode: "PlateFirstWord" })
15
+ .toPromise()
16
+ .then((list) => {
17
+ const options = [];
18
+ // 添加空选项(与旧版保持一致)
19
+ options.push({ label: "", value: "" });
20
+ // 转换数据格式
21
+ for (let i = 0; i < list.length; i++) {
22
+ options.push({
23
+ label: list[i],
24
+ value: list[i],
25
+ });
26
+ }
27
+ return options;
28
+ })
29
+ .catch(() => {
30
+ // 发生错误时返回空数组
31
+ return [];
32
+ });
33
+ }
34
+ }
35
+ DefaultPlatePrefixLoadService.decorators = [
36
+ { type: Injectable }
37
+ ];
38
+ DefaultPlatePrefixLoadService.ctorParameters = () => [
39
+ { type: HttpClient }
40
+ ];
41
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGxhdGUtcHJlZml4LWxvYWQuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiJEOi9wcm9qZWN0cy92cHMtZnJvbnQvRnJvbnQvRGFzUE1TV2ViL3Byb2plY3RzL3Byby10YWJsZS9zcmMvIiwic291cmNlcyI6WyJsaWIvcGxhdGUtaW5wdXQvcGxhdGUtcHJlZml4LWxvYWQuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNDLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUdsRDs7O0dBR0c7QUFFSCxNQUFNLE9BQU8sNkJBQTZCO0lBR3hDLFlBQW9CLElBQWdCO1FBQWhCLFNBQUksR0FBSixJQUFJLENBQVk7UUFGbkIsa0JBQWEsR0FBRyxtQ0FBbUMsQ0FBQztJQUU5QixDQUFDO0lBRXhDLGlCQUFpQjtRQUNmLE9BQU8sSUFBSSxDQUFDLElBQUk7YUFDYixJQUFJLENBQVcsSUFBSSxDQUFDLGFBQWEsRUFBRSxFQUFFLFFBQVEsRUFBRSxnQkFBZ0IsRUFBRSxDQUFDO2FBQ2xFLFNBQVMsRUFBRTthQUNYLElBQUksQ0FBQyxDQUFDLElBQWMsRUFBRSxFQUFFO1lBQ3ZCLE1BQU0sT0FBTyxHQUF3QixFQUFFLENBQUM7WUFDeEMsaUJBQWlCO1lBQ2pCLE9BQU8sQ0FBQyxJQUFJLENBQUMsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBQ3ZDLFNBQVM7WUFDVCxLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtnQkFDcEMsT0FBTyxDQUFDLElBQUksQ0FBQztvQkFDWCxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQztvQkFDZCxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQztpQkFDZixDQUFDLENBQUM7YUFDSjtZQUNELE9BQU8sT0FBTyxDQUFDO1FBQ2pCLENBQUMsQ0FBQzthQUNELEtBQUssQ0FBQyxHQUFHLEVBQUU7WUFDVixhQUFhO1lBQ2IsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDLENBQUMsQ0FBQztJQUNQLENBQUM7OztZQTNCRixVQUFVOzs7WUFQRixVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XHJcbmltcG9ydCB7IEh0dHBDbGllbnQgfSBmcm9tIFwiQGFuZ3VsYXIvY29tbW9uL2h0dHBcIjtcclxuaW1wb3J0IHsgUGxhdGVQcmVmaXhMb2FkU2VydmljZSwgUGxhdGVQcmVmaXhPcHRpb24gfSBmcm9tIFwiLi4vdG9rZW5zXCI7XHJcblxyXG4vKipcclxuICog6buY6K6k55qE6L2m54mM5YmN57yA5Yqg6L295pyN5Yqh5a6e546wXHJcbiAqIOS9v+eUqCBIdHRwQ2xpZW50IOiwg+eUqCBBUEkg6I635Y+W6L2m54mM5YmN57yA6YCJ6aG5XHJcbiAqL1xyXG5ASW5qZWN0YWJsZSgpXHJcbmV4cG9ydCBjbGFzcyBEZWZhdWx0UGxhdGVQcmVmaXhMb2FkU2VydmljZSBpbXBsZW1lbnRzIFBsYXRlUHJlZml4TG9hZFNlcnZpY2Uge1xyXG4gIHByaXZhdGUgcmVhZG9ubHkgZGVmYXVsdEFwaVVybCA9IFwiL1BsYXRlRmlyc3RXb3JkL0dldFBsYXRlRmlyc3RXb3JkXCI7XHJcblxyXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgaHR0cDogSHR0cENsaWVudCkge31cclxuXHJcbiAgbG9hZFByZWZpeE9wdGlvbnMoKTogUHJvbWlzZTxQbGF0ZVByZWZpeE9wdGlvbltdPiB7XHJcbiAgICByZXR1cm4gdGhpcy5odHRwXHJcbiAgICAgIC5wb3N0PHN0cmluZ1tdPih0aGlzLmRlZmF1bHRBcGlVcmwsIHsgZGljdENvZGU6IFwiUGxhdGVGaXJzdFdvcmRcIiB9KVxyXG4gICAgICAudG9Qcm9taXNlKClcclxuICAgICAgLnRoZW4oKGxpc3Q6IHN0cmluZ1tdKSA9PiB7XHJcbiAgICAgICAgY29uc3Qgb3B0aW9uczogUGxhdGVQcmVmaXhPcHRpb25bXSA9IFtdO1xyXG4gICAgICAgIC8vIOa3u+WKoOepuumAiemhue+8iOS4juaXp+eJiOS/neaMgeS4gOiHtO+8iVxyXG4gICAgICAgIG9wdGlvbnMucHVzaCh7IGxhYmVsOiBcIlwiLCB2YWx1ZTogXCJcIiB9KTtcclxuICAgICAgICAvLyDovazmjaLmlbDmja7moLzlvI9cclxuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGxpc3QubGVuZ3RoOyBpKyspIHtcclxuICAgICAgICAgIG9wdGlvbnMucHVzaCh7XHJcbiAgICAgICAgICAgIGxhYmVsOiBsaXN0W2ldLFxyXG4gICAgICAgICAgICB2YWx1ZTogbGlzdFtpXSxcclxuICAgICAgICAgIH0pO1xyXG4gICAgICAgIH1cclxuICAgICAgICByZXR1cm4gb3B0aW9ucztcclxuICAgICAgfSlcclxuICAgICAgLmNhdGNoKCgpID0+IHtcclxuICAgICAgICAvLyDlj5HnlJ/plJnor6/ml7bov5Tlm57nqbrmlbDnu4RcclxuICAgICAgICByZXR1cm4gW107XHJcbiAgICAgIH0pO1xyXG4gIH1cclxufVxyXG5cclxuIl19