@libs-ui/utils 0.2.29 → 0.2.30-6.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 (59) hide show
  1. package/base64.d.ts +5 -0
  2. package/cache.d.ts +16 -15
  3. package/color.d.ts +1 -1
  4. package/communicate-micro.d.ts +4 -5
  5. package/constants.d.ts +3 -0
  6. package/crypto-3rd.d.ts +4 -0
  7. package/crypto.d.ts +4 -0
  8. package/date.d.ts +49 -0
  9. package/dom.d.ts +36 -1
  10. package/download.d.ts +3 -0
  11. package/esm2022/base64.mjs +43 -0
  12. package/esm2022/cache.mjs +121 -132
  13. package/esm2022/color.mjs +2 -2
  14. package/esm2022/communicate-micro.mjs +26 -24
  15. package/esm2022/constants.mjs +4 -1
  16. package/esm2022/crypto-3rd.mjs +5 -1
  17. package/esm2022/crypto.mjs +5 -1
  18. package/esm2022/date.mjs +189 -0
  19. package/esm2022/dom.mjs +200 -22
  20. package/esm2022/download.mjs +37 -0
  21. package/esm2022/file.mjs +79 -0
  22. package/esm2022/format-text.mjs +150 -0
  23. package/esm2022/get-smart-axis-scale.mjs +174 -0
  24. package/esm2022/helpers.mjs +402 -91
  25. package/esm2022/http-params.mjs +15 -3
  26. package/esm2022/index.mjs +12 -5
  27. package/esm2022/inject-token.mjs +5 -0
  28. package/esm2022/key-cache.mjs +20 -8
  29. package/esm2022/language.mjs +67 -4
  30. package/esm2022/pattern.mjs +21 -21
  31. package/esm2022/random.mjs +42 -0
  32. package/esm2022/two-way-signal-object.mjs +113 -0
  33. package/esm2022/uri.mjs +22 -0
  34. package/esm2022/uuid.mjs +3 -2
  35. package/esm2022/xss-filter.mjs +10 -0
  36. package/fesm2022/libs-ui-utils.mjs +2238 -880
  37. package/fesm2022/libs-ui-utils.mjs.map +1 -1
  38. package/file.d.ts +18 -0
  39. package/format-text.d.ts +11 -0
  40. package/get-smart-axis-scale.d.ts +34 -0
  41. package/helpers.d.ts +218 -10
  42. package/http-params.d.ts +2 -2
  43. package/index.d.ts +11 -4
  44. package/inject-token.d.ts +4 -0
  45. package/language.d.ts +32 -0
  46. package/package.json +7 -4
  47. package/pattern.d.ts +20 -20
  48. package/random.d.ts +3 -0
  49. package/two-way-signal-object.d.ts +14 -0
  50. package/uri.d.ts +5 -0
  51. package/xss-filter.d.ts +3 -0
  52. package/delete-unicode.d.ts +0 -1
  53. package/escape-html.d.ts +0 -1
  54. package/esm2022/delete-unicode.mjs +0 -20
  55. package/esm2022/escape-html.mjs +0 -12
  56. package/esm2022/get-color-by-id.mjs +0 -17
  57. package/esm2022/remove-emoji.mjs +0 -10
  58. package/get-color-by-id.d.ts +0 -1
  59. package/remove-emoji.d.ts +0 -2
@@ -1,160 +1,411 @@
1
1
  /* eslint-disable @typescript-eslint/no-explicit-any */
2
- import { ElementRef, TemplateRef, signal } from "@angular/core";
2
+ import { HttpParams } from '@angular/common/http';
3
+ import { ElementRef, TemplateRef, isSignal, signal } from '@angular/core';
4
+ import { Observable } from 'rxjs';
5
+ import dayjs from 'dayjs';
6
+ import { UtilsHttpParamsRequest } from './http-params';
7
+ import { getDayjs } from './date';
8
+ /**Các hàm tương tự thư viện lodash */
9
+ /**
10
+ * Kiểm tra xem một giá trị có phải là null hoặc undefined hay không
11
+ * @param value Giá trị cần kiểm tra
12
+ * @returns true nếu giá trị là null hoặc undefined, false nếu không
13
+ * @example
14
+ * isNil(null); // true
15
+ * isNil(undefined); // true
16
+ * isNil(0); // false
17
+ * isNil('hello'); // false
18
+ */
3
19
  export const isNil = (value) => {
4
20
  return value === null || value === undefined;
5
21
  };
22
+ /**
23
+ * Kiểm tra xem một giá trị có phải là rỗng hay không
24
+ * @param value Giá trị cần kiểm tra
25
+ * @returns true nếu giá trị là null, rỗng hoặc undefined, false nếu không
26
+ * @example
27
+ * isEmpty(null); // true
28
+ * isEmpty(''); // true
29
+ * isEmpty(undefined); // true
30
+ * isEmpty({}); // true
31
+ * isEmpty([]); // true
32
+ * isEmpty([1, 2, 3]); // false
33
+ * isEmpty({ a: 1 }); // false
34
+ */
6
35
  export const isEmpty = (value) => {
36
+ while (isSignal(value)) {
37
+ value = value();
38
+ }
7
39
  return value === null || value === '' || value === undefined || (typeof value === 'object' && (JSON.stringify(value) === '{}' || JSON.stringify(value) === '[]'));
8
40
  };
