raise-common-lib 0.0.205 → 0.0.206

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.
@@ -4,20 +4,24 @@
4
4
  * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
5
  */
6
6
  // 组件类
7
- import { Component, EventEmitter, Input, Output, } from "@angular/core";
7
+ import { Component, EventEmitter, Input, Output, ViewChild, ElementRef, HostListener, ChangeDetectorRef, } from "@angular/core";
8
8
  import { CommonFunctionService } from "../../service/common-function.service";
9
9
  export class RSStepperComponent {
10
10
  /**
11
11
  * @param {?} cf
12
+ * @param {?} ref
12
13
  */
13
- constructor(cf) {
14
+ constructor(cf, ref) {
14
15
  this.cf = cf;
16
+ this.ref = ref;
15
17
  this.steps = [];
16
18
  this.currentStep = 0;
17
19
  this.stepClick = new EventEmitter();
18
20
  this.unlockedStep = 0; //已解锁的最大步骤
21
+ this.showBtn = false;
22
+ this.isAtStart = true;
23
+ this.isAtEnd = false;
19
24
  }
20
- //已解锁的最大步骤
21
25
  /**
22
26
  * @return {?}
23
27
  */
@@ -31,6 +35,15 @@ export class RSStepperComponent {
31
35
  }));
32
36
  this.syncUnlockedStep();
33
37
  }
38
+ /**
39
+ * @return {?}
40
+ */
41
+ ngAfterViewInit() {
42
+ setTimeout((/**
43
+ * @return {?}
44
+ */
45
+ () => this.checkBtnShow(true)));
46
+ }
34
47
  /**
35
48
  * @param {?} changes
36
49
  * @return {?}
@@ -38,6 +51,18 @@ export class RSStepperComponent {
38
51
  ngOnChanges(changes) {
39
52
  if (changes["currentStep"] && !changes["currentStep"].firstChange) {
40
53
  this.syncUnlockedStep();
54
+ // 当外部传入的 currentStep 变化时,滚动到对应步骤
55
+ setTimeout((/**
56
+ * @return {?}
57
+ */
58
+ () => this.scrollToStep()));
59
+ }
60
+ if ((changes["steps"] && changes["steps"].currentValue) ||
61
+ changes["currentStep"]) {
62
+ setTimeout((/**
63
+ * @return {?}
64
+ */
65
+ () => this.checkBtnShow()));
41
66
  }
42
67
  }
43
68
  // 同步unlockedStep的最大值
@@ -52,33 +77,161 @@ export class RSStepperComponent {
52
77
  }
53
78
  /**
54
79
  * @param {?} step
80
+ * @param {?=} index
55
81
  * @return {?}
56
82
  */
57
- onStepClick(step) {
83
+ onStepClick(step, index) {
58
84
  if (step.step <= this.unlockedStep) {
59
85
  this.currentStep = step.step;
60
86
  this.stepClick.emit(step);
87
+ setTimeout((/**
88
+ * @return {?}
89
+ */
90
+ () => this.scrollToStep(index)));
91
+ }
92
+ }
93
+ /**
94
+ * @private
95
+ * @param {?=} index
96
+ * @return {?}
97
+ */
98
+ scrollToStep(index) {
99
+ if (!this.menu || !this.menu.nativeElement)
100
+ return;
101
+ /** @type {?} */
102
+ const menuEl = this.menu.nativeElement;
103
+ /** @type {?} */
104
+ const items = menuEl.querySelectorAll(".step");
105
+ /** @type {?} */
106
+ const targetIndex = typeof index === "number"
107
+ ? index
108
+ : this.steps.findIndex((/**
109
+ * @param {?} s
110
+ * @return {?}
111
+ */
112
+ (s) => s.step === this.currentStep));
113
+ if (targetIndex < 0 || targetIndex >= items.length)
114
+ return;
115
+ /** @type {?} */
116
+ const targetEl = (/** @type {?} */ (items[targetIndex]));
117
+ //获取相对位置
118
+ /** @type {?} */
119
+ const containerRect = menuEl.getBoundingClientRect();
120
+ /** @type {?} */
121
+ const elementRect = targetEl.getBoundingClientRect();
122
+ // 计算元素相对于容器的位置
123
+ /** @type {?} */
124
+ const elementRelativeLeft = elementRect.left - containerRect.left + menuEl.scrollLeft;
125
+ /** @type {?} */
126
+ const elementWidth = elementRect.width;
127
+ /** @type {?} */
128
+ const containerWidth = menuEl.clientWidth;
129
+ // 计算元素中心点相对于容器的位置
130
+ /** @type {?} */
131
+ const elementCenter = elementRelativeLeft + elementWidth / 2;
132
+ // 目标滚动位置:让元素中心出现在可视区域中心
133
+ /** @type {?} */
134
+ const scrollTarget = elementCenter - containerWidth / 2;
135
+ // 限制在有效滚动范围内
136
+ /** @type {?} */
137
+ const maxScroll = Math.max(0, menuEl.scrollWidth - containerWidth);
138
+ /** @type {?} */
139
+ const finalTarget = Math.max(0, Math.min(scrollTarget, maxScroll));
140
+ menuEl.scrollTo({ left: finalTarget, behavior: "smooth" });
141
+ this.onScroll();
142
+ }
143
+ /**
144
+ * @param {?} event
145
+ * @return {?}
146
+ */
147
+ onResize(event) {
148
+ this.checkBtnShow();
149
+ }
150
+ /**
151
+ * @param {?=} init
152
+ * @return {?}
153
+ */
154
+ checkBtnShow(init) {
155
+ if (!this.menu || !this.menu.nativeElement)
156
+ return;
157
+ /** @type {?} */
158
+ const menuEl = this.menu.nativeElement;
159
+ const { scrollWidth, clientWidth } = menuEl;
160
+ // 判断是否需要显示滚动按钮
161
+ /** @type {?} */
162
+ let threshold = clientWidth;
163
+ if (this.containerRefId) {
164
+ /** @type {?} */
165
+ const containerRef = document.getElementById(this.containerRefId);
166
+ if (containerRef) {
167
+ threshold = Math.max(containerRef.offsetWidth * 0.8, 400);
168
+ }
61
169
  }
170
+ this.showBtn = scrollWidth > threshold;
171
+ // 初始化时滚动到当前步骤
172
+ if (init) {
173
+ setTimeout((/**
174
+ * @return {?}
175
+ */
176
+ () => this.scrollToStep()), 0);
177
+ }
178
+ this.onScroll();
179
+ this.ref.markForCheck();
180
+ }
181
+ /**
182
+ * @return {?}
183
+ */
184
+ scrollLeft() {
185
+ const { scrollLeft } = this.menu.nativeElement;
186
+ /** @type {?} */
187
+ const dis = scrollLeft / 100 < 2 ? scrollLeft + 10 : 100;
188
+ this.menu.nativeElement.scrollBy({ left: -dis, behavior: "smooth" });
189
+ }
190
+ /**
191
+ * @return {?}
192
+ */
193
+ scrollRight() {
194
+ const { scrollLeft, scrollWidth, clientWidth } = this.menu.nativeElement;
195
+ /** @type {?} */
196
+ const rest = scrollWidth - clientWidth - scrollLeft;
197
+ /** @type {?} */
198
+ const dis = rest / 100 < 2 ? rest + 10 : 100;
199
+ this.menu.nativeElement.scrollBy({ left: dis, behavior: "smooth" });
200
+ }
201
+ /**
202
+ * @return {?}
203
+ */
204
+ onScroll() {
205
+ const { scrollLeft, scrollWidth, clientWidth } = this.menu.nativeElement;
206
+ this.isAtStart = scrollLeft <= 1;
207
+ this.isAtEnd = scrollLeft + clientWidth >= scrollWidth - 1;
208
+ this.ref.markForCheck();
62
209
  }
63
210
  }
