@libs-ui/utils 0.1.1-1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (65) hide show
  1. package/README.md +3 -0
  2. package/base64.d.ts +5 -0
  3. package/cache.d.ts +42 -0
  4. package/color.d.ts +11 -0
  5. package/communicate-micro.d.ts +16 -0
  6. package/constants.d.ts +10 -0
  7. package/crypto-3rd.d.ts +7 -0
  8. package/crypto.d.ts +8 -0
  9. package/dangerous-object.d.ts +79 -0
  10. package/date.d.ts +44 -0
  11. package/dom.d.ts +52 -0
  12. package/download.d.ts +3 -0
  13. package/esm2022/base64.mjs +43 -0
  14. package/esm2022/cache.mjs +388 -0
  15. package/esm2022/color.mjs +133 -0
  16. package/esm2022/communicate-micro.mjs +149 -0
  17. package/esm2022/constants.mjs +11 -0
  18. package/esm2022/crypto-3rd.mjs +38 -0
  19. package/esm2022/crypto.mjs +41 -0
  20. package/esm2022/dangerous-object.mjs +149 -0
  21. package/esm2022/date.mjs +191 -0
  22. package/esm2022/dom.mjs +256 -0
  23. package/esm2022/download.mjs +41 -0
  24. package/esm2022/file.mjs +90 -0
  25. package/esm2022/format-number.mjs +66 -0
  26. package/esm2022/format-text.mjs +149 -0
  27. package/esm2022/function-check-embed-frame.mjs +10 -0
  28. package/esm2022/get-smart-axis-scale.mjs +174 -0
  29. package/esm2022/helpers.mjs +651 -0
  30. package/esm2022/http-params.mjs +80 -0
  31. package/esm2022/index.mjs +30 -0
  32. package/esm2022/inject-token.mjs +5 -0
  33. package/esm2022/key-cache.mjs +31 -0
  34. package/esm2022/key-code.mjs +123 -0
  35. package/esm2022/language.mjs +70 -0
  36. package/esm2022/libs-ui-utils.mjs +5 -0
  37. package/esm2022/pattern.mjs +62 -0
  38. package/esm2022/random.mjs +42 -0
  39. package/esm2022/two-way-signal-object.mjs +131 -0
  40. package/esm2022/uri.mjs +25 -0
  41. package/esm2022/url-search-params.mjs +99 -0
  42. package/esm2022/uuid.mjs +18 -0
  43. package/esm2022/xss-filter.mjs +10 -0
  44. package/fesm2022/libs-ui-utils.mjs +3234 -0
  45. package/fesm2022/libs-ui-utils.mjs.map +1 -0
  46. package/file.d.ts +18 -0
  47. package/format-number.d.ts +2 -0
  48. package/format-text.d.ts +11 -0
  49. package/function-check-embed-frame.d.ts +2 -0
  50. package/get-smart-axis-scale.d.ts +34 -0
  51. package/helpers.d.ts +270 -0
  52. package/http-params.d.ts +37 -0
  53. package/index.d.ts +29 -0
  54. package/inject-token.d.ts +4 -0
  55. package/key-cache.d.ts +1 -0
  56. package/key-code.d.ts +122 -0
  57. package/language.d.ts +37 -0
  58. package/package.json +29 -0
  59. package/pattern.d.ts +20 -0
  60. package/random.d.ts +3 -0
  61. package/two-way-signal-object.d.ts +15 -0
  62. package/uri.d.ts +5 -0
  63. package/url-search-params.d.ts +25 -0
  64. package/uuid.d.ts +1 -0
  65. package/xss-filter.d.ts +3 -0