9
- export const get = (obj, path, defaultValue = undefined) => {
10
- if (obj == null) {
41
+ /**
42
+ * 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
43
+ * @param objData Đối tượng cần xử lý
44
+ * @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ỏ
45
+ * @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
46
+ * @example
47
+ * const obj = { a: 1, b: null, c: 3, d: undefined };
48
+ * omitBy(obj, isNil); // { a: 1, c: 3 }
49
+ */
50
+ export const omitBy = (objData, predicate) => {
51
+ if (!objData || typeof objData !== 'object' || Array.isArray(objData)) {
52
+ return objData;
53
+ }
54
+ const newObj = {};
55
+ Object.keys(objData).forEach((key) => {
56
+ const valueOfKey = get(objData, key);
57
+ if (!predicate(valueOfKey)) {
58
+ set(newObj, key, valueOfKey);
59
+ }
60
+ });
61
+ return newObj;
62
+ };
63
+ /**
64
+ * Lấy giá trị từ đối tượng theo đường dẫn chỉ định
65
+ *
66
+ * 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,
67
+ * tránh lỗi khi thuộc tính không tồn tại.
68
+ *
69
+ * @param obj Đối tượng nguồn cần lấy giá trị
70
+ * @param path Đường dẫn đến thuộc tính cần lấy. Có thể là:
71
+ * - Chuỗi: 'user.profile.name' hoặc 'items[0].title'
72
+ * - Mảng: ['user', 'profile', 'name'] hoặc ['items', '0', 'title']
73
+ * - Chuỗi rỗng '': trả về chính đối tượng gốc
74
+ * @param defaultValue Giá trị mặc định trả về khi không tìm thấy thuộc tính (mặc định: undefined)
75
+ * @param keepLastValueIfSignal Có giữ nguyên signal cuối cùng hay không (mặc định: false - sẽ gọi signal())
76
+ * @returns Giá trị tìm được hoặc giá trị mặc định
77
+ *
78
+ * @example
79
+ * // Ví dụ cơ bản
80
+ * const user = { name: 'John', age: 30 };
81
+ * get(user, 'name'); // 'John'
82
+ * get(user, 'email'); // undefined
83
+ * get(user, 'email', 'no-email'); // 'no-email'
84
+ *
85
+ * @example
86
+ * // Truyền path rỗng - trả về chính đối tượng gốc
87
+ * const data = { name: 'Alice', age: 25 };
88
+ * get(data, ''); // { name: 'Alice', age: 25 } (chính đối tượng data)
89
+ * get(data, '', 'default'); // { name: 'Alice', age: 25 } (bỏ qua defaultValue)
90
+ *
91
+ * @example
92
+ * // Truy cập thuộc tính sâu
93
+ * const data = {
94
+ * user: {
95
+ * profile: {
96
+ * name: 'Alice',
97
+ * settings: { theme: 'dark' }
98
+ * }
99
+ * }
100
+ * };
101
+ * get(data, 'user.profile.name'); // 'Alice'
102
+ * get(data, 'user.profile.settings.theme'); // 'dark'
103
+ * get(data, 'user.profile.avatar', 'default.jpg'); // 'default.jpg'
104
+ *
105
+ * @example
106
+ * // Truy cập mảng
107
+ * const items = [
108
+ * { name: 'Item 1', price: 100 },
109
+ * { name: 'Item 2', price: 200 }
110
+ * ];
111
+ * get(items, '[0].name'); // 'Item 1'
112
+ * get(items, '[1].name'); // 'Item 2'
113
+ * get(items, '[2].name', 'Not found'); // 'Not found'
114
+ *
115
+ * @example
116
+ * // Sử dụng với mảng path
117
+ * const nested = { a: { b: { c: 'deep value' } } };
118
+ * get(nested, ['a', 'b', 'c']); // 'deep value'
119
+ * get(nested, ['a', 'b', 'd'], 'default'); // 'default'
120
+ *
121
+ * @example
122
+ * // Trường hợp đặc biệt
123
+ * get(null, 'any.path'); // undefined
124
+ * get(undefined, 'any.path', 'fallback'); // 'fallback'
125
+ */
126
+ export const get = (obj, path, defaultValue, keepLastValueIfSignal) => {
127
+ if (isNil(obj)) {
11
128
  return defaultValue;
12
129
  }
13
- const paths = Array.isArray(path) ? path : path.replace(/\[(\d+)]/g, '.$1').split('.');
14
- for (let i = 0; i < paths.length; i++) {
15
- if (obj == null || !Object.prototype.hasOwnProperty.call(obj, paths[i])) {
130
+ while (isSignal(obj)) {
131
+ obj = obj();
132
+ }
133
+ if (path === '') {
134
+ return obj;
135
+ }
136
+ if (obj instanceof HttpParams) {
137
+ return (obj.get(`${path}`) ?? defaultValue);
138
+ }
139
+ if (obj instanceof DOMRect) {
140
+ return obj[path];
141
+ }
142
+ const paths = Array.isArray(path)
143
+ ? path
144
+ : path
145
+ .replace(/\[(\d+)]/g, '.$1')
146
+ .split('.')
147
+ .filter((key) => key);
148
+ for (const index in paths) {
149
+ const key = paths[index];
150
+ if (obj instanceof CSSStyleDeclaration) {
151
+ obj = obj[key];
152
+ continue;
153
+ }
154
+ if (obj instanceof HTMLElement) {
155
+ obj = obj[key];
156
+ continue;
157
+ }
158
+ if (isNil(obj) || !Object.prototype.hasOwnProperty.call(obj, key)) {
16
159
  return defaultValue;
17
160
  }
18
- obj = obj[paths[i]];
161
+ const val = isSignal(obj[key]) && !keepLastValueIfSignal ? obj[key]() : obj[key];
162
+ obj = val;
19
163
  }
20
164
  return obj;
21
165
  };
166
+ /**
167
+ * Thiết lập giá trị cho một thuộc tính trong đối tượng theo đường dẫn chỉ định
168
+ * @param obj Đối tượng cần thiết lập giá trị
169
+ * @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'])
170
+ * @param value Giá trị cần thiết lập
171
+ * @returns Đối tượng sau khi đã thiết lập giá trị
172
+ * @throws Error nếu tham số đầu tiên không phải là đối tượng
173
+ * @example
174
+ * const obj = { a: { b: 1 } };
175
+ * set(obj, 'a.b', 2); // { a: { b: 2 } }
176
+ */
22
177
  export const set = (obj, path, value) => {
23
- if (!obj || typeof obj !== "object") {
24
- throw new Error("The first argument must be an object");
178
+ if (!obj || (typeof obj !== 'object' && !isSignal(obj)) || (isSignal(obj) && typeof obj() !== 'object')) {
179
+ throw new Error('The first argument must be an object');
180
+ }
181
+ if (obj instanceof HttpParams) {
182
+ return obj.set(`${path}`, value);
25
183
  }
26
- const pathArray = Array.isArray(path) ? path : path.replace(/\[(\d+)]/g, '.[$1]').split('.');
27
- let current = obj;
184
+ const pathArray = Array.isArray(path)
185
+ ? path
186
+ : path
187
+ .replace(/\[(\d+)]/g, '.[$1]')
188
+ .split('.')
189
+ .filter((key) => key);
190
+ let currentObjectByKey = isSignal(obj) ? obj() : obj;
191
+ let preObjectByKey = obj;
28
192
  pathArray.forEach((key, index) => {
29
193
  if (index < pathArray.length - 1) {
30
- if (!(key in current) || typeof current[key] !== "object") {
194
+ if (!(key in currentObjectByKey) || (typeof currentObjectByKey[key] !== 'object' && !isSignal(currentObjectByKey[key]))) {
31
195
  const nextKey = pathArray[index + 1];
32
196
  key = key.replace(/\[(\d+)]/g, '$1');
33
- current[key] = /\[(\d+)]/g.test(nextKey) ? [] : {};
197
+ currentObjectByKey[key] = /\[(\d+)]/g.test(nextKey) ? [] : {};
34
198
  }
35
- current = current[key];
199
+ currentObjectByKey = key ? currentObjectByKey[key] : currentObjectByKey;
200
+ preObjectByKey = currentObjectByKey;
201
+ currentObjectByKey = isSignal(currentObjectByKey) ? currentObjectByKey() : currentObjectByKey;
202
+ return;
203
+ }
204
+ if (typeof currentObjectByKey !== 'object') {
36
205
  return;
37
206
  }
38
207
  // Gán giá trị ở cuối đường dẫn
39
- current[key] = value;
208
+ key = key.replace(/\[(\d+)]/g, '$1');
209
+ const valueOfKey = currentObjectByKey[key];
210
+ const valueOfKeyIsSignal = isSignal(valueOfKey);
211
+ if (valueOfKeyIsSignal && (typeof valueOfKey() !== 'object' || valueOfKey() === null)) {
212
+ valueOfKey.set(isSignal(value) ? value() : value);
213
+ return;
214
+ }
215
+ if (isSignal(preObjectByKey)) {
216
+ preObjectByKey.update((data) => {
217
+ const dataOfKey = data[key];
218
+ if (isNil(dataOfKey) || !valueOfKeyIsSignal) {
219
+ data[key] = value;
220
+ }
221
+ if (valueOfKeyIsSignal) {
222
+ valueOfKey.set(isSignal(value) ? value() : value);
223
+ }
224
+ if (Array.isArray(data)) {
225
+ return [...data];
226
+ }
227
+ return { ...data };
228
+ });
229
+ return;
230
+ }
231
+ if (valueOfKeyIsSignal) {
232
+ valueOfKey.set(isSignal(value) ? value() : value);
233
+ return;
234
+ }
235
+ currentObjectByKey[key] = value;
40
236
  });
41
237
  return obj;
42
238
  };
43
- export const cloneDeep = (value) => {
44
- if (value === null || typeof value !== 'object') {
45
- return value;
239
+ /**
240
+ * Tạo một bản sao sâu của một đối tượng hoặc giá trị bất kỳ
241
+ * @param data Dữ liệu cần sao chép
242
+ * @param options Tùy chọn cấu hình
243
+ * @param options.ignoreSignal Nếu true, sẽ không sao chép các signal mà trả về signal gốc
244
+ * @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
245
+ * @returns Bản sao sâu của dữ liệu đầu vào
246
+ * @example
247
+ * const obj = {
248
+ * a: 1,
249
+ * b: { c: 2 },
250
+ * d: [1, 2, 3]
251
+ * };
252
+ * const clone = cloneDeep(obj);
253
+ * // clone là một bản sao hoàn toàn độc lập của obj
254
+ */
255
+ export const cloneDeep = (data, options = { ignoreSignal: false }, seen = new WeakMap()) => {
256
+ if (data === null || (typeof data !== 'object' && !isSignal(data))) {
257
+ return data;
258
+ }
259
+ if (seen.has(data)) {
260
+ return seen.get(data);
261
+ }
262
+ if (data instanceof HttpParams) {
263
+ return new UtilsHttpParamsRequest(undefined, data);
46
264
  }
47
- if (value instanceof Date) {
48
- return new Date(value.getTime());
265
+ if (data instanceof Date) {
266
+ return new Date(data.getTime());
49
267
  }
50
- if (value instanceof RegExp) {
51
- return new RegExp(value.source, value.flags);
268
+ if (dayjs.isDayjs(data)) {
269
+ return getDayjs({ date: data.valueOf() });
52
270
  }
53
- if (value instanceof Map) {
271
+ if (data instanceof RegExp) {
272
+ return new RegExp(data.source, data.flags);
273
+ }
274
+ if (data instanceof Map) {
54
275
  const mapCopy = new Map();
55
- value.forEach((val, key) => {
56
- mapCopy.set(cloneDeep(key), cloneDeep(val));
276
+ seen.set(data, mapCopy);
277
+ data.forEach((val, key) => {
278
+ mapCopy.set(cloneDeep(key, options, seen), cloneDeep(val, options, seen));
57
279
  });
58
280
  return mapCopy;
59
281
  }
60
- if (value instanceof Set) {
282
+ if (data instanceof Set) {
61
283
  const setCopy = new Set();
62
- value.forEach(val => {
63
- setCopy.add(cloneDeep(val));
284
+ seen.set(data, setCopy);
285
+ data.forEach((val) => {
286
+ setCopy.add(cloneDeep(val, options, seen));
64
287
  });
65
288
  return setCopy;
66
289
  }
67
- if (Array.isArray(value)) {
68
- return value.map(item => cloneDeep(item));
69
- }
70
- const result = {};
71
- for (const key in value) {
72
- if (value[key] instanceof TemplateRef || value[key] instanceof ElementRef || value[key] instanceof Element) {
73
- result[key] = value[key];
74
- continue;
75
- }
76
- if (Object.prototype.hasOwnProperty.call(value, key)) {
77
- result[key] = cloneDeep(value[key]);
78
- }
79
- }
80
- return result;
81
- };
82
- export const convertObjectToSignal = (data) => {
83
- if (data === null || typeof data !== 'object') {
84
- return data;
85
- }
86
290
  if (Array.isArray(data)) {
87
- if (typeof data[0] !== 'object') {
88
- return signal(data);
89
- }
90
- return signal(data.map(item => convertObjectToSignal(cloneDeep(item))));
291
+ seen.set(data, data.map((item) => cloneDeep(item, options, seen)));
292
+ return seen.get(data);
91
293
  }
92
- data = signal(data);
93
- for (const key in data()) {
94
- const value = data()[key];
95
- if (value instanceof TemplateRef || value instanceof ElementRef || value instanceof Element || value instanceof Date || value instanceof RegExp || value instanceof Set || value instanceof Map) {
96
- continue;
97
- }
98
- if (Object.prototype.hasOwnProperty.call(data(), key)) {
99
- data()[key] = convertObjectToSignal(cloneDeep(value));
100
- }
294
+ if (data instanceof File || data instanceof Blob || Object.prototype.toString.call(data) === '[object File]') {
295
+ return data;
101
296
  }
102
- return data;
103
- };
104
- export const isSignal = (data) => {
105
- return typeof data === 'function' && data.toString().includes('Signal:');
106
- };
107
- export const convertSignalToObject = (data) => {
108
- if (data === null || !isSignal(data)) {
297
+ if (data instanceof TemplateRef || data instanceof ElementRef || data instanceof Element || data instanceof Promise || data instanceof Observable) {
109
298
  return data;
110
299
  }
111
- data = data();
112
- if (Array.isArray(data)) {
113
- if (!isSignal(data[0])) {
300
+ if (isSignal(data)) {
301
+ if (options?.ignoreSignal) {
114
302
  return data;
115
303
  }
116
- return data.map(item => convertSignalToObject(cloneDeep(item)));
304
+ seen.set(data, signal(cloneDeep(data(), options, seen)));
305
+ return seen.get(data);
117
306
  }
307
+ const result = {};
308
+ seen.set(data, result);
118
309
  for (const key in data) {
119
310
  const value = data[key];
120
- if (!isSignal(value)) {
311
+ if (value instanceof HttpParams) {
312
+ result[key] = new UtilsHttpParamsRequest(undefined, value);
313
+ continue;
314
+ }
315
+ if (dayjs.isDayjs(value)) {
316
+ result[key] = getDayjs({ date: value.valueOf() });
317
+ continue;
318
+ }
319
+ if (value instanceof TemplateRef || value instanceof ElementRef || value instanceof Element || value instanceof Promise || value instanceof Observable) {
320
+ result[key] = value;
121
321
  continue;
122
322
  }
123
323
  if (Object.prototype.hasOwnProperty.call(data, key)) {
124
- data[key] = convertSignalToObject(cloneDeep(value));
324
+ result[key] = cloneDeep(value, options, seen);
125
325
  }
126
326
  }
127
- return data;
327
+ return result;
128
328
  };
329
+ /**
330
+ * 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
331
+ * @param data Mảng các đối tượng cần chuyển đổi
332
+ * @param key Tên thuộc tính được sử dụng làm khóa trong đối tượng kết quả
333
+ * @returns Đối tượng với các giá trị từ mảng được đánh key theo thuộc tính đã chỉ định
334
+ * @example
335
+ * const data = [
336
+ * { id: 1, name: 'John' },
337
+ * { id: 2, name: 'Jane' }
338
+ * ];
339
+ * keyBy(data, 'id');
340
+ * // Kết quả: {
341
+ * // '1': { id: 1, name: 'John' },
342
+ * // '2': { id: 2, name: 'Jane' }
343
+ * // }
344
+ */
129
345
  export const keyBy = (data, key) => {
130
346
  if (!data || !data.length || !key) {
131
347
  return {};
132
348
  }
133
349
  return data.reduce((dir, nextItem) => {
134
- const valueOfKey = nextItem[key];
135
- if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number') {
350
+ const valueOfKey = get(nextItem, key);
351
+ if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {
136
352
  return dir;
137
353
  }
138
- dir[valueOfKey] = nextItem;
354
+ if (!Object.keys(dir).includes(`${valueOfKey}`)) {
355
+ dir[`${valueOfKey}`] = nextItem;
356
+ }
139
357
  return dir;
140
358
  }, {});
141
359
  };
360
+ /**
361
+ * 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ể
362
+ * @param data Mảng các đối tượng cần nhóm
363
+ * @param key Tên thuộc tính được sử dụng làm khóa nhóm
364
+ * @returns Đối tượng với các giá trị từ mảng được nhóm theo thuộc tính đã chỉ định
365
+ * @example
366
+ * const data = [
367
+ * { id: 1, name: 'John' },
368
+ * { id: 2, name: 'Jane' },
369
+ * { id: 1, name: 'John' }
370
+ * ];
371
+ * groupBy(data, 'id');
372
+ * // Kết quả: {
373
+ * // '1': [
374
+ * // { id: 1, name: 'John' },
375
+ * // { id: 1, name: 'John' }
376
+ * // ],
377
+ * // '2': [
378
+ * // { id: 2, name: 'Jane' }
379
+ * // }
380
+ */
142
381
  export const groupBy = (data, key) => {
143
- if (!data || !Object.keys(data).length || !key) {
382
+ if (!data || !data.length || !Object.keys(get(data, '0')).length || !key) {
144
383
  return {};
145
384
  }
146
385
  return data.reduce((dir, nextItem) => {
147
- const valueOfKey = nextItem[key];
148
- if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number') {
386
+ const valueOfKey = get(nextItem, key);
387
+ if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number' && typeof valueOfKey !== 'boolean') {
149
388
  return dir;
150
389
  }
151
- if (!dir[valueOfKey]) {
152
- dir[valueOfKey] = [];
390
+ if (!Object.keys(dir).includes(`${valueOfKey}`)) {
391
+ dir[`${valueOfKey}`] = [];
153
392
  }
154
- dir[valueOfKey].push(nextItem);
393
+ dir[`${valueOfKey}`].push(nextItem);
155
394
  return dir;
156
395
  }, {});
157
396
  };
397
+ /**
398
+ * 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
399
+ * @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
400
+ * @param end Giá trị kết thúc của dãy số (tùy chọn)
401
+ * @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
402
+ * @returns Mảng các số từ start đến end với bước nhảy step
403
+ * @example
404
+ * range(4); // [0, 1, 2, 3]
405
+ * range(1, 5); // [1, 2, 3, 4]
406
+ * range(0, 20, 5); // [0, 5, 10, 15]
407
+ * range(5, 2); // [5, 4, 3]
408
+ */
158
409
  export const range = (start, end, step) => {
159
410
  if (end === undefined || end === null) {
160
411
  end = start;
@@ -163,6 +414,9 @@ export const range = (start, end, step) => {
163
414
  if (!step) {
164
415
  step = end < 0 ? -1 : 1;
165
416
  }
417
+ if (end < start && step > 0) {
418
+ step *= -1;
419
+ }
166
420
  const valueStartByStep = step + start;
167
421
  const direction = start < end ? 'asc' : 'desc';
168
422
  if ((direction === 'asc' && (valueStartByStep < start || valueStartByStep > end)) || (direction === 'desc' && (valueStartByStep > start || valueStartByStep < end))) {
@@ -181,31 +435,88 @@ export const range = (start, end, step) => {
181
435
  }
182
436
  return arr;
183
437
  };
184
- export const isEqual = (value1, value2) => {
438
+ /**
439
+ * So sánh hai giá trị bất kỳ có bằng nhau hay không
440
+ * @param value1 Giá trị thứ nhất cần so sánh
441
+ * @param value2 Giá trị thứ hai cần so sánh
442
+ * @param exactlyPosition Có so sánh chính xác vị trí các phần tử trong mảng hay không
443
+ * @returns true nếu hai giá trị bằng nhau, false nếu không bằng nhau
444
+ * @example
445
+ * isEqual([1,2,3], [1,2,3]); // true
446
+ * isEqual([1,2,3], [3,2,1]); // true khi exactlyPosition = false
447
+ * isEqual([1,2,3], [3,2,1]); // false khi exactlyPosition = true
448
+ * isEqual({a:1}, {a:1}); // true
449
+ */
450
+ export const isEqual = (value1, value2, options) => {
451
+ const { exactlyPosition = false, ignoreExactlyDataType = false } = options || {};
185
452
  if (value1 === value2 || (value1 === null && value2 === null) || (value1 === undefined && value2 === undefined)) {
186
453
  return true;
187
454
  }
188
- if (typeof value1 !== 'object' || typeof value2 !== 'object' || (Array.isArray(value1) && !Array.isArray(value2))) {
455
+ if (ignoreExactlyDataType) {
456
+ return isEqual(isNil(value1) ? undefined : `${value1}`, isNil(value2) ? undefined : `${value2}`);
457
+ }
458
+ // Handle signals
459
+ while (isSignal(value1)) {
460
+ value1 = value1();
461
+ }
462
+ while (isSignal(value2)) {
463
+ value2 = value2();
464
+ }
465
+ if (typeof value1 !== 'object' || typeof value2 !== 'object' || (Array.isArray(value1) && !Array.isArray(value2)) || (!Array.isArray(value1) && Array.isArray(value2))) {
189
466
  return false;
190
467
  }
191
468
  if (Array.isArray(value1)) {
192
469
  if (value1.length !== value2.length) {
193
470
  return false;
194
471
  }
195
- return !value1.some((item, index) => !isEqual(item, value2[index]));
472
+ if (!exactlyPosition) {
473
+ return !value1.some((item) => !value2.includes(item));
474
+ }
475
+ return !value1.some((item, index) => !isEqual(item, value2[index], options));
196
476
  }
197
477
  if (Object.keys(value1).length !== Object.keys(value2).length) {
198
478
  return false;
199
479
  }
200
- return !Object.keys(value1).some((key) => !isEqual(value1[key], value2[key]));
480
+ return !Object.keys(value1).some((key) => !isEqual(value1[key], value2[key], options));
201
481
  };
482
+ /**
483
+ * 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
484
+ * @param data Mảng dữ liệu cần xử lý
485
+ * @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ị
486
+ * @returns Mảng mới chứa các phần tử không trùng lặp
487
+ * @example
488
+ * const arr = [
489
+ * { id: 1, name: 'A' },
490
+ * { id: 2, name: 'B' },
491
+ * { id: 1, name: 'C' }
492
+ * ];
493
+ * uniqBy(arr, 'id'); // [{ id: 1, name: 'A' }, { id: 2, name: 'B' }]
494
+ *
495
+ * const numbers = [1, 2, 2, 3, 3];
496
+ * uniqBy(numbers); // [1, 2, 3]
497
+ *
498
+ * const numbersSignal = [signal(1), signal(2), signal(3), signal(2), signal(5), signal(4), signal(1), signal(6), signal(7), signal(6)];
499
+ * uniqBy(numbersSignal); // [signal(1), signal(2), signal(3), signal(5), signal(4), signal(6), signal(7)]
500
+ */
202
501
  export const uniqBy = (data, key) => {
203
- if (!key || !data || !data.length || typeof data[0] !== 'object') {
204
- console.log('vao dauy r', !key, !data, !data.length, typeof data[0] !== 'object');
502
+ if (!key || !data?.length || typeof get(data, '0') !== 'object') {
503
+ // Xử mảng chứa signal values
504
+ if (data[0] && isSignal(data[0])) {
505
+ const seen = new Set();
506
+ return data.filter((item) => {
507
+ const value = `${get(item, '')}`;
508
+ if (seen.has(value)) {
509
+ return false;
510
+ }
511
+ seen.add(value);
512
+ return true;
513
+ });
514
+ }
515
+ // Xử lý mảng primitive values
205
516
  return Array.from(new Set(data));
206
517
  }
207
518
  const dataUnique = keyBy(data, key);
208
- return Object.keys(dataUnique).map(key => dataUnique[key]);
519
+ return Object.keys(dataUnique).map((key) => dataUnique[key]);
209
520
  };
210
521
  export const generateInterface = (obj, interfaceName) => {
211
522
  const generateType = (value) => {
@@ -253,4 +564,4 @@ export const generateInterface = (obj, interfaceName) => {
253
564
  const interfaceStr = `interface ${interfaceName} ${generateType(obj)}`;
254
565
  return interfaceStr;
255
566
  };
256
- //# 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 { ElementRef, TemplateRef, WritableSignal, signal } from \"@angular/core\";\n\nexport const isNil = (value: unknown): boolean => {\n  return value === null || value === undefined;\n};\n\nexport const isEmpty = (value: unknown) => {\n  return value === null || value === '' || value === undefined || (typeof value === 'object' && (JSON.stringify(value) === '{}' || JSON.stringify(value) === '[]'));\n};\n\nexport const get = (obj: Record<string, any> | any, path: string | string[], defaultValue: any = undefined): any => {\n  if (obj == null) {\n    return defaultValue;\n  }\n  const paths = Array.isArray(path) ? path : path.replace(/\\[(\\d+)]/g, '.$1').split('.');\n  for (let i = 0; i < paths.length; i++) {\n    if (obj == null || !Object.prototype.hasOwnProperty.call(obj, paths[i])) {\n      return defaultValue;\n    }\n    obj = obj[paths[i]];\n  }\n\n  return obj;\n};\n\nexport const set = (obj: Record<string, any>, path: string | string[], value: any): any => {\n  if (!obj || typeof obj !== \"object\") {\n    throw new Error(\"The first argument must be an object\");\n  }\n\n  const pathArray = Array.isArray(path) ? path : path.replace(/\\[(\\d+)]/g, '.[$1]').split('.');\n\n  let current: any = obj;\n\n  pathArray.forEach((key, index) => {\n    if (index < pathArray.length - 1) {\n      if (!(key in current) || typeof current[key] !== \"object\") {\n        const nextKey = pathArray[index + 1];\n\n        key = key.replace(/\\[(\\d+)]/g, '$1');\n        current[key] = /\\[(\\d+)]/g.test(nextKey) ? [] : {};\n      }\n      current = current[key];\n      return;\n    }\n    // Gán giá trị ở cuối đường dẫn\n    current[key] = value;\n  });\n\n  return obj;\n};\n\nexport const cloneDeep = (value: any): any => {\n  if (value === null || typeof value !== 'object') {\n    return value;\n  }\n\n  if (value instanceof Date) {\n    return new Date(value.getTime());\n  }\n\n  if (value instanceof RegExp) {\n    return new RegExp(value.source, value.flags);\n  }\n\n  if (value instanceof Map) {\n    const mapCopy = new Map();\n    value.forEach((val, key) => {\n      mapCopy.set(cloneDeep(key), cloneDeep(val));\n    });\n    return mapCopy;\n  }\n\n  if (value instanceof Set) {\n    const setCopy = new Set();\n    value.forEach(val => {\n      setCopy.add(cloneDeep(val));\n    });\n    return setCopy;\n  }\n\n  if (Array.isArray(value)) {\n    return value.map(item => cloneDeep(item));\n  }\n\n  const result: Record<string, any> = {};\n  for (const key in value) {\n    if (value[key] instanceof TemplateRef || value[key] instanceof ElementRef || value[key] instanceof Element) {\n      result[key] = value[key];\n      continue;\n    }\n    if (Object.prototype.hasOwnProperty.call(value, key)) {\n      result[key] = cloneDeep(value[key]);\n    }\n  }\n\n  return result;\n};\n\nexport const convertObjectToSignal = (data: any): any => {\n  if (data === null || typeof data !== 'object') {\n    return data;\n  }\n  if (Array.isArray(data)) {\n    if (typeof data[0] !== 'object') {\n      return signal(data);\n    }\n    return signal(data.map(item => convertObjectToSignal(cloneDeep(item))));\n  }\n\n  data = signal(data);\n\n  for (const key in data()) {\n    const value = data()[key];\n    if (value instanceof TemplateRef || value instanceof ElementRef || value instanceof Element || value instanceof Date || value instanceof RegExp || value instanceof Set || value instanceof Map) {\n      continue;\n    }\n    if (Object.prototype.hasOwnProperty.call(data(), key)) {\n      data()[key] = convertObjectToSignal(cloneDeep(value));\n    }\n  }\n\n  return data;\n};\n\nexport const isSignal = (data: any) => {\n  return typeof data === 'function' && data.toString().includes('Signal:');\n}\n\nexport const convertSignalToObject = (data: any): any => {\n  if (data === null || !isSignal(data)) {\n    return data;\n  }\n  data = (data as WritableSignal<unknown>)();\n  if (Array.isArray(data)) {\n    if (!isSignal(data[0])) {\n      return data;\n    }\n    return data.map(item => convertSignalToObject(cloneDeep(item)));\n  }\n\n  for (const key in data) {\n    const value = data[key];\n    if (!isSignal(value)) {\n      continue;\n    }\n    if (Object.prototype.hasOwnProperty.call(data, key)) {\n      data[key] = convertSignalToObject(cloneDeep(value));\n    }\n  }\n\n  return data;\n};\n\nexport const keyBy = (data: Array<Record<any, any>>, key: string): Record<any, any> => {\n  if (!data || !data.length || !key) {\n    return {};\n  }\n  return data.reduce((dir, nextItem) => {\n    const valueOfKey = nextItem[key];\n    if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number') {\n      return dir;\n    }\n    dir[valueOfKey] = nextItem;\n    return dir;\n  }, {});\n};\n\nexport const groupBy = (data: Array<Record<any, any>>, key: string): Record<any, any> => {\n  if (!data || !Object.keys(data).length || !key) {\n    return {};\n  }\n  return data.reduce((dir, nextItem) => {\n    const valueOfKey = nextItem[key];\n    if (typeof valueOfKey !== 'string' && typeof valueOfKey !== 'number') {\n      return dir;\n    }\n    if (!dir[valueOfKey]) {\n      dir[valueOfKey] = [];\n    }\n    dir[valueOfKey].push(nextItem);\n    return dir;\n  }, {});\n};\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  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\nexport const isEqual = (value1: any, value2: any): boolean => {\n  if (value1 === value2 || (value1 === null && value2 === null) || (value1 === undefined && value2 === undefined)) {\n    return true;\n  }\n  if (typeof value1 !== 'object' || typeof value2 !== 'object' || (Array.isArray(value1) && !Array.isArray(value2))) {\n    return false;\n  }\n\n  if (Array.isArray(value1)) {\n    if (value1.length !== value2.length) {\n      return false\n    }\n\n    return !value1.some((item, index) => !isEqual(item, value2[index]));\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]));\n}\n\nexport const uniqBy = <T>(data: Array<T>, key?: string): Array<T> => {\n  if (!key || !data || !data.length || typeof data[0] !== 'object') {\n    console.log('vao dauy r', !key, !data, !data.length, typeof data[0] !== 'object')\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};"]}
567
+ //# 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 { ElementRef, Signal, TemplateRef, WritableSignal, isSignal, signal } from '@angular/core';\nimport { GetValueByPath, PathsToStringProps, TYPE_OBJECT } from '@libs-ui/interfaces-types';\nimport { Observable } from 'rxjs';\nimport dayjs from 'dayjs';\nimport { UtilsHttpParamsRequest } from './http-params';\nimport { getDayjs } from './date';\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): value is null | undefined => {\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: unknown) => {\n  while (isSignal(value)) {\n    value = value();\n  }\n  return value === null || value === '' || value === undefined || (typeof value === 'object' && (JSON.stringify(value) === '{}' || JSON.stringify(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 PathsToStringProps<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 PathsToStringProps<O> = PathsToStringProps<O>, T extends GetValueByPath<O, P> = GetValueByPath<O, P>>(obj: Signal<O> | O, path: P, defaultValue?: T, keepLastValueIfSignal?: boolean): T => {\n  if (isNil(obj)) {\n    return defaultValue as T;\n  }\n  while (isSignal(obj)) {\n    obj = obj();\n  }\n  if (path === '') {\n    return obj as unknown as T;\n  }\n  if (obj instanceof HttpParams) {\n    return (obj.get(`${path}`) ?? defaultValue) as T;\n  }\n  if (obj instanceof DOMRect) {\n    return obj[path as keyof DOMRect] as T;\n  }\n\n  const paths = Array.isArray(path)\n    ? path\n    : path\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 defaultValue as T;\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 obj as unknown as T;\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 PathsToStringProps<O> = PathsToStringProps<O>, T extends GetValueByPath<O, P> = GetValueByPath<O, P>>(obj: Signal<O> | O, path: P, value: T): 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\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        currentObjectByKey[key] = /\\[(\\d+)]/g.test(nextKey) ? [] : {};\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))) {\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 (dayjs.isDayjs(data)) {\n    return getDayjs({ date: data.valueOf() }) as T;\n  }\n\n  if (data instanceof RegExp) {\n    return new RegExp(data.source, data.flags) as T;\n  }\n\n  if (data instanceof Map) {\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 (data instanceof Set) {\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 (Array.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 (data instanceof File || data instanceof Blob || Object.prototype.toString.call(data) === '[object File]') {\n    return data as T;\n  }\n  if (data instanceof TemplateRef || data instanceof ElementRef || data instanceof Element || data instanceof Promise || data instanceof Observable) {\n    return data as T;\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    if (dayjs.isDayjs(value)) {\n      result[key] = getDayjs({ date: value.valueOf() });\n      continue;\n    }\n    if (value instanceof TemplateRef || value instanceof ElementRef || value instanceof Element || value instanceof Promise || value instanceof Observable) {\n      result[key] = value;\n      continue;\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);\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);\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  }\n): boolean => {\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  // Handle signals\n  while (isSignal(value1)) {\n    value1 = value1();\n  }\n  while (isSignal(value2)) {\n    value2 = 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)) {\n    if (value1.length !== value2.length) {\n      return false;\n    }\n    if (!exactlyPosition) {\n      return !value1.some((item) => !value2.includes(item));\n    }\n\n    return !value1.some((item, index) => !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"]}