64
211
  RSStepperComponent.decorators = [
65
212
  { type: Component, args: [{
66
213
  selector: "rs-stepper",
67
- template: "<div class=\"rs-stepper\">\r\n <div class=\"steps-wrap\">\r\n <ng-container *ngFor=\"let item of steps; let i = index\">\r\n <div\r\n class=\"step\"\r\n [ngClass]=\"{\r\n done: item.step <= unlockedStep,\r\n active: item.step === currentStep\r\n }\"\r\n (click)=\"onStepClick(item)\"\r\n >\r\n <div\r\n class=\"step-label\"\r\n [matTooltip]=\"item.label !== item.displayTitle ? item.label : ''\"\r\n matTooltipPosition=\"above\"\r\n >\r\n {{ item.displayTitle }}\r\n </div>\r\n </div>\r\n <div class=\"step-arrow\" *ngIf=\"i < steps.length - 1\">\r\n <img src=\"/assets/img/step-arrow.svg\" />\r\n </div>\r\n </ng-container>\r\n </div>\r\n</div>\r\n",
68
- styles: [".rs-stepper{font-family:Arial;width:100%;border-bottom:1px solid var(--rs-border-color)}.rs-stepper .steps-wrap{display:flex;align-items:center;justify-content:space-between;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;margin:0 auto;gap:8px}.rs-stepper .steps-wrap .step{flex:1}.rs-stepper .steps-wrap .step .step-label{color:var(--rs-labels-color);font-size:12px;font-weight:400;line-height:22px;text-align:center;white-space:nowrap;padding:0 12px 12px;display:flex;align-items:center}.rs-stepper .steps-wrap .step.done{cursor:pointer}.rs-stepper .steps-wrap .step.done .step-label{color:var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step.active .step-label{color:var(--rs-active-labels-color);font-weight:700;border-bottom:1px solid var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step-arrow{padding:0 12px 12px;display:flex;align-items:center}"]
214
+ template: "<div class=\"rs-stepper\">\r\n <div class=\"steps-wrap\">\r\n <div\r\n class=\"row-btn\"\r\n [class.hidden]=\"isAtStart || !showBtn\"\r\n (click)=\"scrollLeft()\"\r\n >\r\n <div class=\"hover\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M10.4995 3.99955L7.24219 8.24219L10.4995 12.4848\"\r\n stroke=\"#1F7BFF\"\r\n />\r\n </svg>\r\n </div>\r\n <div class=\"normal\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M10.4995 3.99955L7.24219 8.24219L10.4995 12.4848\"\r\n stroke=\"#6B6B6B\"\r\n />\r\n </svg>\r\n </div>\r\n </div>\r\n <div\r\n class=\"steps-content\"\r\n [ngClass]=\"{\r\n leftMask: showBtn && !isAtStart,\r\n rightMask: showBtn && !isAtEnd,\r\n bothMask: showBtn && !isAtStart && showBtn && !isAtEnd\r\n }\"\r\n #menu\r\n (scroll)=\"onScroll()\"\r\n >\r\n <ng-container *ngFor=\"let item of steps; let i = index\">\r\n <div\r\n class=\"step\"\r\n [ngClass]=\"{\r\n done: item.step <= unlockedStep,\r\n active: item.step === currentStep\r\n }\"\r\n (click)=\"onStepClick(item, i)\"\r\n >\r\n <div\r\n class=\"step-label\"\r\n [matTooltip]=\"item.label !== item.displayTitle ? item.label : ''\"\r\n matTooltipPosition=\"above\"\r\n >\r\n {{ item.displayTitle }}\r\n </div>\r\n </div>\r\n <div class=\"step-arrow\" *ngIf=\"i < steps.length - 1\">\r\n <img src=\"/assets/img/step-arrow.svg\" />\r\n </div>\r\n </ng-container>\r\n </div>\r\n <div\r\n class=\"row-btn\"\r\n [class.hidden]=\"isAtEnd || !showBtn\"\r\n (click)=\"scrollRight()\"\r\n >\r\n <div class=\"hover\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M7.50045 3.99955L10.7578 8.24219L7.50045 12.4848\"\r\n stroke=\"#1F7BFF\"\r\n />\r\n </svg>\r\n </div>\r\n <div class=\"normal\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M7.50045 3.99955L10.7578 8.24219L7.50045 12.4848\"\r\n stroke=\"#6B6B6B\"\r\n />\r\n </svg>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n",
215
+ styles: [".rs-stepper{font-family:Arial;width:100%;border-bottom:1px solid var(--rs-border-color)}.rs-stepper .steps-wrap{display:flex;align-items:center;justify-content:center;gap:12px;margin:0 auto}.rs-stepper .steps-wrap .steps-content{display:flex;align-items:center;flex-wrap:nowrap;min-width:0;overflow-x:auto;overflow-y:hidden;gap:8px}.rs-stepper .steps-wrap .steps-content.leftMask{-webkit-mask-image:linear-gradient(90deg,transparent,#fff 10%,#d8d3d3 100%);mask-image:linear-gradient(90deg,transparent,#fff 10%,#d8d3d3 100%)}.rs-stepper .steps-wrap .steps-content.rightMask{-webkit-mask-image:linear-gradient(90deg,#fff 0,#fff 90%,transparent);mask-image:linear-gradient(90deg,#fff 0,#fff 90%,transparent)}.rs-stepper .steps-wrap .steps-content.bothMask{-webkit-mask-image:linear-gradient(90deg,transparent,#fff 10%,#fff 90%,transparent)!important;mask-image:linear-gradient(90deg,transparent,#fff 10%,#fff 90%,transparent)!important}.rs-stepper .steps-wrap .steps-content::-webkit-scrollbar{width:0;height:0}.rs-stepper .steps-wrap .step{flex:0 0 auto}.rs-stepper .steps-wrap .step .step-label{color:var(--rs-labels-color);font-size:12px;font-weight:400;line-height:22px;text-align:center;white-space:nowrap;padding:0 12px 12px;display:flex;align-items:center}.rs-stepper .steps-wrap .step.done{cursor:pointer}.rs-stepper .steps-wrap .step.done .step-label{color:var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step.active .step-label{color:var(--rs-active-labels-color);font-weight:700;border-bottom:1px solid var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step-arrow{flex:0 0 auto;padding:0 12px 12px;display:flex;align-items:center}.rs-stepper .row-btn{margin-bottom:12px;display:flex;width:20px;height:20px;align-items:center;justify-content:center;border-radius:4px;border:1px solid #bdc4ca;flex-shrink:0;cursor:pointer;visibility:visible}.rs-stepper .row-btn .hover{display:none}.rs-stepper .row-btn .normal{display:flex}.rs-stepper .row-btn:hover{background-color:rgba(31,123,255,.04)}.rs-stepper .row-btn:hover .normal{display:none}.rs-stepper .row-btn:hover .hover{display:flex}.rs-stepper .row-btn.hidden{visibility:hidden}"]
69
216
  }] }
70
217
  ];
71
218
  /** @nocollapse */
72
219
  RSStepperComponent.ctorParameters = () => [
73
- { type: CommonFunctionService }
220
+ { type: CommonFunctionService },
221
+ { type: ChangeDetectorRef }
74
222
  ];
75
223
  RSStepperComponent.propDecorators = {
224
+ menu: [{ type: ViewChild, args: ["menu", { static: false },] }],
76
225
  steps: [{ type: Input }],
77
226
  currentStep: [{ type: Input }],
78
227
  stepClick: [{ type: Output }],
79
- unlockedStep: [{ type: Input }]
228
+ unlockedStep: [{ type: Input }],
229
+ containerRefId: [{ type: Input }],
230
+ onResize: [{ type: HostListener, args: ["window:resize", ["$event"],] }]
80
231
  };
