@yibozhang/pro-table 0.0.4 → 0.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/yibozhang-pro-table.umd.js +612 -194
- package/bundles/yibozhang-pro-table.umd.js.map +1 -1
- package/bundles/yibozhang-pro-table.umd.min.js +1 -1
- package/bundles/yibozhang-pro-table.umd.min.js.map +1 -1
- package/esm2015/lib/page-container/page-container.component.js +4 -2
- package/esm2015/lib/page-public/antd-form.js +168 -36
- package/esm2015/lib/page-public/array-form.js +230 -131
- package/esm2015/lib/pro-table.component.js +156 -25
- package/fesm2015/yibozhang-pro-table.js +554 -190
- package/fesm2015/yibozhang-pro-table.js.map +1 -1
- package/lib/page-container/page-container.component.d.ts +1 -0
- package/lib/page-container/page-container.component.d.ts.map +1 -1
- package/lib/page-public/antd-form.d.ts +15 -1
- package/lib/page-public/antd-form.d.ts.map +1 -1
- package/lib/page-public/array-form.d.ts +56 -144
- package/lib/page-public/array-form.d.ts.map +1 -1
- package/lib/pro-table.component.d.ts +5 -0
- package/lib/pro-table.component.d.ts.map +1 -1
- package/package.json +1 -1
- package/yibozhang-pro-table.metadata.json +1 -1
|
@@ -1,59 +1,75 @@
|
|
|
1
1
|
import { Injectable } from "@angular/core";
|
|
2
2
|
import { v4 as uuidv4 } from "uuid";
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
|
-
/**
|
|
5
|
-
* 数组型数据收集基础服务
|
|
6
|
-
*
|
|
7
|
-
* 功能:
|
|
8
|
-
* 1. 支持通过 ngModel 绑定对应行的数据源
|
|
9
|
-
* 2. 自动为数组数据添加 isEdit 和 uid 字段
|
|
10
|
-
* 3. 支持每个字段的自定义校验方法,返回 'success' | 'warning' | 'error'
|
|
11
|
-
* 4. 提供公共方法支持删除、新增、编辑等操作
|
|
12
|
-
*/
|
|
4
|
+
/** 数组型数据收集基础服务 */
|
|
13
5
|
export class ArrayFormService {
|
|
14
6
|
constructor() {
|
|
15
7
|
this.formStore = {};
|
|
16
8
|
}
|
|
17
|
-
/**
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
9
|
+
/** 深度克隆值 */
|
|
10
|
+
cloneValue(value) {
|
|
11
|
+
if (value === null || value === undefined) {
|
|
12
|
+
return value;
|
|
13
|
+
}
|
|
14
|
+
if (Array.isArray(value)) {
|
|
15
|
+
return value.map((item) => this.cloneValue(item));
|
|
16
|
+
}
|
|
17
|
+
if (typeof value === "object") {
|
|
18
|
+
const cloned = {};
|
|
19
|
+
for (const key of Object.keys(value)) {
|
|
20
|
+
cloned[key] = this.cloneValue(value[key]);
|
|
21
|
+
}
|
|
22
|
+
return cloned;
|
|
23
|
+
}
|
|
24
|
+
return value;
|
|
25
|
+
}
|
|
26
|
+
/** 比较两个值是否相等 */
|
|
27
|
+
isEqual(a, b) {
|
|
28
|
+
if (a === b)
|
|
29
|
+
return true;
|
|
30
|
+
if (a === null || b === null)
|
|
31
|
+
return a === b;
|
|
32
|
+
if (a === undefined || b === undefined)
|
|
33
|
+
return a === b;
|
|
34
|
+
if (typeof a !== typeof b)
|
|
35
|
+
return false;
|
|
36
|
+
if (Array.isArray(a) && Array.isArray(b)) {
|
|
37
|
+
if (a.length !== b.length)
|
|
38
|
+
return false;
|
|
39
|
+
return a.every((item, index) => this.isEqual(item, b[index]));
|
|
40
|
+
}
|
|
41
|
+
if (typeof a === "object" && typeof b === "object") {
|
|
42
|
+
const keysA = Object.keys(a);
|
|
43
|
+
const keysB = Object.keys(b);
|
|
44
|
+
if (keysA.length !== keysB.length)
|
|
45
|
+
return false;
|
|
46
|
+
return keysA.every((key) => this.isEqual(a[key], b[key]));
|
|
47
|
+
}
|
|
48
|
+
return false;
|
|
49
|
+
}
|
|
50
|
+
/** 创建行的初始值快照 */
|
|
51
|
+
createRowSnapshot(row, fields) {
|
|
52
|
+
const snapshot = {};
|
|
53
|
+
fields.forEach((field) => {
|
|
54
|
+
snapshot[field.name] = this.cloneValue(row[field.name]);
|
|
55
|
+
});
|
|
56
|
+
return snapshot;
|
|
57
|
+
}
|
|
58
|
+
/** 初始化数组数据,添加 uid 和 isEdit 字段 */
|
|
25
59
|
initializeData(data) {
|
|
26
60
|
return data.map((item) => this.enrichRow(item));
|
|
27
61
|
}
|
|
28
|
-
/**
|
|
29
|
-
* 为单行数据添加 uid 和 isEdit 字段
|
|
30
|
-
*
|
|
31
|
-
* @param row 原始行数据
|
|
32
|
-
* @returns 增强后的行数据
|
|
33
|
-
*/
|
|
62
|
+
/** 为单行数据添加 uid 和 isEdit 字段 */
|
|
34
63
|
enrichRow(row) {
|
|
35
|
-
return Object.assign(Object.assign({}, row), { uid: row.uid || uuidv4(), isEdit: row.isEdit !== undefined ? row.isEdit : false });
|
|
64
|
+
return Object.assign(Object.assign({}, row), { uid: row.uid || uuidv4(), isEdit: row.isEdit !== undefined ? row.isEdit : false, disabled: row.disabled !== undefined ? row.disabled : false });
|
|
36
65
|
}
|
|
37
|
-
/**
|
|
38
|
-
* 新增一行数据
|
|
39
|
-
*
|
|
40
|
-
* @param data 数据数组引用
|
|
41
|
-
* @param defaultValues 默认值
|
|
42
|
-
* @returns 新增的行数据
|
|
43
|
-
*/
|
|
66
|
+
/** 新增一行数据 */
|
|
44
67
|
addRow(data, defaultValues) {
|
|
45
|
-
const newRow = Object.assign(Object.assign({}, defaultValues), { uid: uuidv4(), isEdit: true, isAdd: true });
|
|
68
|
+
const newRow = Object.assign(Object.assign({}, defaultValues), { uid: uuidv4(), isEdit: true, isAdd: true, disabled: false });
|
|
46
69
|
data.push(newRow);
|
|
47
70
|
return newRow;
|
|
48
71
|
}
|
|
49
|
-
/**
|
|
50
|
-
* 校验单个字段
|
|
51
|
-
*
|
|
52
|
-
* @param row 行数据
|
|
53
|
-
* @param fieldName 字段名
|
|
54
|
-
* @param config 表单配置
|
|
55
|
-
* @returns 校验结果
|
|
56
|
-
*/
|
|
72
|
+
/** 校验单个字段 */
|
|
57
73
|
validateField(row, fieldName, config) {
|
|
58
74
|
const fieldConfig = config.fields.find((f) => f.name === fieldName);
|
|
59
75
|
if (!fieldConfig) {
|
|
@@ -64,13 +80,7 @@ export class ArrayFormService {
|
|
|
64
80
|
}
|
|
65
81
|
return "";
|
|
66
82
|
}
|
|
67
|
-
/**
|
|
68
|
-
* 校验整行数据
|
|
69
|
-
*
|
|
70
|
-
* @param row 行数据
|
|
71
|
-
* @param config 表单配置
|
|
72
|
-
* @returns 校验结果对象,key为字段名,value为校验结果
|
|
73
|
-
*/
|
|
83
|
+
/** 校验整行数据 */
|
|
74
84
|
validateRow(row, config) {
|
|
75
85
|
const results = {};
|
|
76
86
|
config.fields.forEach((fieldConfig) => {
|
|
@@ -78,13 +88,7 @@ export class ArrayFormService {
|
|
|
78
88
|
});
|
|
79
89
|
return results;
|
|
80
90
|
}
|
|
81
|
-
/**
|
|
82
|
-
* 校验整个数组
|
|
83
|
-
*
|
|
84
|
-
* @param data 数据数组
|
|
85
|
-
* @param config 表单配置
|
|
86
|
-
* @returns 校验结果对象,key为行的uid,value为该行的校验结果
|
|
87
|
-
*/
|
|
91
|
+
/** 校验整个数组 */
|
|
88
92
|
validateData(data, config) {
|
|
89
93
|
const results = {};
|
|
90
94
|
data.forEach((row) => {
|
|
@@ -92,90 +96,126 @@ export class ArrayFormService {
|
|
|
92
96
|
});
|
|
93
97
|
return results;
|
|
94
98
|
}
|
|
95
|
-
/**
|
|
96
|
-
* 注册表单到存储
|
|
97
|
-
* 如果表单已存在,则更新其配置
|
|
98
|
-
*
|
|
99
|
-
* @param formName 表单名称
|
|
100
|
-
* @param data 数据数组
|
|
101
|
-
* @param config 表单配置
|
|
102
|
-
*/
|
|
99
|
+
/** 注册表单到存储 */
|
|
103
100
|
registerForm(formName, data, config) {
|
|
101
|
+
const initializedData = this.initializeData(data);
|
|
102
|
+
// 记录初始值快照
|
|
103
|
+
const initialSnapshot = {};
|
|
104
|
+
initializedData.forEach((row) => {
|
|
105
|
+
initialSnapshot[row.uid] = this.createRowSnapshot(row, config.fields);
|
|
106
|
+
});
|
|
104
107
|
this.formStore[formName] = {
|
|
105
|
-
data:
|
|
108
|
+
data: initializedData,
|
|
106
109
|
config,
|
|
107
110
|
validationResults: {},
|
|
111
|
+
initialSnapshot,
|
|
112
|
+
editingSnapshot: {},
|
|
113
|
+
allTouched: false,
|
|
114
|
+
rowTouched: {},
|
|
108
115
|
};
|
|
109
116
|
// 如果配置了自动更新,立即更新一次
|
|
110
117
|
this.autoUpdateArrayReference(formName);
|
|
111
118
|
}
|
|
112
|
-
/**
|
|
113
|
-
* 从存储中获取表单
|
|
114
|
-
*
|
|
115
|
-
* @param formName 表单名称
|
|
116
|
-
* @returns 表单存储项或undefined
|
|
117
|
-
*/
|
|
119
|
+
/** 从存储中获取表单 */
|
|
118
120
|
getForm(formName) {
|
|
119
121
|
return this.formStore[formName];
|
|
120
122
|
}
|
|
121
|
-
/**
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
123
|
+
/** 从存储中获取表单数据 */
|
|
124
|
+
getFormData(formName) {
|
|
125
|
+
const form = this.formStore[formName];
|
|
126
|
+
if (!form) {
|
|
127
|
+
return undefined;
|
|
128
|
+
}
|
|
129
|
+
return form.data.map((item) => {
|
|
130
|
+
delete item.isEdit;
|
|
131
|
+
delete item.isAdd;
|
|
132
|
+
delete item.uid;
|
|
133
|
+
return item;
|
|
134
|
+
});
|
|
135
|
+
}
|
|
136
|
+
/** 向表单添加行 */
|
|
128
137
|
addRowToForm(formName, defaultValues) {
|
|
129
138
|
const form = this.formStore[formName];
|
|
130
139
|
if (!form) {
|
|
131
140
|
return undefined;
|
|
132
141
|
}
|
|
133
142
|
const newRow = this.addRow(form.data, defaultValues);
|
|
143
|
+
// 为新行记录初始值快照
|
|
144
|
+
form.initialSnapshot[newRow.uid] = this.createRowSnapshot(newRow, form.config.fields);
|
|
134
145
|
// 自动更新外部数组引用
|
|
135
146
|
this.autoUpdateArrayReference(formName);
|
|
136
147
|
return newRow;
|
|
137
148
|
}
|
|
138
|
-
/**
|
|
139
|
-
* 校验表单
|
|
140
|
-
*
|
|
141
|
-
* @param formName 表单名称
|
|
142
|
-
* @returns 校验结果或undefined
|
|
143
|
-
*/
|
|
149
|
+
/** 校验表单,自动标记所有字段为已触碰 */
|
|
144
150
|
validateForm(formName) {
|
|
145
151
|
const form = this.formStore[formName];
|
|
146
152
|
if (!form) {
|
|
147
|
-
return
|
|
153
|
+
return true;
|
|
148
154
|
}
|
|
155
|
+
// 标记所有字段为已触碰
|
|
156
|
+
form.allTouched = true;
|
|
149
157
|
const results = this.validateData(form.data, form.config);
|
|
150
158
|
form.validationResults = results;
|
|
151
|
-
|
|
159
|
+
// 检查是否有任何错误
|
|
160
|
+
for (const uid of Object.keys(results)) {
|
|
161
|
+
for (const fieldName of Object.keys(results[uid])) {
|
|
162
|
+
if (results[uid][fieldName]) {
|
|
163
|
+
return false;
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
return true;
|
|
152
168
|
}
|
|
153
|
-
/**
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
169
|
+
/** 开启编辑模式,保存快照用于取消恢复 */
|
|
170
|
+
enableEdit(formName, row) {
|
|
171
|
+
const form = this.formStore[formName];
|
|
172
|
+
if (form && row.uid) {
|
|
173
|
+
// 保存编辑前的快照
|
|
174
|
+
form.editingSnapshot[row.uid] = this.createRowSnapshot(row, form.config.fields);
|
|
175
|
+
}
|
|
159
176
|
row.isEdit = true;
|
|
160
177
|
}
|
|
161
|
-
/**
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
178
|
+
/** 取消编辑,恢复到编辑前的数据 */
|
|
179
|
+
disableEdit(formName, row) {
|
|
180
|
+
const form = this.formStore[formName];
|
|
181
|
+
if (form && row.uid && form.editingSnapshot[row.uid]) {
|
|
182
|
+
// 恢复编辑前的数据
|
|
183
|
+
const snapshot = form.editingSnapshot[row.uid];
|
|
184
|
+
form.config.fields.forEach((field) => {
|
|
185
|
+
row[field.name] = this.cloneValue(snapshot[field.name]);
|
|
186
|
+
});
|
|
187
|
+
// 更新初始快照为恢复后的值,消除 touched 状态
|
|
188
|
+
form.initialSnapshot[row.uid] = this.cloneValue(snapshot);
|
|
189
|
+
// 清理编辑快照和 touched 状态
|
|
190
|
+
delete form.editingSnapshot[row.uid];
|
|
191
|
+
delete form.rowTouched[row.uid];
|
|
192
|
+
// 清理该行的校验结果
|
|
193
|
+
if (form.validationResults) {
|
|
194
|
+
delete form.validationResults[row.uid];
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
row.isEdit = false;
|
|
198
|
+
}
|
|
199
|
+
/** 确认编辑,保留修改后的数据 */
|
|
200
|
+
confirmEdit(formName, row) {
|
|
201
|
+
const form = this.formStore[formName];
|
|
202
|
+
if (form && row.uid) {
|
|
203
|
+
// 清理编辑快照(保留当前数据)
|
|
204
|
+
if (form.editingSnapshot[row.uid]) {
|
|
205
|
+
delete form.editingSnapshot[row.uid];
|
|
206
|
+
}
|
|
207
|
+
// 更新初始快照为当前值(用于后续 touched 检测)
|
|
208
|
+
form.initialSnapshot[row.uid] = this.createRowSnapshot(row, form.config.fields);
|
|
209
|
+
// 清除 touched 状态,基于新的 initialSnapshot 重新判断
|
|
210
|
+
delete form.rowTouched[row.uid];
|
|
211
|
+
}
|
|
167
212
|
row.isEdit = false;
|
|
213
|
+
row.isAdd = false;
|
|
168
214
|
}
|
|
169
|
-
/**
|
|
170
|
-
* 从表单中删除指定行
|
|
171
|
-
*
|
|
172
|
-
* @param formName 表单名称
|
|
173
|
-
* @param row 行的uid
|
|
174
|
-
* @returns 是否删除成功
|
|
175
|
-
*/
|
|
215
|
+
/** 从表单中删除指定行 */
|
|
176
216
|
deleteRowFromForm(formName, row) {
|
|
177
217
|
const form = this.formStore[formName];
|
|
178
|
-
if (!form) {
|
|
218
|
+
if (!form || !row.uid) {
|
|
179
219
|
return false;
|
|
180
220
|
}
|
|
181
221
|
const index = form.data.findIndex((item) => item.uid === row.uid);
|
|
@@ -183,31 +223,44 @@ export class ArrayFormService {
|
|
|
183
223
|
return false;
|
|
184
224
|
}
|
|
185
225
|
form.data.splice(index, 1);
|
|
186
|
-
|
|
226
|
+
// 清理该行的所有快照、touched 状态和校验结果
|
|
227
|
+
delete form.initialSnapshot[row.uid];
|
|
228
|
+
delete form.editingSnapshot[row.uid];
|
|
229
|
+
delete form.rowTouched[row.uid];
|
|
230
|
+
if (form.validationResults) {
|
|
187
231
|
delete form.validationResults[row.uid];
|
|
188
232
|
}
|
|
189
233
|
// 自动更新外部数组引用
|
|
190
234
|
this.autoUpdateArrayReference(formName);
|
|
191
235
|
return true;
|
|
192
236
|
}
|
|
193
|
-
/**
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
237
|
+
/** 检测字段是否已被触碰 */
|
|
238
|
+
isFieldTouched(form, row, fieldName) {
|
|
239
|
+
var _a;
|
|
240
|
+
// 整个表单已 touched(整个表单提交时)
|
|
241
|
+
if (form.allTouched) {
|
|
242
|
+
return true;
|
|
243
|
+
}
|
|
244
|
+
// 当前行已 touched(单行保存时)
|
|
245
|
+
if (form.rowTouched[row.uid]) {
|
|
246
|
+
return true;
|
|
247
|
+
}
|
|
248
|
+
// 检测当前值与初始值是否不同
|
|
249
|
+
const initialValue = (_a = form.initialSnapshot[row.uid]) === null || _a === void 0 ? void 0 : _a[fieldName];
|
|
250
|
+
const currentValue = row[fieldName];
|
|
251
|
+
return !this.isEqual(initialValue, currentValue);
|
|
252
|
+
}
|
|
253
|
+
/** 校验指定行的指定字段,未修改时不显示校验状态 */
|
|
202
254
|
validateFieldInRow(formName, row, fieldName) {
|
|
203
255
|
const form = this.formStore[formName];
|
|
204
256
|
if (!form) {
|
|
205
|
-
return "
|
|
257
|
+
return "";
|
|
206
258
|
}
|
|
207
259
|
const fieldConfig = form.config.fields.find((f) => f.name === fieldName);
|
|
208
260
|
if (!fieldConfig) {
|
|
209
|
-
return "
|
|
261
|
+
return "";
|
|
210
262
|
}
|
|
263
|
+
// 执行校验
|
|
211
264
|
let result = "";
|
|
212
265
|
if (fieldConfig.validator) {
|
|
213
266
|
result = fieldConfig.validator(row[fieldName], row, fieldName);
|
|
@@ -220,20 +273,35 @@ export class ArrayFormService {
|
|
|
220
273
|
form.validationResults[row.uid] = {};
|
|
221
274
|
}
|
|
222
275
|
form.validationResults[row.uid][fieldName] = result;
|
|
276
|
+
// 如果字段未被触碰,不显示校验状态
|
|
277
|
+
if (!this.isFieldTouched(form, row, fieldName)) {
|
|
278
|
+
return "";
|
|
279
|
+
}
|
|
223
280
|
return result ? "error" : "success";
|
|
224
281
|
}
|
|
225
|
-
/**
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
282
|
+
/** 获取指定行指定字段的校验错误信息 */
|
|
283
|
+
getFieldErrorMessage(formName, row, fieldName) {
|
|
284
|
+
var _a, _b;
|
|
285
|
+
const form = this.formStore[formName];
|
|
286
|
+
if (!form || !row.uid) {
|
|
287
|
+
return "";
|
|
288
|
+
}
|
|
289
|
+
// 如果字段未被触碰,不显示错误信息
|
|
290
|
+
if (!this.isFieldTouched(form, row, fieldName)) {
|
|
291
|
+
return "";
|
|
292
|
+
}
|
|
293
|
+
return ((_b = (_a = form.validationResults) === null || _a === void 0 ? void 0 : _a[row.uid]) === null || _b === void 0 ? void 0 : _b[fieldName]) || "";
|
|
294
|
+
}
|
|
295
|
+
/** 校验指定行的所有字段 */
|
|
232
296
|
validateRowAllFields(formName, row) {
|
|
233
297
|
const form = this.formStore[formName];
|
|
234
|
-
if (!form) {
|
|
298
|
+
if (!form || !row.uid) {
|
|
235
299
|
return true;
|
|
236
300
|
}
|
|
301
|
+
// 更新初始快照为当前值
|
|
302
|
+
form.initialSnapshot[row.uid] = this.createRowSnapshot(row, form.config.fields);
|
|
303
|
+
// 标记当前行为 touched(只影响当前行,不影响其他行)
|
|
304
|
+
form.rowTouched[row.uid] = true;
|
|
237
305
|
let hasError = false;
|
|
238
306
|
form.config.fields.forEach((fieldConfig) => {
|
|
239
307
|
const result = this.validateFieldInRow(formName, row, fieldConfig.name);
|
|
@@ -243,12 +311,7 @@ export class ArrayFormService {
|
|
|
243
311
|
});
|
|
244
312
|
return !hasError;
|
|
245
313
|
}
|
|
246
|
-
/**
|
|
247
|
-
* 自动更新外部数组引用(如果配置了)
|
|
248
|
-
* 私有方法,在数据变化时自动调用
|
|
249
|
-
*
|
|
250
|
-
* @param formName 表单名称
|
|
251
|
-
*/
|
|
314
|
+
/** 自动更新外部数组引用 */
|
|
252
315
|
autoUpdateArrayReference(formName) {
|
|
253
316
|
const form = this.formStore[formName];
|
|
254
317
|
if (!form || !form.config.targetObject || !form.config.arrayPropertyName) {
|
|
@@ -258,6 +321,42 @@ export class ArrayFormService {
|
|
|
258
321
|
// 更新目标对象的数组属性为新数组引用
|
|
259
322
|
form.config.targetObject[form.config.arrayPropertyName] = newArray;
|
|
260
323
|
}
|
|
324
|
+
/** 是否有正在编辑的行 */
|
|
325
|
+
hasEditingRow(formName) {
|
|
326
|
+
const form = this.formStore[formName];
|
|
327
|
+
if (!form) {
|
|
328
|
+
return false;
|
|
329
|
+
}
|
|
330
|
+
return form.data.some((item) => item.isEdit);
|
|
331
|
+
}
|
|
332
|
+
// 销毁对应的表单
|
|
333
|
+
destory(names) {
|
|
334
|
+
names.forEach((name) => {
|
|
335
|
+
if (this.formStore[name]) {
|
|
336
|
+
// 清理表单数据
|
|
337
|
+
delete this.formStore[name];
|
|
338
|
+
}
|
|
339
|
+
});
|
|
340
|
+
}
|
|
341
|
+
// 重置表单数组
|
|
342
|
+
resetFormArray(name) {
|
|
343
|
+
const form = this.formStore[name];
|
|
344
|
+
if (!form) {
|
|
345
|
+
return;
|
|
346
|
+
}
|
|
347
|
+
// 1. 清空数据数组
|
|
348
|
+
form.data = [];
|
|
349
|
+
// 2. 清理所有快照
|
|
350
|
+
form.initialSnapshot = {};
|
|
351
|
+
form.editingSnapshot = {};
|
|
352
|
+
// 3. 清理校验结果
|
|
353
|
+
form.validationResults = {};
|
|
354
|
+
// 4. 重置 touched 状态
|
|
355
|
+
form.allTouched = false;
|
|
356
|
+
form.rowTouched = {};
|
|
357
|
+
// 5. 如果配置了自动更新,更新外部数组引用
|
|
358
|
+
this.autoUpdateArrayReference(name);
|
|
359
|
+
}
|
|
261
360
|
}
|
|
262
361
|
ArrayFormService.ɵprov = i0.ɵɵdefineInjectable({ factory: function ArrayFormService_Factory() { return new ArrayFormService(); }, token: ArrayFormService, providedIn: "root" });
|
|
263
362
|
ArrayFormService.decorators = [
|
|
@@ -266,4 +365,4 @@ ArrayFormService.decorators = [
|
|
|
266
365
|
},] }
|
|
267
366
|
];
|
|
268
367
|
ArrayFormService.ctorParameters = () => [];
|
|
269
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"array-form.js","sourceRoot":"D:/projects/vps-front/Front/DasPMSWeb/lib/pro-table/src/","sources":["lib/page-public/array-form.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from \"@angular/core\";\r\nimport { v4 as uuidv4 } from \"uuid\";\r\n\r\n/**\r\n * 字段校验结果类型\r\n */\r\nexport type ValidationResult = string;\r\n\r\n/**\r\n * 字段校验方法类型\r\n */\r\nexport type FieldValidator = (\r\n  value: any,\r\n  row: ArrayFormRow,\r\n  fieldName: string\r\n) => ValidationResult;\r\n\r\n/**\r\n * 字段配置接口\r\n */\r\nexport interface FieldConfig {\r\n  name: string;\r\n  validator?: FieldValidator;\r\n}\r\n\r\n/**\r\n * 数组表单行数据接口\r\n */\r\nexport interface ArrayFormRow {\r\n  uid: string;\r\n  isEdit: boolean;\r\n  isAdd?: boolean;\r\n  [key: string]: any;\r\n}\r\n\r\n/**\r\n * 数组表单配置接口\r\n */\r\nexport interface ArrayFormConfig {\r\n  fields: FieldConfig[];\r\n  defaultValues?: Record<string, any>;\r\n  // 自动更新目标对象和属性名，用于触发 Angular 变化检测\r\n  targetObject?: Record<string, any>;\r\n  arrayPropertyName?: string;\r\n}\r\n\r\n/**\r\n * 表单存储项接口\r\n */\r\nexport interface ArrayFormStoreItem {\r\n  data: ArrayFormRow[];\r\n  config: ArrayFormConfig;\r\n  validationResults?: Record<string, Record<string, ValidationResult>>;\r\n}\r\n\r\n/**\r\n * 表单存储接口\r\n */\r\nexport interface ArrayFormStore {\r\n  [formName: string]: ArrayFormStoreItem;\r\n}\r\n\r\n/**\r\n * 数组型数据收集基础服务\r\n *\r\n * 功能：\r\n * 1. 支持通过 ngModel 绑定对应行的数据源\r\n * 2. 自动为数组数据添加 isEdit 和 uid 字段\r\n * 3. 支持每个字段的自定义校验方法，返回 'success' | 'warning' | 'error'\r\n * 4. 提供公共方法支持删除、新增、编辑等操作\r\n */\r\n@Injectable({\r\n  providedIn: \"root\",\r\n})\r\nexport class ArrayFormService {\r\n  private formStore: ArrayFormStore = {};\r\n\r\n  constructor() {}\r\n\r\n  /**\r\n   * 初始化数组数据\r\n   * 为现有数据添加 uid 和 isEdit 字段\r\n   *\r\n   * @param data 原始数据数组\r\n   * @param config 表单配置\r\n   * @returns 初始化后的数据数组\r\n   */\r\n  private initializeData(data: any[]): ArrayFormRow[] {\r\n    return data.map((item) => this.enrichRow(item));\r\n  }\r\n\r\n  /**\r\n   * 为单行数据添加 uid 和 isEdit 字段\r\n   *\r\n   * @param row 原始行数据\r\n   * @returns 增强后的行数据\r\n   */\r\n  private enrichRow(row: any): ArrayFormRow {\r\n    return {\r\n      ...row,\r\n      uid: row.uid || uuidv4(),\r\n      isEdit: row.isEdit !== undefined ? row.isEdit : false,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * 新增一行数据\r\n   *\r\n   * @param data 数据数组引用\r\n   * @param defaultValues 默认值\r\n   * @returns 新增的行数据\r\n   */\r\n  private addRow(\r\n    data: ArrayFormRow[],\r\n    defaultValues?: Record<string, any>\r\n  ): ArrayFormRow {\r\n    const newRow: ArrayFormRow = {\r\n      ...defaultValues,\r\n      uid: uuidv4(),\r\n      isEdit: true,\r\n      isAdd: true,\r\n    };\r\n    data.push(newRow);\r\n    return newRow;\r\n  }\r\n\r\n  /**\r\n   * 校验单个字段\r\n   *\r\n   * @param row 行数据\r\n   * @param fieldName 字段名\r\n   * @param config 表单配置\r\n   * @returns 校验结果\r\n   */\r\n  private validateField(\r\n    row: ArrayFormRow,\r\n    fieldName: string,\r\n    config: ArrayFormConfig\r\n  ): ValidationResult {\r\n    const fieldConfig = config.fields.find((f) => f.name === fieldName);\r\n\r\n    if (!fieldConfig) {\r\n      return \"\";\r\n    }\r\n\r\n    if (fieldConfig.validator) {\r\n      return fieldConfig.validator(row[fieldName], row, fieldName);\r\n    }\r\n\r\n    return \"\";\r\n  }\r\n\r\n  /**\r\n   * 校验整行数据\r\n   *\r\n   * @param row 行数据\r\n   * @param config 表单配置\r\n   * @returns 校验结果对象，key为字段名，value为校验结果\r\n   */\r\n  private validateRow(\r\n    row: ArrayFormRow,\r\n    config: ArrayFormConfig\r\n  ): Record<string, ValidationResult> {\r\n    const results: Record<string, ValidationResult> = {};\r\n\r\n    config.fields.forEach((fieldConfig) => {\r\n      results[fieldConfig.name] = this.validateField(\r\n        row,\r\n        fieldConfig.name,\r\n        config\r\n      );\r\n    });\r\n\r\n    return results;\r\n  }\r\n\r\n  /**\r\n   * 校验整个数组\r\n   *\r\n   * @param data 数据数组\r\n   * @param config 表单配置\r\n   * @returns 校验结果对象，key为行的uid，value为该行的校验结果\r\n   */\r\n  private validateData(\r\n    data: ArrayFormRow[],\r\n    config: ArrayFormConfig\r\n  ): Record<string, Record<string, ValidationResult>> {\r\n    const results: Record<string, Record<string, ValidationResult>> = {};\r\n\r\n    data.forEach((row) => {\r\n      results[row.uid] = this.validateRow(row, config);\r\n    });\r\n\r\n    return results;\r\n  }\r\n\r\n  /**\r\n   * 注册表单到存储\r\n   * 如果表单已存在，则更新其配置\r\n   *\r\n   * @param formName 表单名称\r\n   * @param data 数据数组\r\n   * @param config 表单配置\r\n   */\r\n  registerForm(\r\n    formName: string,\r\n    data: ArrayFormRow[],\r\n    config: ArrayFormConfig\r\n  ): void {\r\n    this.formStore[formName] = {\r\n      data: this.initializeData(data),\r\n      config,\r\n      validationResults: {},\r\n    };\r\n    // 如果配置了自动更新，立即更新一次\r\n    this.autoUpdateArrayReference(formName);\r\n  }\r\n\r\n  /**\r\n   * 从存储中获取表单\r\n   *\r\n   * @param formName 表单名称\r\n   * @returns 表单存储项或undefined\r\n   */\r\n  getForm(formName: string): ArrayFormStoreItem | undefined {\r\n    return this.formStore[formName];\r\n  }\r\n\r\n  /**\r\n   * 向表单添加行\r\n   *\r\n   * @param formName 表单名称\r\n   * @param defaultValues 默认值\r\n   * @returns 新增的行数据或undefined\r\n   */\r\n  addRowToForm(\r\n    formName: string,\r\n    defaultValues?: Record<string, any>\r\n  ): ArrayFormRow | undefined {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return undefined;\r\n    }\r\n    const newRow = this.addRow(form.data, defaultValues);\r\n    // 自动更新外部数组引用\r\n    this.autoUpdateArrayReference(formName);\r\n    return newRow;\r\n  }\r\n\r\n  /**\r\n   * 校验表单\r\n   *\r\n   * @param formName 表单名称\r\n   * @returns 校验结果或undefined\r\n   */\r\n  validateForm(\r\n    formName: string\r\n  ): Record<string, Record<string, ValidationResult>> | undefined {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return undefined;\r\n    }\r\n    const results = this.validateData(form.data, form.config);\r\n    form.validationResults = results;\r\n    return results;\r\n  }\r\n\r\n  /**\r\n   * 开启指定行的编辑模式\r\n   *\r\n   * @param row 行数据\r\n   */\r\n  enableEdit(row: ArrayFormRow): void {\r\n    row.isEdit = true;\r\n  }\r\n\r\n  /**\r\n   * 关闭指定行的编辑模式\r\n   *\r\n   * @param row 行数据\r\n   */\r\n  disableEdit(row: ArrayFormRow): void {\r\n    row.isEdit = false;\r\n  }\r\n\r\n  /**\r\n   * 从表单中删除指定行\r\n   *\r\n   * @param formName 表单名称\r\n   * @param row 行的uid\r\n   * @returns 是否删除成功\r\n   */\r\n  deleteRowFromForm(formName: string, row: ArrayFormRow): boolean {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return false;\r\n    }\r\n    const index = form.data.findIndex((item) => item.uid === row.uid);\r\n    if (index === -1) {\r\n      return false;\r\n    }\r\n    form.data.splice(index, 1);\r\n    if (form.validationResults && form.validationResults[row.uid]) {\r\n      delete form.validationResults[row.uid];\r\n    }\r\n    // 自动更新外部数组引用\r\n    this.autoUpdateArrayReference(formName);\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * 校验指定行的指定字段并返回校验结果\r\n   * 实时执行校验，用于模板中的实时校验显示\r\n   *\r\n   * @param formName 表单名称\r\n   * @param row 行数据\r\n   * @param fieldName 字段名\r\n   * @returns 校验结果\r\n   */\r\n  validateFieldInRow(\r\n    formName: string,\r\n    row: ArrayFormRow,\r\n    fieldName: string\r\n  ): \"success\" | \"error\" {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return \"success\";\r\n    }\r\n\r\n    const fieldConfig = form.config.fields.find((f) => f.name === fieldName);\r\n    if (!fieldConfig) {\r\n      return \"success\";\r\n    }\r\n\r\n    let result: ValidationResult = \"\";\r\n    if (fieldConfig.validator) {\r\n      result = fieldConfig.validator(row[fieldName], row, fieldName);\r\n    }\r\n\r\n    // 更新校验结果存储\r\n    if (!form.validationResults) {\r\n      form.validationResults = {};\r\n    }\r\n    if (!form.validationResults[row.uid]) {\r\n      form.validationResults[row.uid] = {};\r\n    }\r\n    form.validationResults[row.uid][fieldName] = result;\r\n\r\n    return result ? \"error\" : \"success\";\r\n  }\r\n\r\n  /**\r\n   * 校验指定行的所有字段\r\n   *\r\n   * @param formName 表单名称\r\n   * @param row 行数据\r\n   * @returns 是否所有字段都通过校验\r\n   */\r\n  validateRowAllFields(formName: string, row: ArrayFormRow): boolean {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return true;\r\n    }\r\n\r\n    let hasError = false;\r\n\r\n    form.config.fields.forEach((fieldConfig) => {\r\n      const result = this.validateFieldInRow(formName, row, fieldConfig.name);\r\n      if (result === \"error\") {\r\n        hasError = true;\r\n      }\r\n    });\r\n\r\n    return !hasError;\r\n  }\r\n\r\n  /**\r\n   * 自动更新外部数组引用（如果配置了）\r\n   * 私有方法，在数据变化时自动调用\r\n   *\r\n   * @param formName 表单名称\r\n   */\r\n  private autoUpdateArrayReference(formName: string): void {\r\n    const form = this.formStore[formName];\r\n    if (!form || !form.config.targetObject || !form.config.arrayPropertyName) {\r\n      return;\r\n    }\r\n    const newArray = form.data ? [...form.data] : [];\r\n    // 更新目标对象的数组属性为新数组引用\r\n    (form.config.targetObject as any)[form.config.arrayPropertyName] = newArray;\r\n  }\r\n}\r\n"]}
|
|
368
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"array-form.js","sourceRoot":"D:/projects/vps-front/Front/DasPMSWeb/lib/pro-table/src/","sources":["lib/page-public/array-form.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,EAAE,EAAE,IAAI,MAAM,EAAE,MAAM,MAAM,CAAC;;AAwDpC,kBAAkB;AAIlB,MAAM,OAAO,gBAAgB;IAG3B;QAFQ,cAAS,GAAmB,EAAE,CAAC;IAExB,CAAC;IAEhB,YAAY;IACJ,UAAU,CAAC,KAAU;QAC3B,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,SAAS,EAAE;YACzC,OAAO,KAAK,CAAC;SACd;QACD,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACxB,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;SACnD;QACD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;YAC7B,MAAM,MAAM,GAAwB,EAAE,CAAC;YACvC,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE;gBACpC,MAAM,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC;aAC3C;YACD,OAAO,MAAM,CAAC;SACf;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,gBAAgB;IACR,OAAO,CAAC,CAAM,EAAE,CAAM;QAC5B,IAAI,CAAC,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC;QACzB,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,IAAI;YAAE,OAAO,CAAC,KAAK,CAAC,CAAC;QAC7C,IAAI,CAAC,KAAK,SAAS,IAAI,CAAC,KAAK,SAAS;YAAE,OAAO,CAAC,KAAK,CAAC,CAAC;QACvD,IAAI,OAAO,CAAC,KAAK,OAAO,CAAC;YAAE,OAAO,KAAK,CAAC;QAExC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;YACxC,IAAI,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC,MAAM;gBAAE,OAAO,KAAK,CAAC;YACxC,OAAO,CAAC,CAAC,KAAK,CAAC,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SAC/D;QAED,IAAI,OAAO,CAAC,KAAK,QAAQ,IAAI,OAAO,CAAC,KAAK,QAAQ,EAAE;YAClD,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC7B,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC7B,IAAI,KAAK,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM;gBAAE,OAAO,KAAK,CAAC;YAChD,OAAO,KAAK,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;SAC3D;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED,gBAAgB;IACR,iBAAiB,CACvB,GAAiB,EACjB,MAAqB;QAErB,MAAM,QAAQ,GAAwB,EAAE,CAAC;QACzC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YACvB,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;QAC1D,CAAC,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,iCAAiC;IACzB,cAAc,CAAC,IAAW;QAChC,OAAO,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;IAClD,CAAC;IAED,8BAA8B;IACtB,SAAS,CAAC,GAAQ;QACxB,uCACK,GAAG,KACN,GAAG,EAAE,GAAG,CAAC,GAAG,IAAI,MAAM,EAAE,EACxB,MAAM,EAAE,GAAG,CAAC,MAAM,KAAK,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,EACrD,QAAQ,EAAE,GAAG,CAAC,QAAQ,KAAK,SAAS,CAAC,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,IAC3D;IACJ,CAAC;IAED,aAAa;IACL,MAAM,CACZ,IAAoB,EACpB,aAAmC;QAEnC,MAAM,MAAM,mCACP,aAAa,KAChB,GAAG,EAAE,MAAM,EAAE,EACb,MAAM,EAAE,IAAI,EACZ,KAAK,EAAE,IAAI,EACX,QAAQ,EAAE,KAAK,GAChB,CAAC;QACF,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAClB,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,aAAa;IACL,aAAa,CACnB,GAAiB,EACjB,SAAiB,EACjB,MAAuB;QAEvB,MAAM,WAAW,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC;QAEpE,IAAI,CAAC,WAAW,EAAE;YAChB,OAAO,EAAE,CAAC;SACX;QAED,IAAI,WAAW,CAAC,SAAS,EAAE;YACzB,OAAO,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;SAC9D;QAED,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,aAAa;IACL,WAAW,CACjB,GAAiB,EACjB,MAAuB;QAEvB,MAAM,OAAO,GAAqC,EAAE,CAAC;QAErD,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,EAAE;YACpC,OAAO,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,aAAa,CAC5C,GAAG,EACH,WAAW,CAAC,IAAI,EAChB,MAAM,CACP,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,aAAa;IACL,YAAY,CAClB,IAAoB,EACpB,MAAuB;QAEvB,MAAM,OAAO,GAAqD,EAAE,CAAC;QAErE,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YACnB,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;QAEH,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,cAAc;IACd,YAAY,CACV,QAAgB,EAChB,IAAoB,EACpB,MAAuB;QAEvB,MAAM,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QAElD,UAAU;QACV,MAAM,eAAe,GAAwC,EAAE,CAAC;QAChE,eAAe,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YAC9B,eAAe,CAAC,GAAG,CAAC,GAAI,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QACzE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG;YACzB,IAAI,EAAE,eAAe;YACrB,MAAM;YACN,iBAAiB,EAAE,EAAE;YACrB,eAAe;YACf,eAAe,EAAE,EAAE;YACnB,UAAU,EAAE,KAAK;YACjB,UAAU,EAAE,EAAE;SACf,CAAC;QACF,mBAAmB;QACnB,IAAI,CAAC,wBAAwB,CAAC,QAAQ,CAAC,CAAC;IAC1C,CAAC;IAED,eAAe;IACf,OAAO,CAAC,QAAgB;QACtB,OAAO,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;IAClC,CAAC;IAED,iBAAiB;IACjB,WAAW,CACT,QAAgB;QAIhB,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,EAAE;YACT,OAAO,SAAS,CAAC;SAClB;QACD,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;YAC5B,OAAO,IAAI,CAAC,MAAM,CAAC;YACnB,OAAO,IAAI,CAAC,KAAK,CAAC;YAClB,OAAO,IAAI,CAAC,GAAG,CAAC;YAChB,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC;IACL,CAAC;IAED,aAAa;IACb,YAAY,CACV,QAAgB,EAChB,aAAmC;QAEnC,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,EAAE;YACT,OAAO,SAAS,CAAC;SAClB;QACD,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;QACrD,aAAa;QACb,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,GAAI,CAAC,GAAG,IAAI,CAAC,iBAAiB,CACxD,MAAM,EACN,IAAI,CAAC,MAAM,CAAC,MAAM,CACnB,CAAC;QACF,aAAa;QACb,IAAI,CAAC,wBAAwB,CAAC,QAAQ,CAAC,CAAC;QACxC,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,wBAAwB;IACxB,YAAY,CAAC,QAAgB;QAC3B,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,EAAE;YACT,OAAO,IAAI,CAAC;SACb;QACD,aAAa;QACb,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;QAC1D,IAAI,CAAC,iBAAiB,GAAG,OAAO,CAAC;QAEjC,YAAY;QACZ,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YACtC,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,EAAE;gBACjD,IAAI,OAAO,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,EAAE;oBAC3B,OAAO,KAAK,CAAC;iBACd;aACF;SACF;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,wBAAwB;IACxB,UAAU,CAAC,QAAgB,EAAE,GAAiB;QAC5C,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,IAAI,IAAI,GAAG,CAAC,GAAG,EAAE;YACnB,WAAW;YACX,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,iBAAiB,CACpD,GAAG,EACH,IAAI,CAAC,MAAM,CAAC,MAAM,CACnB,CAAC;SACH;QACD,GAAG,CAAC,MAAM,GAAG,IAAI,CAAC;IACpB,CAAC;IAED,qBAAqB;IACrB,WAAW,CAAC,QAAgB,EAAE,GAAiB;QAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,IAAI,IAAI,GAAG,CAAC,GAAG,IAAI,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;YACpD,WAAW;YACX,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YAC/C,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;gBACnC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;YAC1D,CAAC,CAAC,CAAC;YACH,6BAA6B;YAC7B,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;YAC1D,qBAAqB;YACrB,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YACrC,OAAO,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;YAChC,YAAY;YACZ,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBAC1B,OAAO,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;aACxC;SACF;QACD,GAAG,CAAC,MAAM,GAAG,KAAK,CAAC;IACrB,CAAC;IAED,oBAAoB;IACpB,WAAW,CAAC,QAAgB,EAAE,GAAiB;QAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,IAAI,IAAI,GAAG,CAAC,GAAG,EAAE;YACnB,iBAAiB;YACjB,IAAI,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;gBACjC,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;aACtC;YACD,8BAA8B;YAC9B,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,iBAAiB,CACpD,GAAG,EACH,IAAI,CAAC,MAAM,CAAC,MAAM,CACnB,CAAC;YACF,0CAA0C;YAC1C,OAAO,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;SACjC;QACD,GAAG,CAAC,MAAM,GAAG,KAAK,CAAC;QACnB,GAAG,CAAC,KAAK,GAAG,KAAK,CAAC;IACpB,CAAC;IAED,gBAAgB;IAChB,iBAAiB,CAAC,QAAgB,EAAE,GAAiB;QACnD,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE;YACrB,OAAO,KAAK,CAAC;SACd;QACD,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,KAAK,GAAG,CAAC,GAAG,CAAC,CAAC;QAClE,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE;YAChB,OAAO,KAAK,CAAC;SACd;QACD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QAC3B,4BAA4B;QAC5B,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACrC,OAAO,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACrC,OAAO,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAChC,IAAI,IAAI,CAAC,iBAAiB,EAAE;YAC1B,OAAO,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;SACxC;QACD,aAAa;QACb,IAAI,CAAC,wBAAwB,CAAC,QAAQ,CAAC,CAAC;QACxC,OAAO,IAAI,CAAC;IACd,CAAC;IAED,iBAAiB;IACT,cAAc,CACpB,IAAwB,EACxB,GAAiB,EACjB,SAAiB;;QAEjB,yBAAyB;QACzB,IAAI,IAAI,CAAC,UAAU,EAAE;YACnB,OAAO,IAAI,CAAC;SACb;QACD,sBAAsB;QACtB,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAI,CAAC,EAAE;YAC7B,OAAO,IAAI,CAAC;SACb;QACD,gBAAgB;QAChB,MAAM,YAAY,SAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAI,CAAC,0CAAG,SAAS,CAAC,CAAC;QACjE,MAAM,YAAY,GAAG,GAAG,CAAC,SAAS,CAAC,CAAC;QACpC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;IACnD,CAAC;IAED,6BAA6B;IAC7B,kBAAkB,CAChB,QAAgB,EAChB,GAAiB,EACjB,SAAiB;QAEjB,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,EAAE;YACT,OAAO,EAAE,CAAC;SACX;QAED,MAAM,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC;QACzE,IAAI,CAAC,WAAW,EAAE;YAChB,OAAO,EAAE,CAAC;SACX;QAED,OAAO;QACP,IAAI,MAAM,GAAqB,EAAE,CAAC;QAClC,IAAI,WAAW,CAAC,SAAS,EAAE;YACzB,MAAM,GAAG,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,SAAS,CAAC,CAAC;SAChE;QAED,WAAW;QACX,IAAI,CAAC,IAAI,CAAC,iBAAiB,EAAE;YAC3B,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;SAC7B;QACD,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAI,CAAC,EAAE;YACrC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAI,CAAC,GAAG,EAAE,CAAC;SACvC;QACD,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAI,CAAC,CAAC,SAAS,CAAC,GAAG,MAAM,CAAC;QAErD,mBAAmB;QACnB,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,EAAE,GAAG,EAAE,SAAS,CAAC,EAAE;YAC9C,OAAO,EAAE,CAAC;SACX;QAED,OAAO,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC;IACtC,CAAC;IAED,uBAAuB;IACvB,oBAAoB,CAClB,QAAgB,EAChB,GAAiB,EACjB,SAAiB;;QAEjB,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE;YACrB,OAAO,EAAE,CAAC;SACX;QACD,mBAAmB;QACnB,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,IAAI,EAAE,GAAG,EAAE,SAAS,CAAC,EAAE;YAC9C,OAAO,EAAE,CAAC;SACX;QACD,OAAO,aAAA,IAAI,CAAC,iBAAiB,0CAAG,GAAG,CAAC,GAAG,2CAAI,SAAS,MAAK,EAAE,CAAC;IAC9D,CAAC;IAED,iBAAiB;IACjB,oBAAoB,CAAC,QAAgB,EAAE,GAAiB;QACtD,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE;YACrB,OAAO,IAAI,CAAC;SACb;QAED,aAAa;QACb,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,iBAAiB,CACpD,GAAG,EACH,IAAI,CAAC,MAAM,CAAC,MAAM,CACnB,CAAC;QACF,gCAAgC;QAChC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC;QAEhC,IAAI,QAAQ,GAAG,KAAK,CAAC;QAErB,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,EAAE;YACzC,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,GAAG,EAAE,WAAW,CAAC,IAAI,CAAC,CAAC;YACxE,IAAI,MAAM,KAAK,OAAO,EAAE;gBACtB,QAAQ,GAAG,IAAI,CAAC;aACjB;QACH,CAAC,CAAC,CAAC;QAEH,OAAO,CAAC,QAAQ,CAAC;IACnB,CAAC;IAED,iBAAiB;IACT,wBAAwB,CAAC,QAAgB;QAC/C,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,EAAE;YACxE,OAAO;SACR;QACD,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACjD,oBAAoB;QACnB,IAAI,CAAC,MAAM,CAAC,YAAoB,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAG,QAAQ,CAAC;IAC9E,CAAC;IAED,gBAAgB;IAChB,aAAa,CAAC,QAAgB;QAC5B,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QACtC,IAAI,CAAC,IAAI,EAAE;YACT,OAAO,KAAK,CAAC;SACd;QACD,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC/C,CAAC;IAED,UAAU;IACV,OAAO,CAAC,KAAe;QACrB,KAAK,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,EAAE;YACrB,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE;gBACxB,SAAS;gBACT,OAAO,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;aAC7B;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,SAAS;IACT,cAAc,CAAC,IAAY;QACzB,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,CAAC,IAAI,EAAE;YACT,OAAO;SACR;QAED,YAAY;QACZ,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC;QAEf,YAAY;QACZ,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC;QAC1B,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC;QAE1B,YAAY;QACZ,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;QAE5B,mBAAmB;QACnB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QACxB,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;QAErB,wBAAwB;QACxB,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,CAAC;IACtC,CAAC;;;;YApdF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB","sourcesContent":["import { Injectable } from \"@angular/core\";\r\nimport { v4 as uuidv4 } from \"uuid\";\r\n\r\n/** 字段校验结果类型 */\r\nexport type ValidationResult = string;\r\n\r\n/** 字段校验方法类型 */\r\nexport type FieldValidator = (\r\n  value: any,\r\n  row: ArrayFormRow,\r\n  fieldName: string\r\n) => ValidationResult;\r\n\r\n/** 字段配置接口 */\r\nexport interface FieldConfig {\r\n  name: string;\r\n  validator?: FieldValidator;\r\n}\r\n\r\n/** 数组表单行数据接口 */\r\nexport interface ArrayFormRow {\r\n  uid?: string;\r\n  isEdit: boolean;\r\n  isAdd?: boolean;\r\n  disabled?: boolean;\r\n  [key: string]: any;\r\n}\r\n\r\n/** 数组表单配置接口 */\r\nexport interface ArrayFormConfig {\r\n  fields: FieldConfig[];\r\n  defaultValues?: Record<string, any>;\r\n  // 自动更新目标对象和属性名，用于触发 Angular 变化检测\r\n  targetObject?: Record<string, any>;\r\n  arrayPropertyName?: string;\r\n}\r\n\r\n/** 表单存储项接口 */\r\nexport interface ArrayFormStoreItem {\r\n  data: ArrayFormRow[];\r\n  config: ArrayFormConfig;\r\n  validationResults?: Record<string, Record<string, ValidationResult>>;\r\n  /** 初始值快照，用于自动检测字段是否被修改 */\r\n  initialSnapshot: Record<string, Record<string, any>>;\r\n  /** 编辑时快照，用于取消编辑时恢复数据 */\r\n  editingSnapshot: Record<string, Record<string, any>>;\r\n  /** 是否标记所有字段为已触碰（整个表单提交时设为 true） */\r\n  allTouched: boolean;\r\n  /** 每行的 touched 状态（单行保存时设为 true） */\r\n  rowTouched: Record<string, boolean>;\r\n}\r\n\r\n/** 表单存储接口 */\r\nexport interface ArrayFormStore {\r\n  [formName: string]: ArrayFormStoreItem;\r\n}\r\n\r\n/** 数组型数据收集基础服务 */\r\n@Injectable({\r\n  providedIn: \"root\",\r\n})\r\nexport class ArrayFormService {\r\n  private formStore: ArrayFormStore = {};\r\n\r\n  constructor() {}\r\n\r\n  /** 深度克隆值 */\r\n  private cloneValue(value: any): any {\r\n    if (value === null || value === undefined) {\r\n      return value;\r\n    }\r\n    if (Array.isArray(value)) {\r\n      return value.map((item) => this.cloneValue(item));\r\n    }\r\n    if (typeof value === \"object\") {\r\n      const cloned: Record<string, any> = {};\r\n      for (const key of Object.keys(value)) {\r\n        cloned[key] = this.cloneValue(value[key]);\r\n      }\r\n      return cloned;\r\n    }\r\n    return value;\r\n  }\r\n\r\n  /** 比较两个值是否相等 */\r\n  private isEqual(a: any, b: any): boolean {\r\n    if (a === b) return true;\r\n    if (a === null || b === null) return a === b;\r\n    if (a === undefined || b === undefined) return a === b;\r\n    if (typeof a !== typeof b) return false;\r\n\r\n    if (Array.isArray(a) && Array.isArray(b)) {\r\n      if (a.length !== b.length) return false;\r\n      return a.every((item, index) => this.isEqual(item, b[index]));\r\n    }\r\n\r\n    if (typeof a === \"object\" && typeof b === \"object\") {\r\n      const keysA = Object.keys(a);\r\n      const keysB = Object.keys(b);\r\n      if (keysA.length !== keysB.length) return false;\r\n      return keysA.every((key) => this.isEqual(a[key], b[key]));\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  /** 创建行的初始值快照 */\r\n  private createRowSnapshot(\r\n    row: ArrayFormRow,\r\n    fields: FieldConfig[]\r\n  ): Record<string, any> {\r\n    const snapshot: Record<string, any> = {};\r\n    fields.forEach((field) => {\r\n      snapshot[field.name] = this.cloneValue(row[field.name]);\r\n    });\r\n    return snapshot;\r\n  }\r\n\r\n  /** 初始化数组数据，添加 uid 和 isEdit 字段 */\r\n  private initializeData(data: any[]): ArrayFormRow[] {\r\n    return data.map((item) => this.enrichRow(item));\r\n  }\r\n\r\n  /** 为单行数据添加 uid 和 isEdit 字段 */\r\n  private enrichRow(row: any): ArrayFormRow {\r\n    return {\r\n      ...row,\r\n      uid: row.uid || uuidv4(),\r\n      isEdit: row.isEdit !== undefined ? row.isEdit : false,\r\n      disabled: row.disabled !== undefined ? row.disabled : false,\r\n    };\r\n  }\r\n\r\n  /** 新增一行数据 */\r\n  private addRow(\r\n    data: ArrayFormRow[],\r\n    defaultValues?: Record<string, any>\r\n  ): ArrayFormRow {\r\n    const newRow: ArrayFormRow = {\r\n      ...defaultValues,\r\n      uid: uuidv4(),\r\n      isEdit: true,\r\n      isAdd: true,\r\n      disabled: false,\r\n    };\r\n    data.push(newRow);\r\n    return newRow;\r\n  }\r\n\r\n  /** 校验单个字段 */\r\n  private validateField(\r\n    row: ArrayFormRow,\r\n    fieldName: string,\r\n    config: ArrayFormConfig\r\n  ): ValidationResult {\r\n    const fieldConfig = config.fields.find((f) => f.name === fieldName);\r\n\r\n    if (!fieldConfig) {\r\n      return \"\";\r\n    }\r\n\r\n    if (fieldConfig.validator) {\r\n      return fieldConfig.validator(row[fieldName], row, fieldName);\r\n    }\r\n\r\n    return \"\";\r\n  }\r\n\r\n  /** 校验整行数据 */\r\n  private validateRow(\r\n    row: ArrayFormRow,\r\n    config: ArrayFormConfig\r\n  ): Record<string, ValidationResult> {\r\n    const results: Record<string, ValidationResult> = {};\r\n\r\n    config.fields.forEach((fieldConfig) => {\r\n      results[fieldConfig.name] = this.validateField(\r\n        row,\r\n        fieldConfig.name,\r\n        config\r\n      );\r\n    });\r\n\r\n    return results;\r\n  }\r\n\r\n  /** 校验整个数组 */\r\n  private validateData(\r\n    data: ArrayFormRow[],\r\n    config: ArrayFormConfig\r\n  ): Record<string, Record<string, ValidationResult>> {\r\n    const results: Record<string, Record<string, ValidationResult>> = {};\r\n\r\n    data.forEach((row) => {\r\n      results[row.uid] = this.validateRow(row, config);\r\n    });\r\n\r\n    return results;\r\n  }\r\n\r\n  /** 注册表单到存储 */\r\n  registerForm(\r\n    formName: string,\r\n    data: ArrayFormRow[],\r\n    config: ArrayFormConfig\r\n  ): void {\r\n    const initializedData = this.initializeData(data);\r\n\r\n    // 记录初始值快照\r\n    const initialSnapshot: Record<string, Record<string, any>> = {};\r\n    initializedData.forEach((row) => {\r\n      initialSnapshot[row.uid!] = this.createRowSnapshot(row, config.fields);\r\n    });\r\n\r\n    this.formStore[formName] = {\r\n      data: initializedData,\r\n      config,\r\n      validationResults: {},\r\n      initialSnapshot,\r\n      editingSnapshot: {},\r\n      allTouched: false,\r\n      rowTouched: {},\r\n    };\r\n    // 如果配置了自动更新，立即更新一次\r\n    this.autoUpdateArrayReference(formName);\r\n  }\r\n\r\n  /** 从存储中获取表单 */\r\n  getForm(formName: string): ArrayFormStoreItem | undefined {\r\n    return this.formStore[formName];\r\n  }\r\n\r\n  /** 从存储中获取表单数据 */\r\n  getFormData(\r\n    formName: string\r\n  ):\r\n    | Array<Pick<ArrayFormRow, \"uid\" | \"isEdit\" | \"isAdd\" | keyof ArrayFormRow>>\r\n    | undefined {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return undefined;\r\n    }\r\n    return form.data.map((item) => {\r\n      delete item.isEdit;\r\n      delete item.isAdd;\r\n      delete item.uid;\r\n      return item;\r\n    });\r\n  }\r\n\r\n  /** 向表单添加行 */\r\n  addRowToForm(\r\n    formName: string,\r\n    defaultValues?: Record<string, any>\r\n  ): ArrayFormRow | undefined {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return undefined;\r\n    }\r\n    const newRow = this.addRow(form.data, defaultValues);\r\n    // 为新行记录初始值快照\r\n    form.initialSnapshot[newRow.uid!] = this.createRowSnapshot(\r\n      newRow,\r\n      form.config.fields\r\n    );\r\n    // 自动更新外部数组引用\r\n    this.autoUpdateArrayReference(formName);\r\n    return newRow;\r\n  }\r\n\r\n  /** 校验表单，自动标记所有字段为已触碰 */\r\n  validateForm(formName: string): boolean {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return true;\r\n    }\r\n    // 标记所有字段为已触碰\r\n    form.allTouched = true;\r\n    const results = this.validateData(form.data, form.config);\r\n    form.validationResults = results;\r\n\r\n    // 检查是否有任何错误\r\n    for (const uid of Object.keys(results)) {\r\n      for (const fieldName of Object.keys(results[uid])) {\r\n        if (results[uid][fieldName]) {\r\n          return false;\r\n        }\r\n      }\r\n    }\r\n    return true;\r\n  }\r\n\r\n  /** 开启编辑模式，保存快照用于取消恢复 */\r\n  enableEdit(formName: string, row: ArrayFormRow): void {\r\n    const form = this.formStore[formName];\r\n    if (form && row.uid) {\r\n      // 保存编辑前的快照\r\n      form.editingSnapshot[row.uid] = this.createRowSnapshot(\r\n        row,\r\n        form.config.fields\r\n      );\r\n    }\r\n    row.isEdit = true;\r\n  }\r\n\r\n  /** 取消编辑，恢复到编辑前的数据 */\r\n  disableEdit(formName: string, row: ArrayFormRow): void {\r\n    const form = this.formStore[formName];\r\n    if (form && row.uid && form.editingSnapshot[row.uid]) {\r\n      // 恢复编辑前的数据\r\n      const snapshot = form.editingSnapshot[row.uid];\r\n      form.config.fields.forEach((field) => {\r\n        row[field.name] = this.cloneValue(snapshot[field.name]);\r\n      });\r\n      // 更新初始快照为恢复后的值，消除 touched 状态\r\n      form.initialSnapshot[row.uid] = this.cloneValue(snapshot);\r\n      // 清理编辑快照和 touched 状态\r\n      delete form.editingSnapshot[row.uid];\r\n      delete form.rowTouched[row.uid];\r\n      // 清理该行的校验结果\r\n      if (form.validationResults) {\r\n        delete form.validationResults[row.uid];\r\n      }\r\n    }\r\n    row.isEdit = false;\r\n  }\r\n\r\n  /** 确认编辑，保留修改后的数据 */\r\n  confirmEdit(formName: string, row: ArrayFormRow): void {\r\n    const form = this.formStore[formName];\r\n    if (form && row.uid) {\r\n      // 清理编辑快照（保留当前数据）\r\n      if (form.editingSnapshot[row.uid]) {\r\n        delete form.editingSnapshot[row.uid];\r\n      }\r\n      // 更新初始快照为当前值（用于后续 touched 检测）\r\n      form.initialSnapshot[row.uid] = this.createRowSnapshot(\r\n        row,\r\n        form.config.fields\r\n      );\r\n      // 清除 touched 状态，基于新的 initialSnapshot 重新判断\r\n      delete form.rowTouched[row.uid];\r\n    }\r\n    row.isEdit = false;\r\n    row.isAdd = false;\r\n  }\r\n\r\n  /** 从表单中删除指定行 */\r\n  deleteRowFromForm(formName: string, row: ArrayFormRow): boolean {\r\n    const form = this.formStore[formName];\r\n    if (!form || !row.uid) {\r\n      return false;\r\n    }\r\n    const index = form.data.findIndex((item) => item.uid === row.uid);\r\n    if (index === -1) {\r\n      return false;\r\n    }\r\n    form.data.splice(index, 1);\r\n    // 清理该行的所有快照、touched 状态和校验结果\r\n    delete form.initialSnapshot[row.uid];\r\n    delete form.editingSnapshot[row.uid];\r\n    delete form.rowTouched[row.uid];\r\n    if (form.validationResults) {\r\n      delete form.validationResults[row.uid];\r\n    }\r\n    // 自动更新外部数组引用\r\n    this.autoUpdateArrayReference(formName);\r\n    return true;\r\n  }\r\n\r\n  /** 检测字段是否已被触碰 */\r\n  private isFieldTouched(\r\n    form: ArrayFormStoreItem,\r\n    row: ArrayFormRow,\r\n    fieldName: string\r\n  ): boolean {\r\n    // 整个表单已 touched（整个表单提交时）\r\n    if (form.allTouched) {\r\n      return true;\r\n    }\r\n    // 当前行已 touched（单行保存时）\r\n    if (form.rowTouched[row.uid!]) {\r\n      return true;\r\n    }\r\n    // 检测当前值与初始值是否不同\r\n    const initialValue = form.initialSnapshot[row.uid!]?.[fieldName];\r\n    const currentValue = row[fieldName];\r\n    return !this.isEqual(initialValue, currentValue);\r\n  }\r\n\r\n  /** 校验指定行的指定字段，未修改时不显示校验状态 */\r\n  validateFieldInRow(\r\n    formName: string,\r\n    row: ArrayFormRow,\r\n    fieldName: string\r\n  ): \"success\" | \"error\" | \"\" {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return \"\";\r\n    }\r\n\r\n    const fieldConfig = form.config.fields.find((f) => f.name === fieldName);\r\n    if (!fieldConfig) {\r\n      return \"\";\r\n    }\r\n\r\n    // 执行校验\r\n    let result: ValidationResult = \"\";\r\n    if (fieldConfig.validator) {\r\n      result = fieldConfig.validator(row[fieldName], row, fieldName);\r\n    }\r\n\r\n    // 更新校验结果存储\r\n    if (!form.validationResults) {\r\n      form.validationResults = {};\r\n    }\r\n    if (!form.validationResults[row.uid!]) {\r\n      form.validationResults[row.uid!] = {};\r\n    }\r\n    form.validationResults[row.uid!][fieldName] = result;\r\n\r\n    // 如果字段未被触碰，不显示校验状态\r\n    if (!this.isFieldTouched(form, row, fieldName)) {\r\n      return \"\";\r\n    }\r\n\r\n    return result ? \"error\" : \"success\";\r\n  }\r\n\r\n  /** 获取指定行指定字段的校验错误信息 */\r\n  getFieldErrorMessage(\r\n    formName: string,\r\n    row: ArrayFormRow,\r\n    fieldName: string\r\n  ): string {\r\n    const form = this.formStore[formName];\r\n    if (!form || !row.uid) {\r\n      return \"\";\r\n    }\r\n    // 如果字段未被触碰，不显示错误信息\r\n    if (!this.isFieldTouched(form, row, fieldName)) {\r\n      return \"\";\r\n    }\r\n    return form.validationResults?.[row.uid]?.[fieldName] || \"\";\r\n  }\r\n\r\n  /** 校验指定行的所有字段 */\r\n  validateRowAllFields(formName: string, row: ArrayFormRow): boolean {\r\n    const form = this.formStore[formName];\r\n    if (!form || !row.uid) {\r\n      return true;\r\n    }\r\n\r\n    // 更新初始快照为当前值\r\n    form.initialSnapshot[row.uid] = this.createRowSnapshot(\r\n      row,\r\n      form.config.fields\r\n    );\r\n    // 标记当前行为 touched（只影响当前行，不影响其他行）\r\n    form.rowTouched[row.uid] = true;\r\n\r\n    let hasError = false;\r\n\r\n    form.config.fields.forEach((fieldConfig) => {\r\n      const result = this.validateFieldInRow(formName, row, fieldConfig.name);\r\n      if (result === \"error\") {\r\n        hasError = true;\r\n      }\r\n    });\r\n\r\n    return !hasError;\r\n  }\r\n\r\n  /** 自动更新外部数组引用 */\r\n  private autoUpdateArrayReference(formName: string): void {\r\n    const form = this.formStore[formName];\r\n    if (!form || !form.config.targetObject || !form.config.arrayPropertyName) {\r\n      return;\r\n    }\r\n    const newArray = form.data ? [...form.data] : [];\r\n    // 更新目标对象的数组属性为新数组引用\r\n    (form.config.targetObject as any)[form.config.arrayPropertyName] = newArray;\r\n  }\r\n\r\n  /** 是否有正在编辑的行 */\r\n  hasEditingRow(formName: string): boolean {\r\n    const form = this.formStore[formName];\r\n    if (!form) {\r\n      return false;\r\n    }\r\n    return form.data.some((item) => item.isEdit);\r\n  }\r\n\r\n  // 销毁对应的表单\r\n  destory(names: string[]): void {\r\n    names.forEach((name) => {\r\n      if (this.formStore[name]) {\r\n        // 清理表单数据\r\n        delete this.formStore[name];\r\n      }\r\n    });\r\n  }\r\n\r\n  // 重置表单数组\r\n  resetFormArray(name: string): void {\r\n    const form = this.formStore[name];\r\n    if (!form) {\r\n      return;\r\n    }\r\n\r\n    // 1. 清空数据数组\r\n    form.data = [];\r\n\r\n    // 2. 清理所有快照\r\n    form.initialSnapshot = {};\r\n    form.editingSnapshot = {};\r\n\r\n    // 3. 清理校验结果\r\n    form.validationResults = {};\r\n\r\n    // 4. 重置 touched 状态\r\n    form.allTouched = false;\r\n    form.rowTouched = {};\r\n\r\n    // 5. 如果配置了自动更新，更新外部数组引用\r\n    this.autoUpdateArrayReference(name);\r\n  }\r\n}\r\n"]}
|