@@ -0,0 +1,651 @@
1
+ /* eslint-disable @typescript-eslint/no-explicit-any */
2
+ import { HttpParams } from '@angular/common/http';
3
+ import { isSignal, signal } from '@angular/core';
4
+ import dayjs from 'dayjs';
5
+ import { isArray, isDangerousObject, isDayjsObject, isMap, isRegExp, isReturnAsIsObject, isSet } from './dangerous-object';
6
+ import { getDayjs } from './date';
7
+ import { UtilsHttpParamsRequest } from './http-params';
8
+ import { unwrapSignal } from './two-way-signal-object';
9
+ /**Các hàm tương tự thư viện lodash */
10
+ /**
11
+ * Kiểm tra xem một giá trị có phải là null hoặc undefined hay không
12
+ * @param value Giá trị cần kiểm tra
13
+ * @returns true nếu giá trị là null hoặc undefined, false nếu không
14
+ * @example
15
+ * isNil(null); // true
16
+ * isNil(undefined); // true
17
+ * isNil(0); // false
18
+ * isNil('hello'); // false
19
+ */
20
+ export const isNil = (value, options) => {
21
+ if (!options?.ignoreUnWrapSignal) {
22
+ value = unwrapSignal(value);
23
+ }
24
+ return value === null || value === undefined;
25
+ };
26
+ /**
27
+ * Kiểm tra xem một giá trị có phải là rỗng hay không
28
+ * @param value Giá trị cần kiểm tra
29
+ * @returns true nếu giá trị là null, rỗng hoặc undefined, false nếu không
30
+ * @example
31
+ * isEmpty(null); // true
32
+ * isEmpty(''); // true
33
+ * isEmpty(undefined); // true
34
+ * isEmpty({}); // true
35
+ * isEmpty([]); // true
36
+ * isEmpty([1, 2, 3]); // false
37
+ * isEmpty({ a: 1 }); // false
38
+ */
39
+ export const isEmpty = (value, options) => {
40
+ if (!options?.ignoreUnWrapSignal) {
41
+ value = unwrapSignal(value);
42
+ }
43
+ if (options?.ignoreCheckString && value === '') {
44
+ return false;
45
+ }
46
+ if (options?.ignoreCheckTypePrimitive) {
47
+ return typeof value === 'object' && isEmptyTypeObject(value);
48
+ }
49
+ return isEmptyTypeObject(value);
50
+ };
51
+ /**
52
+ * Kiểm tra xem một giá trị có phải là object rỗng hay không
53
+ * @param value Giá trị cần kiểm tra
54
+ * @returns true nếu giá trị là object rỗng, false nếu không
55
+ * @example
56
+ * isEmptyTypeObject({}); // true
57
+ * isEmptyTypeObject({ a: 1 }); // false
58
+ */
59
+ const isEmptyTypeObject = (value) => {
60
+ try {
61
+ if (isNil(value, { ignoreUnWrapSignal: true }) || value === '') {
62
+ return true;
63
+ }
64
+ if (typeof value !== 'object') {
65
+ return false;
66
+ }
67
+ if (Array.isArray(value)) {
68
+ return value.length === 0;
69
+ }
70
+ return Object.keys(value).length === 0;
71
+ }
72
+ catch (error) {
73
+ console.error(error);
74
+ return false;
75
+ }
76
+ };
77
+ /**
78
+ * Kiểm tra xem một giá trị có phải là null, rỗng, undefined hoặc 0 hay không
79
+ * @param value Giá trị cần kiểm tra
80
+ * @param options Cấu hình tùy chọn
81
+ * @param options.ignoreZero Nếu true, sẽ không kiểm tra giá trị 0
82
+ * @returns true nếu giá trị là null, rỗng, undefined hoặc 0, false nếu không
83
+ * @example
84
+ * isTruthy(null); // false
85
+ * isTruthy(''); // false
86
+ * isTruthy(undefined); // false
87
+ * isTruthy(0); // false
88
+ * isTruthy({}); // true
89
+ * isTruthy(0, { ignoreZero: true }); // true
90
+ */
91
+ export const isTruthy = (value, options) => {
92
+ return !isFalsy(value, options);
93
+ };
94
+ /**
95
+ * Kiểm tra xem một giá trị có phải là null, rỗng, undefined hoặc 0 hay không
96
+ * @param value Giá trị cần kiểm tra
97
+ * @param options Cấu hình tùy chọn
98
+ * @param options.ignoreZero Nếu true, sẽ không kiểm tra giá trị 0
99
+ * @returns true nếu giá trị là null, rỗng, undefined hoặc 0, false nếu không
100
+ * @example
101
+ * isFalsy(null); // true
102
+ * isFalsy(''); // true
103
+ * isFalsy(undefined); // true
104
+ * isFalsy(0); // true
105
+ * isFalsy({}); // false
106
+ * isFalsy(0, { ignoreZero: true }); // false
107
+ */
108
+ export const isFalsy = (value, options) => {
109
+ if (!options?.ignoreUnWrapSignal) {
110
+ value = unwrapSignal(value);
111
+ }
112
+ if (options?.ignoreZero && value === 0) {
113
+ return false;
114
+ }
115
+ if (options?.ignoreCheckString && value === '') {
116
+ return false;
117
+ }
118
+ return !value;
119
+ };
120
+ /**
121
+ * Loại bỏ các thuộc tính của đối tượng dựa trên một hàm điều kiện
122
+ * @param objData Đối tượng cần xử lý
123
+ * @param predicate Hàm điều kiện để kiểm tra giá trị của thuộc tính. Nếu hàm trả về true thì thuộc tính đó sẽ bị loại bỏ
124
+ * @returns Đối tượng mới sau khi đã loại bỏ các thuộc tính thỏa mãn điều kiện
125
+ * @example
126
+ * const obj = { a: 1, b: null, c: 3, d: undefined };
127
+ * omitBy(obj, isNil); // { a: 1, c: 3 }
128
+ */
129
+ export const omitBy = (objData, predicate) => {
130
+ if (!objData || typeof objData !== 'object' || Array.isArray(objData)) {
131
+ return objData;
132
+ }
133
+ const newObj = {};
134
+ Object.keys(objData).forEach((key) => {
135
+ const valueOfKey = get(objData, key);
136
+ if (!predicate(valueOfKey)) {
137
+ set(newObj, key, valueOfKey);
138
+ }
139
+ });
140
+ return newObj;
141
+ };
142
+ /**
143
+ * Lấy giá trị từ đối tượng theo đường dẫn chỉ định
144
+ *
145
+ * Hàm này giúp bạn truy cập vào các thuộc tính sâu bên trong một đối tượng một cách an toàn,
146
+ * tránh lỗi khi thuộc tính không tồn tại.
147
+ *
148
+ * @param obj Đối tượng nguồn cần lấy giá trị
149
+ * @param path Đường dẫn đến thuộc tính cần lấy. Có thể là:
150
+ * - Chuỗi: 'user.profile.name' hoặc 'items[0].title'
151
+ * - Mảng: ['user', 'profile', 'name'] hoặc ['items', '0', 'title']
152
+ * - Chuỗi rỗng '': trả về chính đối tượng gốc
153
+ * @param defaultValue Giá trị mặc định trả về khi không tìm thấy thuộc tính (mặc định: undefined)
154
+ * @param keepLastValueIfSignal Có giữ nguyên signal cuối cùng hay không (mặc định: false - sẽ gọi signal())
155
+ * @returns Giá trị tìm được hoặc giá trị mặc định
156
+ *
157
+ * @example
158
+ * // Ví dụ cơ bản
159
+ * const user = { name: 'John', age: 30 };
160
+ * get(user, 'name'); // 'John'
161
+ * get(user, 'email'); // undefined
162
+ * get(user, 'email', 'no-email'); // 'no-email'
163
+ *
164
+ * @example
165
+ * // Truyền path rỗng - trả về chính đối tượng gốc
166
+ * const data = { name: 'Alice', age: 25 };
167
+ * get(data, ''); // { name: 'Alice', age: 25 } (chính đối tượng data)
168
+ * get(data, '', 'default'); // { name: 'Alice', age: 25 } (bỏ qua defaultValue)
169
+ *
170
+ * @example
171
+ * // Truy cập thuộc tính sâu
172
+ * const data = {
173
+ * user: {
174
+ * profile: {
175
+ * name: 'Alice',
176
+ * settings: { theme: 'dark' }
177
+ * }
178
+ * }
179
+ * };
180
+ * get(data, 'user.profile.name'); // 'Alice'
181
+ * get(data, 'user.profile.settings.theme'); // 'dark'
182
+ * get(data, 'user.profile.avatar', 'default.jpg'); // 'default.jpg'
183
+ *
184
+ * @example
185
+ * // Truy cập mảng
186
+ * const items = [
187
+ * { name: 'Item 1', price: 100 },
188
+ * { name: 'Item 2', price: 200 }
189
+ * ];
190
+ * get(items, '[0].name'); // 'Item 1'
191
+ * get(items, '[1].name'); // 'Item 2'
192
+ * get(items, '[2].name', 'Not found'); // 'Not found'
193
+ *
194
+ * @example
195
+ * // Sử dụng với mảng path
196
+ * const nested = { a: { b: { c: 'deep value' } } };
197
+ * get(nested, ['a', 'b', 'c']); // 'deep value'
198
+ * get(nested, ['a', 'b', 'd'], 'default'); // 'default'
199
+ *
200
+ * @example
201
+ * // Trường hợp đặc biệt
202
+ * get(null, 'any.path'); // undefined
203
+ * get(undefined, 'any.path', 'fallback'); // 'fallback'
204
+ */
205
+ export const get = (obj, path, defaultValue, keepLastValueIfSignal) => {
206
+ // helper cast để tránh lặp lại kiểu điều kiện ở các return
207
+ const out = (v) => v;
208
+ if (isNil(obj)) {
209
+ return out(defaultValue);
210
+ }
211
+ obj = unwrapSignal(obj);
212
+ if (path === '') {
213
+ return out(obj);
214
+ }
215
+ if (obj instanceof HttpParams) {
216
+ return out(obj.get(`${path}`) ?? defaultValue);
217
+ }
218
+ if (obj instanceof DOMRect) {
219
+ return out(obj[path]);
220
+ }
221
+ const paths = Array.isArray(path)
222
+ ? path
223
+ : path
224
+ .replace(/\[(\d+)]/g, '.$1')
225
+ .split('.')
226
+ .filter((key) => key);
227
+ for (const index in paths) {
228
+ const key = paths[index];
229
+ if (obj instanceof CSSStyleDeclaration) {
230
+ obj = obj[key];
231
+ continue;
232
+ }
233
+ if (obj instanceof HTMLElement) {
234
+ obj = obj[key];
235
+ continue;
236
+ }
237
+ if (isNil(obj) || !Object.prototype.hasOwnProperty.call(obj, key)) {
238
+ return out(defaultValue);
239
+ }
240
+ const val = isSignal(obj[key]) && !keepLastValueIfSignal ? obj[key]() : obj[key];
241
+ obj = val;
242
+ }
243
+ return out(obj ?? defaultValue);
244
+ };
245
+ /**
246
+ * Thiết lập giá trị cho một thuộc tính trong đối tượng theo đường dẫn chỉ định
247
+ * @param obj Đối tượng cần thiết lập giá trị
248
+ * @param path Đường dẫn đến thuộc tính, có thể là chuỗi (vd: 'a.b.c') hoặc mảng (vd: ['a', 'b', 'c'])
249
+ * @param value Giá trị cần thiết lập
250
+ * @returns Đối tượng sau khi đã thiết lập giá trị
251
+ * @throws Error nếu tham số đầu tiên không phải là đối tượng
252
+ * @example
253
+ * const obj = { a: { b: 1 } };
254
+ * set(obj, 'a.b', 2); // { a: { b: 2 } }
255
+ */
256
+ export const set = (obj, path, value, options) => {
257
+ if (!obj || (typeof obj !== 'object' && !isSignal(obj)) || (isSignal(obj) && typeof obj() !== 'object')) {
258
+ throw new Error('The first argument must be an object');
259
+ }
260
+ if (obj instanceof HttpParams) {
261
+ return obj.set(`${path}`, value);
262
+ }
263
+ const pathArray = Array.isArray(path)
264
+ ? path
265
+ : path
266
+ .replace(/\[(\d+)]/g, '.[$1]')
267
+ .split('.')
268
+ .filter((key) => key);
269
+ let currentObjectByKey = isSignal(obj) ? obj() : obj;
270
+ let preObjectByKey = obj;
271
+ pathArray.forEach((key, index) => {
272
+ if (index < pathArray.length - 1) {
273
+ if (!(key in currentObjectByKey) || (typeof currentObjectByKey[key] !== 'object' && !isSignal(currentObjectByKey[key]))) {
274
+ const nextKey = pathArray[index + 1];
275
+ key = key.replace(/\[(\d+)]/g, '$1');
276
+ if (isNil(currentObjectByKey[key])) {
277
+ currentObjectByKey[key] = /\[(\d+)]/g.test(nextKey) ? (options?.valueDefaultPathArrayUndefined ?? []) : (options?.valueDefaultPathObjectUndefined ?? {});
278
+ }
279
+ }
280
+ currentObjectByKey = key ? currentObjectByKey[key] : currentObjectByKey;
281
+ preObjectByKey = currentObjectByKey;
282
+ currentObjectByKey = isSignal(currentObjectByKey) ? currentObjectByKey() : currentObjectByKey;
283
+ return;
284
+ }
285
+ if (typeof currentObjectByKey !== 'object') {
286
+ return;
287
+ }
288
+ // Gán giá trị ở cuối đường dẫn
289
+ key = key.replace(/\[(\d+)]/g, '$1');
290
+ const valueOfKey = currentObjectByKey[key];
291
+ const valueOfKeyIsSignal = isSignal(valueOfKey);
292
+ if (valueOfKeyIsSignal && (typeof valueOfKey() !== 'object' || valueOfKey() === null)) {
293
+ valueOfKey.set(isSignal(value) ? value() : value);
294
+ return;
295
+ }
296
+ if (isSignal(preObjectByKey)) {
297
+ preObjectByKey.update((data) => {
298
+ const dataOfKey = data[key];
299
+ if (isNil(dataOfKey) || !valueOfKeyIsSignal) {
300
+ data[key] = value;
301
+ }
302
+ if (valueOfKeyIsSignal) {
303
+ valueOfKey.set(isSignal(value) ? value() : value);
304
+ }
305
+ if (Array.isArray(data)) {
306
+ return [...data];
307
+ }
308
+ return { ...data };
309
+ });
310
+ return;
311
+ }
312
+ if (valueOfKeyIsSignal) {
313
+ valueOfKey.set(isSignal(value) ? value() : value);
314
+ return;
315
+ }
316
+ currentObjectByKey[key] = value;
317
+ });
318
+ return obj;
319
+ };
320
+ /**
321
+ * Tạo một bản sao sâu của một đối tượng hoặc giá trị bất kỳ
322
+ * @param data Dữ liệu cần sao chép
323
+ * @param options Tùy chọn cấu hình
324
+ * @param options.ignoreSignal Nếu true, sẽ không sao chép các signal mà trả về signal gốc
325
+ * @param seen WeakMap để theo dõi các tham chiếu đã được sao chép, tránh lặp vô hạn với các tham chiếu vòng
326
+ * @returns Bản sao sâu của dữ liệu đầu vào
327
+ * @example
328
+ * const obj = {
329
+ * a: 1,
330
+ * b: { c: 2 },
331
+ * d: [1, 2, 3]
332
+ * };
333
+ * const clone = cloneDeep(obj);
334
+ * // clone là một bản sao hoàn toàn độc lập của obj
335
+ */
336
+ export const cloneDeep = (data, options = { ignoreSignal: false }, seen = new WeakMap()) => {
337
+ if (data === null || (typeof data !== 'object' && !isSignal(data)) || isDangerousObject(data)) {
338
+ return data;
339
+ }
340
+ if (seen.has(data)) {
341
+ return seen.get(data);
342
+ }
343
+ if (data instanceof HttpParams) {
344
+ return new UtilsHttpParamsRequest(undefined, data);
345
+ }
346
+ if (data instanceof Date) {
347
+ return new Date(data.getTime());
348
+ }
349
+ if (isDayjsObject(data)) {
350
+ return getDayjs({ date: data.valueOf() });
351
+ }
352
+ if (isRegExp(data)) {
353
+ return new RegExp(data.source, data.flags);
354
+ }
355
+ if (isMap(data)) {
356
+ const mapCopy = new Map();
357
+ seen.set(data, mapCopy);
358
+ data.forEach((val, key) => {
359
+ mapCopy.set(cloneDeep(key, options, seen), cloneDeep(val, options, seen));
360
+ });
361
+ return mapCopy;
362
+ }
363
+ if (isSet(data)) {
364
+ const setCopy = new Set();
365
+ seen.set(data, setCopy);
366
+ data.forEach((val) => {
367
+ setCopy.add(cloneDeep(val, options, seen));
368
+ });
369
+ return setCopy;
370
+ }
371
+ if (isArray(data)) {
372
+ seen.set(data, data.map((item) => cloneDeep(item, options, seen)));
373
+ return seen.get(data);
374
+ }
375
+ if (isReturnAsIsObject(data)) {
376
+ return data;
377
+ }
378
+ if (isSignal(data)) {
379
+ if (options?.ignoreSignal) {
380
+ return data;
381
+ }
382
+ seen.set(data, signal(cloneDeep(data(), options, seen)));
383
+ return seen.get(data);
384
+ }
385
+ const result = {};
386
+ seen.set(data, result);
387
+ for (const key in data) {
388
+ const value = data[key];
389
+ if (value instanceof HttpParams) {
390
+ result[key] = new UtilsHttpParamsRequest(undefined, value);
391
+ continue;
392
+ }
393
+ if (value instanceof Date) {
394
+ result[key] = new Date(value.getTime());
395
+ continue;
396
+ }
397
+ if (dayjs.isDayjs(value)) {
398
+ result[key] = getDayjs({ date: value.valueOf() });
399
+ continue;
400
+ }
401
+ if (isRegExp(data)) {
402
+ result[key] = new RegExp(value.source, value.flags);
403
+ continue;
404
+ }
405
+ if (isReturnAsIsObject(data)) {
406
+ result[key] = value;
407
+ continue;
408
+ }
409
+ if (Object.prototype.hasOwnProperty.call(data, key)) {
410
+ result[key] = cloneDeep(value, options, seen);
411
+ }
412
+ }
413
+ return result;
414
+ };
415
+ /**
416
+ * Chuyển đổi một mảng các đối tượng thành một đối tượng với khóa được chỉ định
417
+ * @param data Mảng các đối tượng cần chuyển đổi
418
+ * @param key Tên thuộc tính được sử dụng làm khóa trong đối tượng kết quả
419
+ * @returns Đối tượng với các giá trị từ mảng được đánh key theo thuộc tính đã chỉ định
420
+ * @example
421
+ * const data = [
422
+ * { id: 1, name: 'John' },
423
+ * { id: 2, name: 'Jane' }
424
+ * ];
425
+ * keyBy(data, 'id');
426
+ * // Kết quả: {
427
+ * // '1': { id: 1, name: 'John' },
428
+ * // '2': { id: 2, name: 'Jane' }
429
+ * // }
430
+ */
431
+ export const keyBy = (data, key) => {
432
+ if (!data || !data.length || !key) {
433
+ return {};
434
+ }
435
+ return data.reduce((dir, nextItem) => {
436
+ const valueOfKey = get(nextItem, key);
437
+ if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {
438
+ return dir;
439
+ }
440
+ if (!Object.keys(dir).includes(`${valueOfKey}`)) {
441
+ dir[`${valueOfKey}`] = nextItem;
442
+ }
443
+ return dir;
444
+ }, {});
445
+ };
446
+ /**
447
+ * Nhóm các đối tượng trong một mảng thành các nhóm dựa trên một thuộc tính cụ thể
448
+ * @param data Mảng các đối tượng cần nhóm
449
+ * @param key Tên thuộc tính được sử dụng làm khóa nhóm
450
+ * @returns Đối tượng với các giá trị từ mảng được nhóm theo thuộc tính đã chỉ định
451
+ * @example
452
+ * const data = [
453
+ * { id: 1, name: 'John' },
454
+ * { id: 2, name: 'Jane' },
455
+ * { id: 1, name: 'John' }
456
+ * ];
457
+ * groupBy(data, 'id');
458
+ * // Kết quả: {
459
+ * // '1': [
460
+ * // { id: 1, name: 'John' },
461
+ * // { id: 1, name: 'John' }
462
+ * // ],
463
+ * // '2': [
464
+ * // { id: 2, name: 'Jane' }
465
+ * // }
466
+ */
467
+ export const groupBy = (data, key) => {
468
+ if (!data || !data.length || !Object.keys(get(data, '[0]')).length || !key) {
469
+ return {};
470
+ }
471
+ return data.reduce((dir, nextItem) => {
472
+ const valueOfKey = get(nextItem, key);
473
+ if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {
474
+ return dir;
475
+ }
476
+ if (!Object.keys(dir).includes(`${valueOfKey}`)) {
477
+ dir[`${valueOfKey}`] = [];
478
+ }
479
+ dir[`${valueOfKey}`].push(nextItem);
480
+ return dir;
481
+ }, {});
482
+ };
483
+ /**
484
+ * Tạo một mảng các số từ giá trị bắt đầu đến giá trị kết thúc với bước nhảy tùy chọn
485
+ * @param start Giá trị bắt đầu của dãy số. Nếu chỉ có một tham số, đây sẽ là giá trị kết thúc và giá trị bắt đầu sẽ là 0
486
+ * @param end Giá trị kết thúc của dãy số (tùy chọn)
487
+ * @param step Bước nhảy giữa các số trong dãy (tùy chọn). Mặc định là 1 nếu end > start, -1 nếu end < start
488
+ * @returns Mảng các số từ start đến end với bước nhảy step
489
+ * @example
490
+ * range(4); // [0, 1, 2, 3]
491
+ * range(1, 5); // [1, 2, 3, 4]
492
+ * range(0, 20, 5); // [0, 5, 10, 15]
493
+ * range(5, 2); // [5, 4, 3]
494
+ */
495
+ export const range = (start, end, step) => {
496
+ if (end === undefined || end === null) {
497
+ end = start;
498
+ start = 0;
499
+ }
500
+ if (!step) {
501
+ step = end < 0 ? -1 : 1;
502
+ }
503
+ if (end < start && step > 0) {
504
+ step *= -1;
505
+ }
506
+ const valueStartByStep = step + start;
507
+ const direction = start < end ? 'asc' : 'desc';
508
+ if ((direction === 'asc' && (valueStartByStep < start || valueStartByStep > end)) || (direction === 'desc' && (valueStartByStep > start || valueStartByStep < end))) {
509
+ return [start];
510
+ }
511
+ const arr = new Array();
512
+ for (let index = 0; index < Math.abs(end - start); index++) {
513
+ let value = start + index * step;
514
+ if (index === 0) {
515
+ value = start;
516
+ }
517
+ if ((direction === 'asc' && (value < start || value > end)) || (direction === 'desc' && (value > start || value < end))) {
518
+ return arr;
519
+ }
520
+ arr.push(value);
521
+ }
522
+ return arr;
523
+ };
524
+ /**
525
+ * So sánh hai giá trị bất kỳ có bằng nhau hay không
526
+ * @param value1 Giá trị thứ nhất cần so sánh
527
+ * @param value2 Giá trị thứ hai cần so sánh
528
+ * @param exactlyPosition Có so sánh chính xác vị trí các phần tử trong mảng hay không
529
+ * @returns true nếu hai giá trị bằng nhau, false nếu không bằng nhau
530
+ * @example
531
+ * isEqual([1,2,3], [1,2,3]); // true
532
+ * isEqual([1,2,3], [3,2,1]); // true khi exactlyPosition = false
533
+ * isEqual([1,2,3], [3,2,1]); // false khi exactlyPosition = true
534
+ * isEqual({a:1}, {a:1}); // true
535
+ */
536
+ export const isEqual = (value1, value2, options) => {
537
+ // Handle signals
538
+ if (!options?.ignoreUnWrapSignal) {
539
+ value1 = unwrapSignal(value1);
540
+ value2 = unwrapSignal(value2);
541
+ }
542
+ const { exactlyPosition = false, ignoreExactlyDataType = false } = options || {};
543
+ if (value1 === value2 || (value1 === null && value2 === null) || (value1 === undefined && value2 === undefined)) {
544
+ return true;
545
+ }
546
+ if (ignoreExactlyDataType) {
547
+ return isEqual(isNil(value1) ? undefined : `${value1}`, isNil(value2) ? undefined : `${value2}`);
548
+ }
549
+ if (typeof value1 !== 'object' || typeof value2 !== 'object' || (Array.isArray(value1) && !Array.isArray(value2)) || (!Array.isArray(value1) && Array.isArray(value2))) {
550
+ return false;
551
+ }
552
+ if (Array.isArray(value1) || Array.isArray(value2)) {
553
+ if (value1.length !== value2.length) {
554
+ return false;
555
+ }
556
+ if (!exactlyPosition) {
557
+ return !value1.some((item) => !value2.includes(item));
558
+ }
559
+ return !value1.some((item, index) => !isEqual(item, value2[index], options));
560
+ }
561
+ if (Object.keys(value1).length !== Object.keys(value2).length) {
562
+ return false;
563
+ }
564
+ return !Object.keys(value1).some((key) => !isEqual(value1[key], value2[key], options));
565
+ };
566
+ /**
567
+ * Loại bỏ các phần tử trùng lặp trong mảng dựa trên một thuộc tính chỉ định
568
+ * @param data Mảng dữ liệu cần xử lý
569
+ * @param key Tên thuộc tính dùng để so sánh trùng lặp. Nếu không có key thì so sánh trực tiếp giá trị
570
+ * @returns Mảng mới chứa các phần tử không trùng lặp
571
+ * @example
572
+ * const arr = [
573
+ * { id: 1, name: 'A' },
574
+ * { id: 2, name: 'B' },
575
+ * { id: 1, name: 'C' }
576
+ * ];
577
+ * uniqBy(arr, 'id'); // [{ id: 1, name: 'A' }, { id: 2, name: 'B' }]
578
+ *
579
+ * const numbers = [1, 2, 2, 3, 3];
580
+ * uniqBy(numbers); // [1, 2, 3]
581
+ *
582
+ * const numbersSignal = [signal(1), signal(2), signal(3), signal(2), signal(5), signal(4), signal(1), signal(6), signal(7), signal(6)];
583
+ * uniqBy(numbersSignal); // [signal(1), signal(2), signal(3), signal(5), signal(4), signal(6), signal(7)]
584
+ */
585
+ export const uniqBy = (data, key) => {
586
+ if (!key || !data?.length || typeof get(data, '[0]') !== 'object') {
587
+ // Xử lý mảng chứa signal values
588
+ if (data[0] && isSignal(data[0])) {
589
+ const seen = new Set();
590
+ return data.filter((item) => {
591
+ const value = `${get(item, '')}`;
592
+ if (seen.has(value)) {
593
+ return false;
594
+ }
595
+ seen.add(value);
596
+ return true;
597
+ });
598
+ }
599
+ // Xử lý mảng primitive values
600
+ return Array.from(new Set(data));
601
+ }
602
+ const dataUnique = keyBy(data, key);
603
+ return Object.keys(dataUnique).map((key) => dataUnique[key]);
604
+ };
605
+ export const generateInterface = (obj, interfaceName) => {
606
+ const generateType = (value) => {
607
+ if (value === null) {
608
+ return 'null';
609
+ }
610
+ const type = typeof value;
611
+ if (type === 'string') {
612
+ return 'string';
613
+ }
614
+ if (type === 'number') {
615
+ return 'number';
616
+ }
617
+ if (type === 'boolean') {
618
+ return 'boolean';
619
+ }
620
+ if (type === 'undefined') {
621
+ return 'any';
622
+ }
623
+ if (value instanceof Date) {
624
+ return 'Date';
625
+ }
626
+ if (value instanceof RegExp) {
627
+ return 'RegExp';
628
+ }
629
+ if (Array.isArray(value)) {
630
+ if (value.length === 0) {
631
+ return 'Array<any>';
632
+ }
633
+ return `Array<${generateType(value[0])}>`;
634
+ }
635
+ if (type === 'object') {
636
+ let interfaceStr = '{\n';
637
+ for (const key in value) {
638
+ if (Object.prototype.hasOwnProperty.call(value, key)) {
639
+ const valueType = generateType(value[key]);
640
+ interfaceStr += ` ${key}: ${valueType};\n`;
641
+ }
642
+ }
643
+ interfaceStr += '}';
644
+ return interfaceStr;
645
+ }
646
+ return 'any';
647
+ };
648
+ const interfaceStr = `interface ${interfaceName} ${generateType(obj)}`;
649
+ return interfaceStr;
650
+ };
651
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/helpers.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { HttpParams } from '@angular/common/http';\nimport { Signal, WritableSignal, isSignal, signal } from '@angular/core';\nimport { GetReturnType, GetValueAtPath, PathOf, TYPE_OBJECT } from '@libs-ui/interfaces-types';\nimport dayjs from 'dayjs';\nimport { isArray, isDangerousObject, isDayjsObject, isMap, isRegExp, isReturnAsIsObject, isSet } from './dangerous-object';\nimport { getDayjs } from './date';\nimport { UtilsHttpParamsRequest } from './http-params';\nimport { unwrapSignal } from './two-way-signal-object';\n\n/**Các hàm tương tự thư viện lodash */\n/**\n * Kiểm tra xem một giá trị có phải là null hoặc undefined hay không\n * @param value Giá trị cần kiểm tra\n * @returns true nếu giá trị là null hoặc undefined, false nếu không\n * @example\n * isNil(null); // true\n * isNil(undefined); // true\n * isNil(0); // false\n * isNil('hello'); // false\n */\nexport const isNil = (value: unknown, options?: { ignoreUnWrapSignal?: boolean }): value is null | undefined => {\n  if (!options?.ignoreUnWrapSignal) {\n    value = unwrapSignal(value);\n  }\n  return value === null || value === undefined;\n};\n\n/**\n * Kiểm tra xem một giá trị có phải là rỗng hay không\n * @param value Giá trị cần kiểm tra\n * @returns true nếu giá trị là null, rỗng hoặc undefined, false nếu không\n * @example\n * isEmpty(null); // true\n * isEmpty(''); // true\n * isEmpty(undefined); // true\n * isEmpty({}); // true\n * isEmpty([]); // true\n * isEmpty([1, 2, 3]); // false\n * isEmpty({ a: 1 }); // false\n */\nexport const isEmpty = (value: any, options?: { ignoreCheckTypePrimitive?: boolean; ignoreUnWrapSignal?: boolean; ignoreCheckString?: boolean }): value is null | undefined | '' | 0 => {\n  if (!options?.ignoreUnWrapSignal) {\n    value = unwrapSignal(value);\n  }\n  if (options?.ignoreCheckString && value === '') {\n    return false;\n  }\n  if (options?.ignoreCheckTypePrimitive) {\n    return typeof value === 'object' && isEmptyTypeObject(value);\n  }\n\n  return isEmptyTypeObject(value);\n};\n\n/**\n * Kiểm tra xem một giá trị có phải là object rỗng hay không\n * @param value Giá trị cần kiểm tra\n * @returns true nếu giá trị là object rỗng, false nếu không\n * @example\n * isEmptyTypeObject({}); // true\n * isEmptyTypeObject({ a: 1 }); // false\n */\nconst isEmptyTypeObject = <T = null>(value: any): value is T => {\n  try {\n    if (isNil(value, { ignoreUnWrapSignal: true }) || value === '') {\n      return true;\n    }\n    if (typeof value !== 'object') {\n      return false;\n    }\n\n    if (Array.isArray(value)) {\n      return value.length === 0;\n    }\n\n    return Object.keys(value).length === 0;\n  } catch (error) {\n    console.error(error);\n    return false;\n  }\n};\n\n/**\n * Kiểm tra xem một giá trị có phải là null, rỗng, undefined hoặc 0 hay không\n * @param value Giá trị cần kiểm tra\n * @param options Cấu hình tùy chọn\n * @param options.ignoreZero Nếu true, sẽ không kiểm tra giá trị 0\n * @returns true nếu giá trị là null, rỗng, undefined hoặc 0, false nếu không\n * @example\n * isTruthy(null); // false\n * isTruthy(''); // false\n * isTruthy(undefined); // false\n * isTruthy(0); // false\n * isTruthy({}); // true\n * isTruthy(0, { ignoreZero: true }); // true\n */\n\nexport const isTruthy = <T>(value: T, options?: { ignoreZero?: boolean; ignoreCheckString?: boolean; ignoreUnWrapSignal?: boolean }): value is NonNullable<T> => {\n  return !isFalsy(value, options);\n};\n\n/**\n * Kiểm tra xem một giá trị có phải là null, rỗng, undefined hoặc 0 hay không\n * @param value Giá trị cần kiểm tra\n * @param options Cấu hình tùy chọn\n * @param options.ignoreZero Nếu true, sẽ không kiểm tra giá trị 0\n * @returns true nếu giá trị là null, rỗng, undefined hoặc 0, false nếu không\n * @example\n * isFalsy(null); // true\n * isFalsy(''); // true\n * isFalsy(undefined); // true\n * isFalsy(0); // true\n * isFalsy({}); // false\n * isFalsy(0, { ignoreZero: true }); // false\n */\nexport const isFalsy = (value: unknown, options?: { ignoreZero?: boolean; ignoreCheckString?: boolean; ignoreUnWrapSignal?: boolean }): value is null | undefined | '' | 0 => {\n  if (!options?.ignoreUnWrapSignal) {\n    value = unwrapSignal(value);\n  }\n\n  if (options?.ignoreZero && value === 0) {\n    return false;\n  }\n\n  if (options?.ignoreCheckString && value === '') {\n    return false;\n  }\n\n  return !value;\n};\n\n/**\n * Loại bỏ các thuộc tính của đối tượng dựa trên một hàm điều kiện\n * @param objData Đối tượng cần xử lý\n * @param predicate Hàm điều kiện để kiểm tra giá trị của thuộc tính. Nếu hàm trả về true thì thuộc tính đó sẽ bị loại bỏ\n * @returns Đối tượng mới sau khi đã loại bỏ các thuộc tính thỏa mãn điều kiện\n * @example\n * const obj = { a: 1, b: null, c: 3, d: undefined };\n * omitBy(obj, isNil); // { a: 1, c: 3 }\n */\nexport const omitBy = <T>(objData: Record<string, any>, predicate: (val: any) => boolean): T => {\n  if (!objData || typeof objData !== 'object' || Array.isArray(objData)) {\n    return objData as T;\n  }\n  const newObj = {} as T;\n\n  Object.keys(objData).forEach((key) => {\n    const valueOfKey = get(objData, key);\n    if (!predicate(valueOfKey)) {\n      set(newObj, key as PathOf<T>, valueOfKey);\n    }\n  });\n  return newObj;\n};\n\n/**\n * Lấy giá trị từ đối tượng theo đường dẫn chỉ định\n *\n * Hàm này giúp bạn truy cập vào các thuộc tính sâu bên trong một đối tượng một cách an toàn,\n * tránh lỗi khi thuộc tính không tồn tại.\n *\n * @param obj Đối tượng nguồn cần lấy giá trị\n * @param path Đường dẫn đến thuộc tính cần lấy. Có thể là:\n *   - Chuỗi: 'user.profile.name' hoặc 'items[0].title'\n *   - Mảng: ['user', 'profile', 'name'] hoặc ['items', '0', 'title']\n *   - Chuỗi rỗng '': trả về chính đối tượng gốc\n * @param defaultValue Giá trị mặc định trả về khi không tìm thấy thuộc tính (mặc định: undefined)\n * @param keepLastValueIfSignal Có giữ nguyên signal cuối cùng hay không (mặc định: false - sẽ gọi signal())\n * @returns Giá trị tìm được hoặc giá trị mặc định\n *\n * @example\n * // Ví dụ cơ bản\n * const user = { name: 'John', age: 30 };\n * get(user, 'name'); // 'John'\n * get(user, 'email'); // undefined\n * get(user, 'email', 'no-email'); // 'no-email'\n *\n * @example\n * // Truyền path rỗng - trả về chính đối tượng gốc\n * const data = { name: 'Alice', age: 25 };\n * get(data, ''); // { name: 'Alice', age: 25 } (chính đối tượng data)\n * get(data, '', 'default'); // { name: 'Alice', age: 25 } (bỏ qua defaultValue)\n *\n * @example\n * // Truy cập thuộc tính sâu\n * const data = {\n *   user: {\n *     profile: {\n *       name: 'Alice',\n *       settings: { theme: 'dark' }\n *     }\n *   }\n * };\n * get(data, 'user.profile.name'); // 'Alice'\n * get(data, 'user.profile.settings.theme'); // 'dark'\n * get(data, 'user.profile.avatar', 'default.jpg'); // 'default.jpg'\n *\n * @example\n * // Truy cập mảng\n * const items = [\n *   { name: 'Item 1', price: 100 },\n *   { name: 'Item 2', price: 200 }\n * ];\n * get(items, '[0].name'); // 'Item 1'\n * get(items, '[1].name'); // 'Item 2'\n * get(items, '[2].name', 'Not found'); // 'Not found'\n *\n * @example\n * // Sử dụng với mảng path\n * const nested = { a: { b: { c: 'deep value' } } };\n * get(nested, ['a', 'b', 'c']); // 'deep value'\n * get(nested, ['a', 'b', 'd'], 'default'); // 'default'\n *\n * @example\n * // Trường hợp đặc biệt\n * get(null, 'any.path'); // undefined\n * get(undefined, 'any.path', 'fallback'); // 'fallback'\n */\nexport const get = <O, P extends PathOf<O> = PathOf<O>, KS extends boolean = false, D extends GetValueAtPath<O, P, KS> | undefined = undefined>(\n  obj: Signal<O> | O,\n  path: P,\n  defaultValue?: D,\n  keepLastValueIfSignal?: KS\n): GetReturnType<O, P, KS, D> => {\n  // helper cast để tránh lặp lại kiểu điều kiện ở các return\n  const out = (v: unknown) => v as GetReturnType<O, P, KS, D>;\n\n  if (isNil(obj)) {\n    return out(defaultValue);\n  }\n  obj = unwrapSignal(obj);\n  if (path === '') {\n    return out(obj);\n  }\n  if (obj instanceof HttpParams) {\n    return out(obj.get(`${path}`) ?? defaultValue);\n  }\n  if (obj instanceof DOMRect) {\n    return out(obj[path as keyof DOMRect]);\n  }\n\n  const paths = Array.isArray(path)\n    ? path\n    : (path as string)\n        .replace(/\\[(\\d+)]/g, '.$1')\n        .split('.')\n        .filter((key) => key);\n  for (const index in paths) {\n    const key = paths[index];\n    if (obj instanceof CSSStyleDeclaration) {\n      obj = obj[key as keyof CSSStyleDeclaration] as O;\n      continue;\n    }\n\n    if (obj instanceof HTMLElement) {\n      obj = obj[key as keyof HTMLElement] as O;\n      continue;\n    }\n    if (isNil(obj) || !Object.prototype.hasOwnProperty.call(obj, key)) {\n      return out(defaultValue);\n    }\n\n    const val = isSignal((obj as any)[key]) && !keepLastValueIfSignal ? (obj as any)[key]() : (obj as any)[key];\n\n    obj = val as O;\n  }\n\n  return out(obj ?? defaultValue);\n};\n\n/**\n * Thiết lập giá trị cho một thuộc tính trong đối tượng theo đường dẫn chỉ định\n * @param obj Đối tượng cần thiết lập giá trị\n * @param path Đường dẫn đến thuộc tính, có thể là chuỗi (vd: 'a.b.c') hoặc mảng (vd: ['a', 'b', 'c'])\n * @param value Giá trị cần thiết lập\n * @returns Đối tượng sau khi đã thiết lập giá trị\n * @throws Error nếu tham số đầu tiên không phải là đối tượng\n * @example\n * const obj = { a: { b: 1 } };\n * set(obj, 'a.b', 2); // { a: { b: 2 } }\n */\nexport const set = <O, P extends PathOf<O> = PathOf<O>, T extends GetValueAtPath<O, P, true> = GetValueAtPath<O, P, true>>(\n  obj: Signal<O> | O,\n  path: P,\n  value: T,\n  options?: { valueDefaultPathObjectUndefined?: any; valueDefaultPathArrayUndefined?: any }\n): O => {\n  if (!obj || (typeof obj !== 'object' && !isSignal(obj)) || (isSignal(obj) && typeof obj() !== 'object')) {\n    throw new Error('The first argument must be an object');\n  }\n\n  if (obj instanceof HttpParams) {\n    return obj.set(`${path}`, value as string) as O;\n  }\n\n  const pathArray = Array.isArray(path)\n    ? path\n    : (path as string)\n        .replace(/\\[(\\d+)]/g, '.[$1]')\n        .split('.')\n        .filter((key) => key);\n\n  let currentObjectByKey: any = isSignal(obj) ? obj() : obj;\n  let preObjectByKey = obj;\n\n  pathArray.forEach((key, index) => {\n    if (index < pathArray.length - 1) {\n      if (!(key in currentObjectByKey) || (typeof currentObjectByKey[key] !== 'object' && !isSignal(currentObjectByKey[key]))) {\n        const nextKey = pathArray[index + 1];\n        key = key.replace(/\\[(\\d+)]/g, '$1');\n        if (isNil(currentObjectByKey[key])) {\n          currentObjectByKey[key] = /\\[(\\d+)]/g.test(nextKey) ? (options?.valueDefaultPathArrayUndefined ?? []) : (options?.valueDefaultPathObjectUndefined ?? {});\n        }\n      }\n      currentObjectByKey = key ? currentObjectByKey[key] : currentObjectByKey;\n      preObjectByKey = currentObjectByKey;\n      currentObjectByKey = isSignal(currentObjectByKey) ? currentObjectByKey() : currentObjectByKey;\n      return;\n    }\n\n    if (typeof currentObjectByKey !== 'object') {\n      return;\n    }\n\n    // Gán giá trị ở cuối đường dẫn\n    key = key.replace(/\\[(\\d+)]/g, '$1');\n    const valueOfKey = currentObjectByKey[key];\n    const valueOfKeyIsSignal = isSignal(valueOfKey);\n\n    if (valueOfKeyIsSignal && (typeof valueOfKey() !== 'object' || valueOfKey() === null)) {\n      (valueOfKey as WritableSignal<any>).set(isSignal(value) ? value() : value);\n      return;\n    }\n    if (isSignal(preObjectByKey)) {\n      (preObjectByKey as WritableSignal<any>).update((data: any) => {\n        const dataOfKey = data[key];\n        if (isNil(dataOfKey) || !valueOfKeyIsSignal) {\n          data[key] = value;\n        }\n\n        if (valueOfKeyIsSignal) {\n          (valueOfKey as WritableSignal<any>).set(isSignal(value) ? value() : value);\n        }\n\n        if (Array.isArray(data)) {\n          return [...data];\n        }\n        return { ...data };\n      });\n      return;\n    }\n    if (valueOfKeyIsSignal) {\n      (valueOfKey as WritableSignal<any>).set(isSignal(value) ? value() : value);\n      return;\n    }\n    currentObjectByKey[key] = value;\n  });\n\n  return obj as O;\n};\n\n/**\n * Tạo một bản sao sâu của một đối tượng hoặc giá trị bất kỳ\n * @param data Dữ liệu cần sao chép\n * @param options Tùy chọn cấu hình\n * @param options.ignoreSignal Nếu true, sẽ không sao chép các signal mà trả về signal gốc\n * @param seen WeakMap để theo dõi các tham chiếu đã được sao chép, tránh lặp vô hạn với các tham chiếu vòng\n * @returns Bản sao sâu của dữ liệu đầu vào\n * @example\n * const obj = {\n *   a: 1,\n *   b: { c: 2 },\n *   d: [1, 2, 3]\n * };\n * const clone = cloneDeep(obj);\n * // clone là một bản sao hoàn toàn độc lập của obj\n */\nexport const cloneDeep = <T = any>(data: any, options = { ignoreSignal: false }, seen = new WeakMap()): T => {\n  if (data === null || (typeof data !== 'object' && !isSignal(data)) || isDangerousObject(data)) {\n    return data;\n  }\n\n  if (seen.has(data)) {\n    return seen.get(data);\n  }\n\n  if (data instanceof HttpParams) {\n    return new UtilsHttpParamsRequest(undefined, data) as T;\n  }\n\n  if (data instanceof Date) {\n    return new Date(data.getTime()) as T;\n  }\n\n  if (isDayjsObject(data)) {\n    return getDayjs({ date: data.valueOf() }) as T;\n  }\n\n  if (isRegExp(data)) {\n    return new RegExp(data.source, data.flags) as T;\n  }\n\n  if (isMap(data)) {\n    const mapCopy = new Map();\n    seen.set(data, mapCopy);\n    data.forEach((val, key) => {\n      mapCopy.set(cloneDeep(key, options, seen), cloneDeep(val, options, seen));\n    });\n    return mapCopy as T;\n  }\n\n  if (isSet(data)) {\n    const setCopy = new Set();\n    seen.set(data, setCopy);\n    data.forEach((val) => {\n      setCopy.add(cloneDeep(val, options, seen));\n    });\n    return setCopy as T;\n  }\n\n  if (isArray(data)) {\n    seen.set(data, data.map((item) => cloneDeep(item, options, seen)) as T);\n    return seen.get(data) as T;\n  }\n\n  if (isReturnAsIsObject(data)) {\n    return data as T;\n  }\n\n  if (isSignal(data)) {\n    if (options?.ignoreSignal) {\n      return data as T;\n    }\n    seen.set(data, signal(cloneDeep(data(), options, seen)));\n    return seen.get(data) as T;\n  }\n\n  const result: Record<string, any> = {};\n  seen.set(data, result);\n  for (const key in data) {\n    const value = data[key];\n    if (value instanceof HttpParams) {\n      result[key] = new UtilsHttpParamsRequest(undefined, value) as T;\n      continue;\n    }\n\n    if (value instanceof Date) {\n      result[key] = new Date(value.getTime()) as T;\n      continue;\n    }\n\n    if (dayjs.isDayjs(value)) {\n      result[key] = getDayjs({ date: value.valueOf() }) as T;\n      continue;\n    }\n\n    if (isRegExp(data)) {\n      result[key] = new RegExp(value.source, value.flags) as T;\n      continue;\n    }\n    if (isReturnAsIsObject(data)) {\n      result[key] = value;\n      continue;\n    }\n\n    if (Object.prototype.hasOwnProperty.call(data, key)) {\n      result[key] = cloneDeep(value, options, seen);\n    }\n  }\n\n  return result as T;\n};\n\n/**\n * Chuyển đổi một mảng các đối tượng thành một đối tượng với khóa được chỉ định\n * @param data Mảng các đối tượng cần chuyển đổi\n * @param key Tên thuộc tính được sử dụng làm khóa trong đối tượng kết quả\n * @returns Đối tượng với các giá trị từ mảng được đánh key theo thuộc tính đã chỉ định\n * @example\n * const data = [\n *   { id: 1, name: 'John' },\n *   { id: 2, name: 'Jane' }\n * ];\n * keyBy(data, 'id');\n * // Kết quả: {\n * //   '1': { id: 1, name: 'John' },\n * //   '2': { id: 2, name: 'Jane' }\n * // }\n */\nexport const keyBy = (data: Array<TYPE_OBJECT>, key: string): TYPE_OBJECT => {\n  if (!data || !data.length || !key) {\n    return {};\n  }\n  return data.reduce((dir, nextItem) => {\n    const valueOfKey = get(nextItem, key as PathOf<TYPE_OBJECT>);\n    if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {\n      return dir;\n    }\n    if (!Object.keys(dir).includes(`${valueOfKey}`)) {\n      dir[`${valueOfKey}`] = nextItem;\n    }\n    return dir;\n  }, {});\n};\n\n/**\n * Nhóm các đối tượng trong một mảng thành các nhóm dựa trên một thuộc tính cụ thể\n * @param data Mảng các đối tượng cần nhóm\n * @param key Tên thuộc tính được sử dụng làm khóa nhóm\n * @returns Đối tượng với các giá trị từ mảng được nhóm theo thuộc tính đã chỉ định\n * @example\n * const data = [\n *   { id: 1, name: 'John' },\n *   { id: 2, name: 'Jane' },\n *   { id: 1, name: 'John' }\n * ];\n * groupBy(data, 'id');\n * // Kết quả: {\n * //   '1': [\n * //     { id: 1, name: 'John' },\n * //     { id: 1, name: 'John' }\n * //   ],\n * //   '2': [\n * //     { id: 2, name: 'Jane' }\n * // }\n */\nexport const groupBy = (data: Array<TYPE_OBJECT>, key: string): TYPE_OBJECT => {\n  if (!data || !data.length || !Object.keys(get(data, '[0]')).length || !key) {\n    return {};\n  }\n  return data.reduce((dir, nextItem) => {\n    const valueOfKey = get(nextItem, key as PathOf<TYPE_OBJECT>);\n    if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {\n      return dir;\n    }\n    if (!Object.keys(dir).includes(`${valueOfKey}`)) {\n      dir[`${valueOfKey}`] = [];\n    }\n    dir[`${valueOfKey}`].push(nextItem);\n    return dir;\n  }, {});\n};\n\n/**\n * Tạo một mảng các số từ giá trị bắt đầu đến giá trị kết thúc với bước nhảy tùy chọn\n * @param start Giá trị bắt đầu của dãy số. Nếu chỉ có một tham số, đây sẽ là giá trị kết thúc và giá trị bắt đầu sẽ là 0\n * @param end Giá trị kết thúc của dãy số (tùy chọn)\n * @param step Bước nhảy giữa các số trong dãy (tùy chọn). Mặc định là 1 nếu end > start, -1 nếu end < start\n * @returns Mảng các số từ start đến end với bước nhảy step\n * @example\n * range(4);      // [0, 1, 2, 3]\n * range(1, 5);   // [1, 2, 3, 4]\n * range(0, 20, 5); // [0, 5, 10, 15]\n * range(5, 2);   // [5, 4, 3]\n */\nexport const range = (start: number, end?: number, step?: number): Array<number> => {\n  if (end === undefined || end === null) {\n    end = start;\n    start = 0;\n  }\n\n  if (!step) {\n    step = end < 0 ? -1 : 1;\n  }\n\n  if (end < start && step > 0) {\n    step *= -1;\n  }\n\n  const valueStartByStep = step + start;\n  const direction = start < end ? 'asc' : 'desc';\n  if ((direction === 'asc' && (valueStartByStep < start || valueStartByStep > end)) || (direction === 'desc' && (valueStartByStep > start || valueStartByStep < end))) {\n    return [start];\n  }\n\n  const arr = new Array<number>();\n\n  for (let index = 0; index < Math.abs(end - start); index++) {\n    let value = start + index * step;\n    if (index === 0) {\n      value = start;\n    }\n\n    if ((direction === 'asc' && (value < start || value > end)) || (direction === 'desc' && (value > start || value < end))) {\n      return arr;\n    }\n\n    arr.push(value);\n  }\n\n  return arr;\n};\n\n/**\n * So sánh hai giá trị bất kỳ có bằng nhau hay không\n * @param value1 Giá trị thứ nhất cần so sánh\n * @param value2 Giá trị thứ hai cần so sánh\n * @param exactlyPosition Có so sánh chính xác vị trí các phần tử trong mảng hay không\n * @returns true nếu hai giá trị bằng nhau, false nếu không bằng nhau\n * @example\n * isEqual([1,2,3], [1,2,3]); // true\n * isEqual([1,2,3], [3,2,1]); // true khi exactlyPosition = false\n * isEqual([1,2,3], [3,2,1]); // false khi exactlyPosition = true\n * isEqual({a:1}, {a:1}); // true\n */\nexport const isEqual = (\n  value1: any,\n  value2: any,\n  options?: {\n    exactlyPosition?: boolean;\n    ignoreExactlyDataType?: boolean;\n    ignoreUnWrapSignal?: boolean;\n  }\n): boolean => {\n  // Handle signals\n  if (!options?.ignoreUnWrapSignal) {\n    value1 = unwrapSignal(value1);\n    value2 = unwrapSignal(value2);\n  }\n  const { exactlyPosition = false, ignoreExactlyDataType = false } = options || {};\n\n  if (value1 === value2 || (value1 === null && value2 === null) || (value1 === undefined && value2 === undefined)) {\n    return true;\n  }\n\n  if (ignoreExactlyDataType) {\n    return isEqual(isNil(value1) ? undefined : `${value1}`, isNil(value2) ? undefined : `${value2}`);\n  }\n\n  if (typeof value1 !== 'object' || typeof value2 !== 'object' || (Array.isArray(value1) && !Array.isArray(value2)) || (!Array.isArray(value1) && Array.isArray(value2))) {\n    return false;\n  }\n\n  if (Array.isArray(value1) || Array.isArray(value2)) {\n    if (value1.length !== value2.length) {\n      return false;\n    }\n    if (!exactlyPosition) {\n      return !value1.some((item: any) => !value2.includes(item));\n    }\n\n    return !value1.some((item: any, index: number) => !isEqual(item, value2[index], options));\n  }\n\n  if (Object.keys(value1).length !== Object.keys(value2).length) {\n    return false;\n  }\n  return !Object.keys(value1).some((key) => !isEqual(value1[key], value2[key], options));\n};\n\n/**\n * Loại bỏ các phần tử trùng lặp trong mảng dựa trên một thuộc tính chỉ định\n * @param data Mảng dữ liệu cần xử lý\n * @param key Tên thuộc tính dùng để so sánh trùng lặp. Nếu không có key thì so sánh trực tiếp giá trị\n * @returns Mảng mới chứa các phần tử không trùng lặp\n * @example\n * const arr = [\n *   { id: 1, name: 'A' },\n *   { id: 2, name: 'B' },\n *   { id: 1, name: 'C' }\n * ];\n * uniqBy(arr, 'id'); // [{ id: 1, name: 'A' }, { id: 2, name: 'B' }]\n *\n * const numbers = [1, 2, 2, 3, 3];\n * uniqBy(numbers); // [1, 2, 3]\n *\n * const numbersSignal = [signal(1), signal(2), signal(3), signal(2), signal(5), signal(4), signal(1), signal(6), signal(7), signal(6)];\n * uniqBy(numbersSignal); // [signal(1), signal(2), signal(3), signal(5), signal(4), signal(6), signal(7)]\n */\nexport const uniqBy = <T>(data: Array<T>, key?: string): Array<T> => {\n  if (!key || !data?.length || typeof get(data, '[0]') !== 'object') {\n    // Xử lý mảng chứa signal values\n    if (data[0] && isSignal(data[0])) {\n      const seen = new Set<string>();\n\n      return data.filter((item) => {\n        const value = `${get<any>(item, '')}`;\n        if (seen.has(value)) {\n          return false;\n        }\n        seen.add(value);\n        return true;\n      });\n    }\n    // Xử lý mảng primitive values\n    return Array.from(new Set(data));\n  }\n\n  const dataUnique = keyBy(data as Array<Record<string, any>>, key);\n\n  return Object.keys(dataUnique).map((key) => dataUnique[key]);\n};\n\nexport const generateInterface = (obj: any, interfaceName: string): string => {\n  const generateType = (value: any): string => {\n    if (value === null) {\n      return 'null';\n    }\n    const type = typeof value;\n\n    if (type === 'string') {\n      return 'string';\n    }\n    if (type === 'number') {\n      return 'number';\n    }\n    if (type === 'boolean') {\n      return 'boolean';\n    }\n    if (type === 'undefined') {\n      return 'any';\n    }\n\n    if (value instanceof Date) {\n      return 'Date';\n    }\n\n    if (value instanceof RegExp) {\n      return 'RegExp';\n    }\n\n    if (Array.isArray(value)) {\n      if (value.length === 0) {\n        return 'Array<any>';\n      }\n      return `Array<${generateType(value[0])}>`;\n    }\n\n    if (type === 'object') {\n      let interfaceStr = '{\\n';\n      for (const key in value) {\n        if (Object.prototype.hasOwnProperty.call(value, key)) {\n          const valueType = generateType(value[key]);\n          interfaceStr += `  ${key}: ${valueType};\\n`;\n        }\n      }\n      interfaceStr += '}';\n      return interfaceStr;\n    }\n\n    return 'any';\n  };\n\n  const interfaceStr = `interface ${interfaceName} ${generateType(obj)}`;\n  return interfaceStr;\n};\n"]}