81
232
  if (false) {
233
+ /** @type {?} */
234
+ RSStepperComponent.prototype.menu;
82
235
  /** @type {?} */
83
236
  RSStepperComponent.prototype.steps;
84
237
  /** @type {?} */
@@ -88,6 +241,19 @@ if (false) {
88
241
  /** @type {?} */
89
242
  RSStepperComponent.prototype.unlockedStep;
90
243
  /** @type {?} */
244
+ RSStepperComponent.prototype.containerRefId;
245
+ /** @type {?} */
246
+ RSStepperComponent.prototype.showBtn;
247
+ /** @type {?} */
248
+ RSStepperComponent.prototype.isAtStart;
249
+ /** @type {?} */
250
+ RSStepperComponent.prototype.isAtEnd;
251
+ /** @type {?} */
91
252
  RSStepperComponent.prototype.cf;
253
+ /**
254
+ * @type {?}
255
+ * @private
256
+ */
257
+ RSStepperComponent.prototype.ref;
92
258
  }
93
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6Im5nOi8vcmFpc2UtY29tbW9uLWxpYi8iLCJzb3VyY2VzIjpbImxpYi9sYXlvdXQvcnMtc3RlcHBlci9pbmRleC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQ0EsT0FBTyxFQUNMLFNBQVMsRUFDVCxZQUFZLEVBQ1osS0FBSyxFQUVMLE1BQU0sR0FFUCxNQUFNLGVBQWUsQ0FBQztBQUV2QixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQU85RSxNQUFNLE9BQU8sa0JBQWtCOzs7O0lBRzdCLFlBQW1CLEVBQXlCO1FBQXpCLE9BQUUsR0FBRixFQUFFLENBQXVCO1FBRm5DLFVBQUssR0FBZSxFQUFFLENBQUM7UUFDdkIsZ0JBQVcsR0FBVyxDQUFDLENBQUM7UUFFdkIsY0FBUyxHQUFHLElBQUksWUFBWSxFQUFZLENBQUM7UUFDMUMsaUJBQVksR0FBVyxDQUFDLENBQUMsQ0FBQyxVQUFVO0lBRkUsQ0FBQzs7Ozs7SUFJaEQsUUFBUTtRQUNOLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTzs7OztRQUFDLENBQUMsSUFBSSxFQUFFLEVBQUU7WUFDMUIsSUFBSSxDQUFDLFlBQVksR0FBRyxJQUFJLENBQUMsRUFBRSxDQUFDLGlCQUFpQixDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsR0FBRyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBQ3JFLENBQUMsRUFBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLGdCQUFnQixFQUFFLENBQUM7SUFDMUIsQ0FBQzs7Ozs7SUFFRCxXQUFXLENBQUMsT0FBc0I7UUFDaEMsSUFBSSxPQUFPLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUMsV0FBVyxFQUFFO1lBQ2pFLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1NBQ3pCO0lBQ0gsQ0FBQzs7Ozs7O0lBR08sZ0JBQWdCO1FBQ3RCLElBQUksSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUMsWUFBWSxFQUFFO1lBQ3hDLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQztTQUN0QztJQUNILENBQUM7Ozs7O0lBRUQsV0FBVyxDQUFDLElBQWM7UUFDeEIsSUFBSSxJQUFJLENBQUMsSUFBSSxJQUFJLElBQUksQ0FBQyxZQUFZLEVBQUU7WUFDbEMsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUMsSUFBSSxDQUFDO1lBQzdCLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1NBQzNCO0lBQ0gsQ0FBQzs7O1lBckNGLFNBQVMsU0FBQztnQkFDVCxRQUFRLEVBQUUsWUFBWTtnQkFDdEIsb3lCQUFxQzs7YUFFdEM7Ozs7WUFOUSxxQkFBcUI7OztvQkFRM0IsS0FBSzswQkFDTCxLQUFLO3dCQUVMLE1BQU07MkJBQ04sS0FBSzs7OztJQUpOLG1DQUFnQzs7SUFDaEMseUNBQWlDOztJQUVqQyx1Q0FBbUQ7O0lBQ25ELDBDQUFrQzs7SUFGdEIsZ0NBQWdDIiwic291cmNlc0NvbnRlbnQiOlsiLy8g57uE5Lu257G7XHJcbmltcG9ydCB7XHJcbiAgQ29tcG9uZW50LFxyXG4gIEV2ZW50RW1pdHRlcixcclxuICBJbnB1dCxcclxuICBPbkluaXQsXHJcbiAgT3V0cHV0LFxyXG4gIFNpbXBsZUNoYW5nZXMsXHJcbn0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcclxuaW1wb3J0IHsgU3RlcEl0ZW0gfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcclxuaW1wb3J0IHsgQ29tbW9uRnVuY3Rpb25TZXJ2aWNlIH0gZnJvbSBcIi4uLy4uL3NlcnZpY2UvY29tbW9uLWZ1bmN0aW9uLnNlcnZpY2VcIjtcclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiBcInJzLXN0ZXBwZXJcIixcclxuICB0ZW1wbGF0ZVVybDogXCIuL2luZGV4LmNvbXBvbmVudC5odG1sXCIsXHJcbiAgc3R5bGVVcmxzOiBbXCIuL2luZGV4LmNvbXBvbmVudC5zY3NzXCJdLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgUlNTdGVwcGVyQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcclxuICBASW5wdXQoKSBzdGVwczogU3RlcEl0ZW1bXSA9IFtdO1xyXG4gIEBJbnB1dCgpIGN1cnJlbnRTdGVwOiBudW1iZXIgPSAwO1xyXG4gIGNvbnN0cnVjdG9yKHB1YmxpYyBjZjogQ29tbW9uRnVuY3Rpb25TZXJ2aWNlKSB7fVxyXG4gIEBPdXRwdXQoKSBzdGVwQ2xpY2sgPSBuZXcgRXZlbnRFbWl0dGVyPFN0ZXBJdGVtPigpO1xyXG4gIEBJbnB1dCgpIHVubG9ja2VkU3RlcDogbnVtYmVyID0gMDsgLy/lt7Lop6PplIHnmoTmnIDlpKfmraXpqqRcclxuXHJcbiAgbmdPbkluaXQoKSB7XHJcbiAgICB0aGlzLnN0ZXBzLmZvckVhY2goKHN0ZXApID0+IHtcclxuICAgICAgc3RlcC5kaXNwbGF5VGl0bGUgPSB0aGlzLmNmLnNldE1pZGRsZUVsbGlwc2lzKHN0ZXAubGFiZWwsIDEzMiwgMTIpO1xyXG4gICAgfSk7XHJcbiAgICB0aGlzLnN5bmNVbmxvY2tlZFN0ZXAoKTtcclxuICB9XHJcblxyXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpIHtcclxuICAgIGlmIChjaGFuZ2VzW1wiY3VycmVudFN0ZXBcIl0gJiYgIWNoYW5nZXNbXCJjdXJyZW50U3RlcFwiXS5maXJzdENoYW5nZSkge1xyXG4gICAgICB0aGlzLnN5bmNVbmxvY2tlZFN0ZXAoKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIC8vIOWQjOatpXVubG9ja2VkU3RlcOeahOacgOWkp+WAvFxyXG4gIHByaXZhdGUgc3luY1VubG9ja2VkU3RlcCgpIHtcclxuICAgIGlmICh0aGlzLmN1cnJlbnRTdGVwID4gdGhpcy51bmxvY2tlZFN0ZXApIHtcclxuICAgICAgdGhpcy51bmxvY2tlZFN0ZXAgPSB0aGlzLmN1cnJlbnRTdGVwO1xyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgb25TdGVwQ2xpY2soc3RlcDogU3RlcEl0ZW0pIHtcclxuICAgIGlmIChzdGVwLnN0ZXAgPD0gdGhpcy51bmxvY2tlZFN0ZXApIHtcclxuICAgICAgdGhpcy5jdXJyZW50U3RlcCA9IHN0ZXAuc3RlcDtcclxuICAgICAgdGhpcy5zdGVwQ2xpY2suZW1pdChzdGVwKTtcclxuICAgIH1cclxuICB9XHJcbn1cclxuIl19
259
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.component.js","sourceRoot":"ng://raise-common-lib/","sources":["lib/layout/rs-stepper/index.component.ts"],"names":[],"mappings":"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","sourcesContent":["// 组件类\r\nimport {\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnInit,\r\n  Output,\r\n  SimpleChanges,\r\n  ViewChild,\r\n  ElementRef,\r\n  HostListener,\r\n  ChangeDetectorRef,\r\n  AfterViewInit,\r\n} from \"@angular/core\";\r\nimport { StepItem } from \"./constants\";\r\nimport { CommonFunctionService } from \"../../service/common-function.service\";\r\n\r\n@Component({\r\n  selector: \"rs-stepper\",\r\n  templateUrl: \"./index.component.html\",\r\n  styleUrls: [\"./index.component.scss\"],\r\n})\r\nexport class RSStepperComponent implements OnInit, AfterViewInit {\r\n  @ViewChild(\"menu\", { static: false }) menu!: ElementRef;\r\n  @Input() steps: StepItem[] = [];\r\n  @Input() currentStep: number = 0;\r\n  constructor(\r\n    public cf: CommonFunctionService,\r\n    private ref: ChangeDetectorRef\r\n  ) {}\r\n  @Output() stepClick = new EventEmitter<StepItem>();\r\n  @Input() unlockedStep: number = 0; //已解锁的最大步骤\r\n  @Input() containerRefId;\r\n\r\n  ngOnInit() {\r\n    this.steps.forEach((step) => {\r\n      step.displayTitle = this.cf.setMiddleEllipsis(step.label, 132, 12);\r\n    });\r\n    this.syncUnlockedStep();\r\n  }\r\n\r\n  ngAfterViewInit() {\r\n    setTimeout(() => this.checkBtnShow(true));\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    if (changes[\"currentStep\"] && !changes[\"currentStep\"].firstChange) {\r\n      this.syncUnlockedStep();\r\n      // 当外部传入的 currentStep 变化时，滚动到对应步骤\r\n      setTimeout(() => this.scrollToStep());\r\n    }\r\n    if (\r\n      (changes[\"steps\"] && changes[\"steps\"].currentValue) ||\r\n      changes[\"currentStep\"]\r\n    ) {\r\n      setTimeout(() => this.checkBtnShow());\r\n    }\r\n  }\r\n\r\n  // 同步unlockedStep的最大值\r\n  private syncUnlockedStep() {\r\n    if (this.currentStep > this.unlockedStep) {\r\n      this.unlockedStep = this.currentStep;\r\n    }\r\n  }\r\n\r\n  onStepClick(step: StepItem, index?: number) {\r\n    if (step.step <= this.unlockedStep) {\r\n      this.currentStep = step.step;\r\n      this.stepClick.emit(step);\r\n      setTimeout(() => this.scrollToStep(index));\r\n    }\r\n  }\r\n\r\n  private scrollToStep(index?: number) {\r\n    if (!this.menu || !this.menu.nativeElement) return;\r\n\r\n    const menuEl = this.menu.nativeElement;\r\n    const items = menuEl.querySelectorAll(\".step\");\r\n    const targetIndex =\r\n      typeof index === \"number\"\r\n        ? index\r\n        : this.steps.findIndex((s) => s.step === this.currentStep);\r\n\r\n    if (targetIndex < 0 || targetIndex >= items.length) return;\r\n\r\n    const targetEl = items[targetIndex] as HTMLElement;\r\n\r\n    //获取相对位置\r\n    const containerRect = menuEl.getBoundingClientRect();\r\n    const elementRect = targetEl.getBoundingClientRect();\r\n\r\n    // 计算元素相对于容器的位置\r\n    const elementRelativeLeft =\r\n      elementRect.left - containerRect.left + menuEl.scrollLeft;\r\n    const elementWidth = elementRect.width;\r\n    const containerWidth = menuEl.clientWidth;\r\n\r\n    // 计算元素中心点相对于容器的位置\r\n    const elementCenter = elementRelativeLeft + elementWidth / 2;\r\n\r\n    // 目标滚动位置：让元素中心出现在可视区域中心\r\n    const scrollTarget = elementCenter - containerWidth / 2;\r\n\r\n    // 限制在有效滚动范围内\r\n    const maxScroll = Math.max(0, menuEl.scrollWidth - containerWidth);\r\n    const finalTarget = Math.max(0, Math.min(scrollTarget, maxScroll));\r\n\r\n    menuEl.scrollTo({ left: finalTarget, behavior: \"smooth\" });\r\n    this.onScroll();\r\n  }\r\n\r\n  @HostListener(\"window:resize\", [\"$event\"])\r\n  onResize(event: Event) {\r\n    this.checkBtnShow();\r\n  }\r\n\r\n  checkBtnShow(init?: boolean) {\r\n    if (!this.menu || !this.menu.nativeElement) return;\r\n\r\n    const menuEl = this.menu.nativeElement;\r\n    const { scrollWidth, clientWidth } = menuEl;\r\n\r\n    // 判断是否需要显示滚动按钮\r\n    let threshold = clientWidth;\r\n    if (this.containerRefId) {\r\n      const containerRef = document.getElementById(this.containerRefId);\r\n      if (containerRef) {\r\n        threshold = Math.max(containerRef.offsetWidth * 0.8, 400);\r\n      }\r\n    }\r\n\r\n    this.showBtn = scrollWidth > threshold;\r\n\r\n    // 初始化时滚动到当前步骤\r\n    if (init) {\r\n      setTimeout(() => this.scrollToStep(), 0);\r\n    }\r\n\r\n    this.onScroll();\r\n    this.ref.markForCheck();\r\n  }\r\n  scrollLeft() {\r\n    const { scrollLeft } = this.menu.nativeElement;\r\n    const dis = scrollLeft / 100 < 2 ? scrollLeft + 10 : 100;\r\n    this.menu.nativeElement.scrollBy({ left: -dis, behavior: \"smooth\" });\r\n  }\r\n  scrollRight() {\r\n    const { scrollLeft, scrollWidth, clientWidth } = this.menu.nativeElement;\r\n    const rest = scrollWidth - clientWidth - scrollLeft;\r\n    const dis = rest / 100 < 2 ? rest + 10 : 100;\r\n    this.menu.nativeElement.scrollBy({ left: dis, behavior: \"smooth\" });\r\n  }\r\n\r\n  showBtn = false;\r\n  isAtStart = true;\r\n  isAtEnd = false;\r\n  onScroll() {\r\n    const { scrollLeft, scrollWidth, clientWidth } = this.menu.nativeElement;\r\n    this.isAtStart = scrollLeft <= 1;\r\n    this.isAtEnd = scrollLeft + clientWidth >= scrollWidth - 1;\r\n    this.ref.markForCheck();\r\n  }\r\n}\r\n"]}
@@ -4,23 +4,24 @@
4
4
  * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
5
5
  */
6
6
  // 组件类
7
- import { Component, EventEmitter, Input, Output, } from "@angular/core";
7
+ import { Component, EventEmitter, Input, Output, ViewChild, ElementRef, HostListener, ChangeDetectorRef, } from "@angular/core";
8
8
  import { CommonFunctionService } from "../../service/common-function.service";
9
9
  var RSStepperComponent = /** @class */ (function () {
10
- function RSStepperComponent(cf) {
10
+ function RSStepperComponent(cf, ref) {
11
11
  this.cf = cf;
12
+ this.ref = ref;
12
13
  this.steps = [];
13
14
  this.currentStep = 0;
14
15
  this.stepClick = new EventEmitter();
15
16
  this.unlockedStep = 0; //已解锁的最大步骤
17
+ this.showBtn = false;
18
+ this.isAtStart = true;
19
+ this.isAtEnd = false;
16
20
  }
17
- //已解锁的最大步骤
18
21
  /**
19
22
  * @return {?}
20
23
  */
21
- RSStepperComponent.prototype.ngOnInit =
22
- //已解锁的最大步骤
23
- /**
24
+ RSStepperComponent.prototype.ngOnInit = /**
24
25
  * @return {?}
25
26
  */
26
27
  function () {
@@ -34,6 +35,19 @@ var RSStepperComponent = /** @class */ (function () {
34
35
  }));
35
36
  this.syncUnlockedStep();
36
37
  };
38
+ /**
39
+ * @return {?}
40
+ */
41
+ RSStepperComponent.prototype.ngAfterViewInit = /**
42
+ * @return {?}
43
+ */
44
+ function () {
45
+ var _this = this;
46
+ setTimeout((/**
47
+ * @return {?}
48
+ */
49
+ function () { return _this.checkBtnShow(true); }));
50
+ };
37
51
  /**
38
52
  * @param {?} changes
39
53
  * @return {?}
@@ -43,8 +57,21 @@ var RSStepperComponent = /** @class */ (function () {
43
57
  * @return {?}
44
58
  */
45
59
  function (changes) {
60
+ var _this = this;
46
61
  if (changes["currentStep"] && !changes["currentStep"].firstChange) {
47
62
  this.syncUnlockedStep();
63
+ // 当外部传入的 currentStep 变化时,滚动到对应步骤
64
+ setTimeout((/**
65
+ * @return {?}
66
+ */
67
+ function () { return _this.scrollToStep(); }));
68
+ }
69
+ if ((changes["steps"] && changes["steps"].currentValue) ||
70
+ changes["currentStep"]) {
71
+ setTimeout((/**
72
+ * @return {?}
73
+ */
74
+ function () { return _this.checkBtnShow(); }));
48
75
  }
49
76
  };
50
77
  // 同步unlockedStep的最大值
@@ -66,39 +93,193 @@ var RSStepperComponent = /** @class */ (function () {
66
93
  };
67
94
  /**
68
95
  * @param {?} step
96
+ * @param {?=} index
69
97
  * @return {?}
70
98
  */
71
99
  RSStepperComponent.prototype.onStepClick = /**
72
100
  * @param {?} step
101
+ * @param {?=} index
73
102
  * @return {?}
74
103
  */
75
- function (step) {
104
+ function (step, index) {
105
+ var _this = this;
76
106
  if (step.step <= this.unlockedStep) {
77
107
  this.currentStep = step.step;
78
108
  this.stepClick.emit(step);
109
+ setTimeout((/**
110
+ * @return {?}
111
+ */
112
+ function () { return _this.scrollToStep(index); }));
79
113
  }
80
114
  };
115
+ /**
116
+ * @private
117
+ * @param {?=} index
118
+ * @return {?}
119
+ */
120
+ RSStepperComponent.prototype.scrollToStep = /**
121
+ * @private
122
+ * @param {?=} index
123
+ * @return {?}
124
+ */
125
+ function (index) {
126
+ var _this = this;
127
+ if (!this.menu || !this.menu.nativeElement)
128
+ return;
129
+ /** @type {?} */
130
+ var menuEl = this.menu.nativeElement;
131
+ /** @type {?} */
132
+ var items = menuEl.querySelectorAll(".step");
133
+ /** @type {?} */
134
+ var targetIndex = typeof index === "number"
135
+ ? index
136
+ : this.steps.findIndex((/**
137
+ * @param {?} s
138
+ * @return {?}
139
+ */
140
+ function (s) { return s.step === _this.currentStep; }));
141
+ if (targetIndex < 0 || targetIndex >= items.length)
142
+ return;
143
+ /** @type {?} */
144
+ var targetEl = (/** @type {?} */ (items[targetIndex]));
145
+ //获取相对位置
146
+ /** @type {?} */
147
+ var containerRect = menuEl.getBoundingClientRect();
148
+ /** @type {?} */
149
+ var elementRect = targetEl.getBoundingClientRect();
150
+ // 计算元素相对于容器的位置
151
+ /** @type {?} */
152
+ var elementRelativeLeft = elementRect.left - containerRect.left + menuEl.scrollLeft;
153
+ /** @type {?} */
154
+ var elementWidth = elementRect.width;
155
+ /** @type {?} */
156
+ var containerWidth = menuEl.clientWidth;
157
+ // 计算元素中心点相对于容器的位置
158
+ /** @type {?} */
159
+ var elementCenter = elementRelativeLeft + elementWidth / 2;
160
+ // 目标滚动位置:让元素中心出现在可视区域中心
161
+ /** @type {?} */
162
+ var scrollTarget = elementCenter - containerWidth / 2;
163
+ // 限制在有效滚动范围内
164
+ /** @type {?} */
165
+ var maxScroll = Math.max(0, menuEl.scrollWidth - containerWidth);
166
+ /** @type {?} */
167
+ var finalTarget = Math.max(0, Math.min(scrollTarget, maxScroll));
168
+ menuEl.scrollTo({ left: finalTarget, behavior: "smooth" });
169
+ this.onScroll();
170
+ };
171
+ /**
172
+ * @param {?} event
173
+ * @return {?}
174
+ */
175
+ RSStepperComponent.prototype.onResize = /**
176
+ * @param {?} event
177
+ * @return {?}
178
+ */
179
+ function (event) {
180
+ this.checkBtnShow();
181
+ };
182
+ /**
183
+ * @param {?=} init
184
+ * @return {?}
185
+ */
186
+ RSStepperComponent.prototype.checkBtnShow = /**
187
+ * @param {?=} init
188
+ * @return {?}
189
+ */
190
+ function (init) {
191
+ var _this = this;
192
+ if (!this.menu || !this.menu.nativeElement)
193
+ return;
194
+ /** @type {?} */
195
+ var menuEl = this.menu.nativeElement;
196
+ var scrollWidth = menuEl.scrollWidth, clientWidth = menuEl.clientWidth;
197
+ // 判断是否需要显示滚动按钮
198
+ /** @type {?} */
199
+ var threshold = clientWidth;
200
+ if (this.containerRefId) {
201
+ /** @type {?} */
202
+ var containerRef = document.getElementById(this.containerRefId);
203
+ if (containerRef) {
204
+ threshold = Math.max(containerRef.offsetWidth * 0.8, 400);
205
+ }
206
+ }
207
+ this.showBtn = scrollWidth > threshold;
208
+ // 初始化时滚动到当前步骤
209
+ if (init) {
210
+ setTimeout((/**
211
+ * @return {?}
212
+ */
213
+ function () { return _this.scrollToStep(); }), 0);
214
+ }
215
+ this.onScroll();
216
+ this.ref.markForCheck();
217
+ };
218
+ /**
219
+ * @return {?}
220
+ */
221
+ RSStepperComponent.prototype.scrollLeft = /**
222
+ * @return {?}
223
+ */
224
+ function () {
225
+ var scrollLeft = this.menu.nativeElement.scrollLeft;
226
+ /** @type {?} */
227
+ var dis = scrollLeft / 100 < 2 ? scrollLeft + 10 : 100;
228
+ this.menu.nativeElement.scrollBy({ left: -dis, behavior: "smooth" });
229
+ };
230
+ /**
231
+ * @return {?}
232
+ */
233
+ RSStepperComponent.prototype.scrollRight = /**
234
+ * @return {?}
235
+ */
236
+ function () {
237
+ var _a = this.menu.nativeElement, scrollLeft = _a.scrollLeft, scrollWidth = _a.scrollWidth, clientWidth = _a.clientWidth;
238
+ /** @type {?} */
239
+ var rest = scrollWidth - clientWidth - scrollLeft;
240
+ /** @type {?} */
241
+ var dis = rest / 100 < 2 ? rest + 10 : 100;
242
+ this.menu.nativeElement.scrollBy({ left: dis, behavior: "smooth" });
243
+ };
244
+ /**
245
+ * @return {?}
246
+ */
247
+ RSStepperComponent.prototype.onScroll = /**
248
+ * @return {?}
249
+ */
250
+ function () {
251
+ var _a = this.menu.nativeElement, scrollLeft = _a.scrollLeft, scrollWidth = _a.scrollWidth, clientWidth = _a.clientWidth;
252
+ this.isAtStart = scrollLeft <= 1;
253
+ this.isAtEnd = scrollLeft + clientWidth >= scrollWidth - 1;
254
+ this.ref.markForCheck();
255
+ };
81
256
  RSStepperComponent.decorators = [
82
257
  { type: Component, args: [{
83
258
  selector: "rs-stepper",
84
- template: "<div class=\"rs-stepper\">\r\n <div class=\"steps-wrap\">\r\n <ng-container *ngFor=\"let item of steps; let i = index\">\r\n <div\r\n class=\"step\"\r\n [ngClass]=\"{\r\n done: item.step <= unlockedStep,\r\n active: item.step === currentStep\r\n }\"\r\n (click)=\"onStepClick(item)\"\r\n >\r\n <div\r\n class=\"step-label\"\r\n [matTooltip]=\"item.label !== item.displayTitle ? item.label : ''\"\r\n matTooltipPosition=\"above\"\r\n >\r\n {{ item.displayTitle }}\r\n </div>\r\n </div>\r\n <div class=\"step-arrow\" *ngIf=\"i < steps.length - 1\">\r\n <img src=\"/assets/img/step-arrow.svg\" />\r\n </div>\r\n </ng-container>\r\n </div>\r\n</div>\r\n",
85
- styles: [".rs-stepper{font-family:Arial;width:100%;border-bottom:1px solid var(--rs-border-color)}.rs-stepper .steps-wrap{display:flex;align-items:center;justify-content:space-between;width:-webkit-fit-content;width:-moz-fit-content;width:fit-content;margin:0 auto;gap:8px}.rs-stepper .steps-wrap .step{flex:1}.rs-stepper .steps-wrap .step .step-label{color:var(--rs-labels-color);font-size:12px;font-weight:400;line-height:22px;text-align:center;white-space:nowrap;padding:0 12px 12px;display:flex;align-items:center}.rs-stepper .steps-wrap .step.done{cursor:pointer}.rs-stepper .steps-wrap .step.done .step-label{color:var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step.active .step-label{color:var(--rs-active-labels-color);font-weight:700;border-bottom:1px solid var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step-arrow{padding:0 12px 12px;display:flex;align-items:center}"]
259
+ template: "<div class=\"rs-stepper\">\r\n <div class=\"steps-wrap\">\r\n <div\r\n class=\"row-btn\"\r\n [class.hidden]=\"isAtStart || !showBtn\"\r\n (click)=\"scrollLeft()\"\r\n >\r\n <div class=\"hover\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M10.4995 3.99955L7.24219 8.24219L10.4995 12.4848\"\r\n stroke=\"#1F7BFF\"\r\n />\r\n </svg>\r\n </div>\r\n <div class=\"normal\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M10.4995 3.99955L7.24219 8.24219L10.4995 12.4848\"\r\n stroke=\"#6B6B6B\"\r\n />\r\n </svg>\r\n </div>\r\n </div>\r\n <div\r\n class=\"steps-content\"\r\n [ngClass]=\"{\r\n leftMask: showBtn && !isAtStart,\r\n rightMask: showBtn && !isAtEnd,\r\n bothMask: showBtn && !isAtStart && showBtn && !isAtEnd\r\n }\"\r\n #menu\r\n (scroll)=\"onScroll()\"\r\n >\r\n <ng-container *ngFor=\"let item of steps; let i = index\">\r\n <div\r\n class=\"step\"\r\n [ngClass]=\"{\r\n done: item.step <= unlockedStep,\r\n active: item.step === currentStep\r\n }\"\r\n (click)=\"onStepClick(item, i)\"\r\n >\r\n <div\r\n class=\"step-label\"\r\n [matTooltip]=\"item.label !== item.displayTitle ? item.label : ''\"\r\n matTooltipPosition=\"above\"\r\n >\r\n {{ item.displayTitle }}\r\n </div>\r\n </div>\r\n <div class=\"step-arrow\" *ngIf=\"i < steps.length - 1\">\r\n <img src=\"/assets/img/step-arrow.svg\" />\r\n </div>\r\n </ng-container>\r\n </div>\r\n <div\r\n class=\"row-btn\"\r\n [class.hidden]=\"isAtEnd || !showBtn\"\r\n (click)=\"scrollRight()\"\r\n >\r\n <div class=\"hover\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M7.50045 3.99955L10.7578 8.24219L7.50045 12.4848\"\r\n stroke=\"#1F7BFF\"\r\n />\r\n </svg>\r\n </div>\r\n <div class=\"normal\">\r\n <svg\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"17\"\r\n height=\"16\"\r\n viewBox=\"0 0 17 16\"\r\n fill=\"none\"\r\n >\r\n <path\r\n d=\"M7.50045 3.99955L10.7578 8.24219L7.50045 12.4848\"\r\n stroke=\"#6B6B6B\"\r\n />\r\n </svg>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n",
260
+ styles: [".rs-stepper{font-family:Arial;width:100%;border-bottom:1px solid var(--rs-border-color)}.rs-stepper .steps-wrap{display:flex;align-items:center;justify-content:center;gap:12px;margin:0 auto}.rs-stepper .steps-wrap .steps-content{display:flex;align-items:center;flex-wrap:nowrap;min-width:0;overflow-x:auto;overflow-y:hidden;gap:8px}.rs-stepper .steps-wrap .steps-content.leftMask{-webkit-mask-image:linear-gradient(90deg,transparent,#fff 10%,#d8d3d3 100%);mask-image:linear-gradient(90deg,transparent,#fff 10%,#d8d3d3 100%)}.rs-stepper .steps-wrap .steps-content.rightMask{-webkit-mask-image:linear-gradient(90deg,#fff 0,#fff 90%,transparent);mask-image:linear-gradient(90deg,#fff 0,#fff 90%,transparent)}.rs-stepper .steps-wrap .steps-content.bothMask{-webkit-mask-image:linear-gradient(90deg,transparent,#fff 10%,#fff 90%,transparent)!important;mask-image:linear-gradient(90deg,transparent,#fff 10%,#fff 90%,transparent)!important}.rs-stepper .steps-wrap .steps-content::-webkit-scrollbar{width:0;height:0}.rs-stepper .steps-wrap .step{flex:0 0 auto}.rs-stepper .steps-wrap .step .step-label{color:var(--rs-labels-color);font-size:12px;font-weight:400;line-height:22px;text-align:center;white-space:nowrap;padding:0 12px 12px;display:flex;align-items:center}.rs-stepper .steps-wrap .step.done{cursor:pointer}.rs-stepper .steps-wrap .step.done .step-label{color:var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step.active .step-label{color:var(--rs-active-labels-color);font-weight:700;border-bottom:1px solid var(--rs-active-labels-color)}.rs-stepper .steps-wrap .step-arrow{flex:0 0 auto;padding:0 12px 12px;display:flex;align-items:center}.rs-stepper .row-btn{margin-bottom:12px;display:flex;width:20px;height:20px;align-items:center;justify-content:center;border-radius:4px;border:1px solid #bdc4ca;flex-shrink:0;cursor:pointer;visibility:visible}.rs-stepper .row-btn .hover{display:none}.rs-stepper .row-btn .normal{display:flex}.rs-stepper .row-btn:hover{background-color:rgba(31,123,255,.04)}.rs-stepper .row-btn:hover .normal{display:none}.rs-stepper .row-btn:hover .hover{display:flex}.rs-stepper .row-btn.hidden{visibility:hidden}"]
86
261
  }] }
87
262
  ];
88
263
  /** @nocollapse */
89
264
  RSStepperComponent.ctorParameters = function () { return [
90
- { type: CommonFunctionService }
265
+ { type: CommonFunctionService },
266
+ { type: ChangeDetectorRef }
91
267
  ]; };
92
268
  RSStepperComponent.propDecorators = {
269
+ menu: [{ type: ViewChild, args: ["menu", { static: false },] }],
93
270
  steps: [{ type: Input }],
94
271
  currentStep: [{ type: Input }],
95
272
  stepClick: [{ type: Output }],
96
- unlockedStep: [{ type: Input }]
273
+ unlockedStep: [{ type: Input }],
274
+ containerRefId: [{ type: Input }],
275
+ onResize: [{ type: HostListener, args: ["window:resize", ["$event"],] }]
97
276
  };
98
277
  return RSStepperComponent;
99
278
  }());
100
279
  export { RSStepperComponent };
101
280
  if (false) {
281
+ /** @type {?} */
282
+ RSStepperComponent.prototype.menu;
102
283
  /** @type {?} */
103
284
  RSStepperComponent.prototype.steps;
104
285
  /** @type {?} */
@@ -108,6 +289,19 @@ if (false) {
108
289
  /** @type {?} */
109
290
  RSStepperComponent.prototype.unlockedStep;
110
291
  /** @type {?} */
292
+ RSStepperComponent.prototype.containerRefId;
293
+ /** @type {?} */
294
+ RSStepperComponent.prototype.showBtn;
295
+ /** @type {?} */
296
+ RSStepperComponent.prototype.isAtStart;
297
+ /** @type {?} */
298
+ RSStepperComponent.prototype.isAtEnd;
299
+ /** @type {?} */
111
300
  RSStepperComponent.prototype.cf;
301
+ /**
302
+ * @type {?}
303
+ * @private
304
+ */
305
+ RSStepperComponent.prototype.ref;
112
306
  }
113
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6Im5nOi8vcmFpc2UtY29tbW9uLWxpYi8iLCJzb3VyY2VzIjpbImxpYi9sYXlvdXQvcnMtc3RlcHBlci9pbmRleC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQ0EsT0FBTyxFQUNMLFNBQVMsRUFDVCxZQUFZLEVBQ1osS0FBSyxFQUVMLE1BQU0sR0FFUCxNQUFNLGVBQWUsQ0FBQztBQUV2QixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUU5RTtJQVFFLDRCQUFtQixFQUF5QjtRQUF6QixPQUFFLEdBQUYsRUFBRSxDQUF1QjtRQUZuQyxVQUFLLEdBQWUsRUFBRSxDQUFDO1FBQ3ZCLGdCQUFXLEdBQVcsQ0FBQyxDQUFDO1FBRXZCLGNBQVMsR0FBRyxJQUFJLFlBQVksRUFBWSxDQUFDO1FBQzFDLGlCQUFZLEdBQVcsQ0FBQyxDQUFDLENBQUMsVUFBVTtJQUZFLENBQUM7Ozs7O0lBSWhELHFDQUFROzs7OztJQUFSO1FBQUEsaUJBS0M7UUFKQyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU87Ozs7UUFBQyxVQUFDLElBQUk7WUFDdEIsSUFBSSxDQUFDLFlBQVksR0FBRyxLQUFJLENBQUMsRUFBRSxDQUFDLGlCQUFpQixDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsR0FBRyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBQ3JFLENBQUMsRUFBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLGdCQUFnQixFQUFFLENBQUM7SUFDMUIsQ0FBQzs7Ozs7SUFFRCx3Q0FBVzs7OztJQUFYLFVBQVksT0FBc0I7UUFDaEMsSUFBSSxPQUFPLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUMsV0FBVyxFQUFFO1lBQ2pFLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1NBQ3pCO0lBQ0gsQ0FBQztJQUVELHFCQUFxQjs7Ozs7O0lBQ2IsNkNBQWdCOzs7Ozs7SUFBeEI7UUFDRSxJQUFJLElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDLFlBQVksRUFBRTtZQUN4QyxJQUFJLENBQUMsWUFBWSxHQUFHLElBQUksQ0FBQyxXQUFXLENBQUM7U0FDdEM7SUFDSCxDQUFDOzs7OztJQUVELHdDQUFXOzs7O0lBQVgsVUFBWSxJQUFjO1FBQ3hCLElBQUksSUFBSSxDQUFDLElBQUksSUFBSSxJQUFJLENBQUMsWUFBWSxFQUFFO1lBQ2xDLElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQztZQUM3QixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQztTQUMzQjtJQUNILENBQUM7O2dCQXJDRixTQUFTLFNBQUM7b0JBQ1QsUUFBUSxFQUFFLFlBQVk7b0JBQ3RCLG95QkFBcUM7O2lCQUV0Qzs7OztnQkFOUSxxQkFBcUI7Ozt3QkFRM0IsS0FBSzs4QkFDTCxLQUFLOzRCQUVMLE1BQU07K0JBQ04sS0FBSzs7SUE0QlIseUJBQUM7Q0FBQSxBQXRDRCxJQXNDQztTQWpDWSxrQkFBa0I7OztJQUM3QixtQ0FBZ0M7O0lBQ2hDLHlDQUFpQzs7SUFFakMsdUNBQW1EOztJQUNuRCwwQ0FBa0M7O0lBRnRCLGdDQUFnQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIOe7hOS7tuexu1xyXG5pbXBvcnQge1xyXG4gIENvbXBvbmVudCxcclxuICBFdmVudEVtaXR0ZXIsXHJcbiAgSW5wdXQsXHJcbiAgT25Jbml0LFxyXG4gIE91dHB1dCxcclxuICBTaW1wbGVDaGFuZ2VzLFxyXG59IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XHJcbmltcG9ydCB7IFN0ZXBJdGVtIH0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XHJcbmltcG9ydCB7IENvbW1vbkZ1bmN0aW9uU2VydmljZSB9IGZyb20gXCIuLi8uLi9zZXJ2aWNlL2NvbW1vbi1mdW5jdGlvbi5zZXJ2aWNlXCI7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogXCJycy1zdGVwcGVyXCIsXHJcbiAgdGVtcGxhdGVVcmw6IFwiLi9pbmRleC5jb21wb25lbnQuaHRtbFwiLFxyXG4gIHN0eWxlVXJsczogW1wiLi9pbmRleC5jb21wb25lbnQuc2Nzc1wiXSxcclxufSlcclxuZXhwb3J0IGNsYXNzIFJTU3RlcHBlckNvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XHJcbiAgQElucHV0KCkgc3RlcHM6IFN0ZXBJdGVtW10gPSBbXTtcclxuICBASW5wdXQoKSBjdXJyZW50U3RlcDogbnVtYmVyID0gMDtcclxuICBjb25zdHJ1Y3RvcihwdWJsaWMgY2Y6IENvbW1vbkZ1bmN0aW9uU2VydmljZSkge31cclxuICBAT3V0cHV0KCkgc3RlcENsaWNrID0gbmV3IEV2ZW50RW1pdHRlcjxTdGVwSXRlbT4oKTtcclxuICBASW5wdXQoKSB1bmxvY2tlZFN0ZXA6IG51bWJlciA9IDA7IC8v5bey6Kej6ZSB55qE5pyA5aSn5q2l6aqkXHJcblxyXG4gIG5nT25Jbml0KCkge1xyXG4gICAgdGhpcy5zdGVwcy5mb3JFYWNoKChzdGVwKSA9PiB7XHJcbiAgICAgIHN0ZXAuZGlzcGxheVRpdGxlID0gdGhpcy5jZi5zZXRNaWRkbGVFbGxpcHNpcyhzdGVwLmxhYmVsLCAxMzIsIDEyKTtcclxuICAgIH0pO1xyXG4gICAgdGhpcy5zeW5jVW5sb2NrZWRTdGVwKCk7XHJcbiAgfVxyXG5cclxuICBuZ09uQ2hhbmdlcyhjaGFuZ2VzOiBTaW1wbGVDaGFuZ2VzKSB7XHJcbiAgICBpZiAoY2hhbmdlc1tcImN1cnJlbnRTdGVwXCJdICYmICFjaGFuZ2VzW1wiY3VycmVudFN0ZXBcIl0uZmlyc3RDaGFuZ2UpIHtcclxuICAgICAgdGhpcy5zeW5jVW5sb2NrZWRTdGVwKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvLyDlkIzmraV1bmxvY2tlZFN0ZXDnmoTmnIDlpKflgLxcclxuICBwcml2YXRlIHN5bmNVbmxvY2tlZFN0ZXAoKSB7XHJcbiAgICBpZiAodGhpcy5jdXJyZW50U3RlcCA+IHRoaXMudW5sb2NrZWRTdGVwKSB7XHJcbiAgICAgIHRoaXMudW5sb2NrZWRTdGVwID0gdGhpcy5jdXJyZW50U3RlcDtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIG9uU3RlcENsaWNrKHN0ZXA6IFN0ZXBJdGVtKSB7XHJcbiAgICBpZiAoc3RlcC5zdGVwIDw9IHRoaXMudW5sb2NrZWRTdGVwKSB7XHJcbiAgICAgIHRoaXMuY3VycmVudFN0ZXAgPSBzdGVwLnN0ZXA7XHJcbiAgICAgIHRoaXMuc3RlcENsaWNrLmVtaXQoc3RlcCk7XHJcbiAgICB9XHJcbiAgfVxyXG59XHJcbiJdfQ==
307
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.component.js","sourceRoot":"ng://raise-common-lib/","sources":["lib/layout/rs-stepper/index.component.ts"],"names":[],"mappings":"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","sourcesContent":["// 组件类\r\nimport {\r\n  Component,\r\n  EventEmitter,\r\n  Input,\r\n  OnInit,\r\n  Output,\r\n  SimpleChanges,\r\n  ViewChild,\r\n  ElementRef,\r\n  HostListener,\r\n  ChangeDetectorRef,\r\n  AfterViewInit,\r\n} from \"@angular/core\";\r\nimport { StepItem } from \"./constants\";\r\nimport { CommonFunctionService } from \"../../service/common-function.service\";\r\n\r\n@Component({\r\n  selector: \"rs-stepper\",\r\n  templateUrl: \"./index.component.html\",\r\n  styleUrls: [\"./index.component.scss\"],\r\n})\r\nexport class RSStepperComponent implements OnInit, AfterViewInit {\r\n  @ViewChild(\"menu\", { static: false }) menu!: ElementRef;\r\n  @Input() steps: StepItem[] = [];\r\n  @Input() currentStep: number = 0;\r\n  constructor(\r\n    public cf: CommonFunctionService,\r\n    private ref: ChangeDetectorRef\r\n  ) {}\r\n  @Output() stepClick = new EventEmitter<StepItem>();\r\n  @Input() unlockedStep: number = 0; //已解锁的最大步骤\r\n  @Input() containerRefId;\r\n\r\n  ngOnInit() {\r\n    this.steps.forEach((step) => {\r\n      step.displayTitle = this.cf.setMiddleEllipsis(step.label, 132, 12);\r\n    });\r\n    this.syncUnlockedStep();\r\n  }\r\n\r\n  ngAfterViewInit() {\r\n    setTimeout(() => this.checkBtnShow(true));\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges) {\r\n    if (changes[\"currentStep\"] && !changes[\"currentStep\"].firstChange) {\r\n      this.syncUnlockedStep();\r\n      // 当外部传入的 currentStep 变化时，滚动到对应步骤\r\n      setTimeout(() => this.scrollToStep());\r\n    }\r\n    if (\r\n      (changes[\"steps\"] && changes[\"steps\"].currentValue) ||\r\n      changes[\"currentStep\"]\r\n    ) {\r\n      setTimeout(() => this.checkBtnShow());\r\n    }\r\n  }\r\n\r\n  // 同步unlockedStep的最大值\r\n  private syncUnlockedStep() {\r\n    if (this.currentStep > this.unlockedStep) {\r\n      this.unlockedStep = this.currentStep;\r\n    }\r\n  }\r\n\r\n  onStepClick(step: StepItem, index?: number) {\r\n    if (step.step <= this.unlockedStep) {\r\n      this.currentStep = step.step;\r\n      this.stepClick.emit(step);\r\n      setTimeout(() => this.scrollToStep(index));\r\n    }\r\n  }\r\n\r\n  private scrollToStep(index?: number) {\r\n    if (!this.menu || !this.menu.nativeElement) return;\r\n\r\n    const menuEl = this.menu.nativeElement;\r\n    const items = menuEl.querySelectorAll(\".step\");\r\n    const targetIndex =\r\n      typeof index === \"number\"\r\n        ? index\r\n        : this.steps.findIndex((s) => s.step === this.currentStep);\r\n\r\n    if (targetIndex < 0 || targetIndex >= items.length) return;\r\n\r\n    const targetEl = items[targetIndex] as HTMLElement;\r\n\r\n    //获取相对位置\r\n    const containerRect = menuEl.getBoundingClientRect();\r\n    const elementRect = targetEl.getBoundingClientRect();\r\n\r\n    // 计算元素相对于容器的位置\r\n    const elementRelativeLeft =\r\n      elementRect.left - containerRect.left + menuEl.scrollLeft;\r\n    const elementWidth = elementRect.width;\r\n    const containerWidth = menuEl.clientWidth;\r\n\r\n    // 计算元素中心点相对于容器的位置\r\n    const elementCenter = elementRelativeLeft + elementWidth / 2;\r\n\r\n    // 目标滚动位置：让元素中心出现在可视区域中心\r\n    const scrollTarget = elementCenter - containerWidth / 2;\r\n\r\n    // 限制在有效滚动范围内\r\n    const maxScroll = Math.max(0, menuEl.scrollWidth - containerWidth);\r\n    const finalTarget = Math.max(0, Math.min(scrollTarget, maxScroll));\r\n\r\n    menuEl.scrollTo({ left: finalTarget, behavior: \"smooth\" });\r\n    this.onScroll();\r\n  }\r\n\r\n  @HostListener(\"window:resize\", [\"$event\"])\r\n  onResize(event: Event) {\r\n    this.checkBtnShow();\r\n  }\r\n\r\n  checkBtnShow(init?: boolean) {\r\n    if (!this.menu || !this.menu.nativeElement) return;\r\n\r\n    const menuEl = this.menu.nativeElement;\r\n    const { scrollWidth, clientWidth } = menuEl;\r\n\r\n    // 判断是否需要显示滚动按钮\r\n    let threshold = clientWidth;\r\n    if (this.containerRefId) {\r\n      const containerRef = document.getElementById(this.containerRefId);\r\n      if (containerRef) {\r\n        threshold = Math.max(containerRef.offsetWidth * 0.8, 400);\r\n      }\r\n    }\r\n\r\n    this.showBtn = scrollWidth > threshold;\r\n\r\n    // 初始化时滚动到当前步骤\r\n    if (init) {\r\n      setTimeout(() => this.scrollToStep(), 0);\r\n    }\r\n\r\n    this.onScroll();\r\n    this.ref.markForCheck();\r\n  }\r\n  scrollLeft() {\r\n    const { scrollLeft } = this.menu.nativeElement;\r\n    const dis = scrollLeft / 100 < 2 ? scrollLeft + 10 : 100;\r\n    this.menu.nativeElement.scrollBy({ left: -dis, behavior: \"smooth\" });\r\n  }\r\n  scrollRight() {\r\n    const { scrollLeft, scrollWidth, clientWidth } = this.menu.nativeElement;\r\n    const rest = scrollWidth - clientWidth - scrollLeft;\r\n    const dis = rest / 100 < 2 ? rest + 10 : 100;\r\n    this.menu.nativeElement.scrollBy({ left: dis, behavior: \"smooth\" });\r\n  }\r\n\r\n  showBtn = false;\r\n  isAtStart = true;\r\n  isAtEnd = false;\r\n  onScroll() {\r\n    const { scrollLeft, scrollWidth, clientWidth } = this.menu.nativeElement;\r\n    this.isAtStart = scrollLeft <= 1;\r\n    this.isAtEnd = scrollLeft + clientWidth >= scrollWidth - 1;\r\n    this.ref.markForCheck();\r\n  }\r\n}\r\n"]}