@libs-ui/utils 0.2.355-9 → 0.2.356-0

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 (52) hide show
  1. package/README.md +633 -2
  2. package/base64.d.ts +25 -0
  3. package/cache.d.ts +73 -0
  4. package/collection.d.ts +12 -0
  5. package/color.d.ts +37 -0
  6. package/communicate-micro.d.ts +34 -1
  7. package/crypto-3rd.d.ts +12 -0
  8. package/crypto.d.ts +17 -0
  9. package/dangerous-object.d.ts +8 -1
  10. package/data.d.ts +5 -3
  11. package/date.d.ts +12 -0
  12. package/dom.d.ts +56 -0
  13. package/download.d.ts +21 -0
  14. package/esm2022/base64.mjs +27 -2
  15. package/esm2022/cache.mjs +74 -1
  16. package/esm2022/collection.mjs +13 -1
  17. package/esm2022/color.mjs +38 -1
  18. package/esm2022/communicate-micro.mjs +37 -5
  19. package/esm2022/crypto-3rd.mjs +13 -1
  20. package/esm2022/crypto.mjs +18 -1
  21. package/esm2022/dangerous-object.mjs +11 -2
  22. package/esm2022/data.mjs +6 -4
  23. package/esm2022/date.mjs +13 -1
  24. package/esm2022/dom.mjs +57 -1
  25. package/esm2022/download.mjs +22 -1
  26. package/esm2022/file.mjs +54 -1
  27. package/esm2022/format-number.mjs +31 -3
  28. package/esm2022/format-text.mjs +73 -2
  29. package/esm2022/function-check-embed-frame.mjs +28 -1
  30. package/esm2022/pattern.mjs +83 -3
  31. package/esm2022/trace.mjs +8 -1
  32. package/esm2022/two-way-signal-object.mjs +14 -1
  33. package/esm2022/uri.mjs +22 -3
  34. package/esm2022/url-search-params.mjs +68 -1
  35. package/esm2022/url.mjs +10 -1
  36. package/esm2022/uuid.mjs +7 -1
  37. package/esm2022/xss-filter.mjs +14 -1
  38. package/fesm2022/libs-ui-utils.mjs +723 -21
  39. package/fesm2022/libs-ui-utils.mjs.map +1 -1
  40. package/file.d.ts +53 -0
  41. package/format-number.d.ts +29 -1
  42. package/format-text.d.ts +71 -0
  43. package/function-check-embed-frame.d.ts +22 -0
  44. package/package.json +2 -2
  45. package/pattern.d.ts +81 -1
  46. package/trace.d.ts +7 -0
  47. package/two-way-signal-object.d.ts +13 -0
  48. package/uri.d.ts +20 -0
  49. package/url-search-params.d.ts +67 -0
  50. package/url.d.ts +9 -0
  51. package/uuid.d.ts +6 -0
  52. package/xss-filter.d.ts +13 -0
package/esm2022/dom.mjs CHANGED
@@ -4,13 +4,24 @@ import { finalize, fromEvent, mergeMap, startWith, takeUntil, tap } from 'rxjs';
4
4
  import { set } from './helpers';
5
5
  let parser = null;
6
6
  let bowserParser = null;
7
+ /**
8
+ * Lấy thông tin thiết bị và trình duyệt (Sử dụng thư viện Bowser).
9
+ * @returns Đối tượng Parser chứa thông tin OS, Browser, Platform...
10
+ */
7
11
  export const getDeviceInfo = () => {
8
12
  if (!bowserParser) {
9
13
  bowserParser = Bowser.getParser(window.navigator.userAgent);
10
14
  }
11
15
  return bowserParser;
12
16
  };
17
+ /**
18
+ * Kiểm tra xem thiết bị hiện tại có hỗ trợ cảm ứng (Touch) không.
19
+ * @returns true nếu là thiết bị cảm ứng
20
+ */
13
21
  export const isTouchDevice = () => ('onpointerdown' in navigator && navigator.maxTouchPoints > 0) || 'ontouchstart' in window || (window.matchMedia?.('(pointer: coarse)').matches ?? false);
22
+ /**
23
+ * Xác định tên sự kiện click phù hợp cho thiết bị (click cho desktop, pointerdown cho touch).
24
+ */
14
25
  export const getEventNameHandleClick = (() => {
15
26
  if (isTouchDevice()) {
16
27
  return 'pointerdown';
@@ -18,12 +29,22 @@ export const getEventNameHandleClick = (() => {
18
29
  const deviceInfo = getDeviceInfo();
19
30
  return deviceInfo.isPlatform('desktop') ? 'click' : 'touchstart';
20
31
  })();
32
+ /**
33
+ * Chuyển đổi một chuỗi HTML thành đối tượng Document.
34
+ * @param htmlStr Chuỗi HTML cần parse
35
+ * @returns Đối tượng Document tương ứng
36
+ */
21
37
  export const getDocumentByString = (htmlStr) => {
22
38
  if (!parser) {
23
39
  parser = new DOMParser();
24
40
  }
25
41
  return parser.parseFromString(htmlStr, 'text/html');
26
42
  };
43
+ /**
44
+ * Sao chép các thuộc tính cơ bản của IBoundingClientRect.
45
+ * @param rect Đối tượng IBoundingClientRect gốc
46
+ * @returns Đối tượng IBoundingClientRect đã được copy
47
+ */
27
48
  export const cloneIBoundingClientRect = (rect) => {
28
49
  return {
29
50
  top: rect['top'],
@@ -33,13 +54,28 @@ export const cloneIBoundingClientRect = (rect) => {
33
54
  bottom: rect['bottom'],
34
55
  };
35
56
  };
57
+ /**
58
+ * Gán nhiều thuộc tính CSS style cho một Element.
59
+ * @param el Phần tử HTML cần gán style
60
+ * @param styles Đối tượng chứa các style (ví dụ: { color: 'red', display: 'block' })
61
+ */
36
62
  export const setStylesElement = (el, styles) => {
37
63
  Object.keys(styles).forEach((key) => set(el, `style.${key}`, styles[key]));
38
64
  };
65
+ /**
66
+ * Lấy kích thước Viewport hiển thị của cửa sổ trình duyệt.
67
+ * @param win Đối tượng Window (mặc định là window)
68
+ * @returns Đối tượng chứa width và height
69
+ */
39
70
  export const getViewport = (win = window) => {
40
71
  const doc = win.document.documentElement, body = win.document.getElementsByTagName('body')[0], w = win.innerWidth || doc.clientWidth || body.clientWidth, h = win.innerHeight || doc.clientHeight || body.clientHeight;
41
72
  return { width: w, height: h };
42
73
  };
74
+ /**
75
+ * Thiết lập vị trí con trỏ (caret) cho thẻ Input hoặc Textarea.
76
+ * @param element Thẻ input hoặc textarea
77
+ * @param position Vị trí cần đặt con trỏ
78
+ */
43
79
  export const setCaretPosition = (element, position) => {
44
80
  if (!element || !element.setSelectionRange) {
45
81
  return;
@@ -47,6 +83,14 @@ export const setCaretPosition = (element, position) => {
47
83
  element.focus();
48
84
  element.setSelectionRange(position, position);
49
85
  };
86
+ /**
87
+ * Kiểm tra xem một phần tử có đang hiển thị trong vùng nhìn thấy (viewport) của một container không.
88
+ * @param container Element cha chứa vùng scroll
89
+ * @param element Element cần kiểm tra
90
+ * @param elementScroll (Tùy chọn) Element đại diện cho vùng scroll nếu khác container
91
+ * @param maxTopLeft (Tùy chọn) Offset bổ sung để kiểm tra
92
+ * @returns true nếu nằm trong vùng nhìn thấy
93
+ */
50
94
  export const checkViewInScreen = (container, element, elementScroll, maxTopLeft) => {
51
95
  if (!container || !element) {
52
96
  return false;
@@ -74,6 +118,13 @@ export const checkViewInScreen = (container, element, elementScroll, maxTopLeft)
74
118
  }
75
119
  return false;
76
120
  };
121
+ /**
122
+ * Kiểm tra xem tọa độ chuột hiện tại có nằm trong vùng của một Element không.
123
+ * @param mousePosition Đối tượng chứa clientX, clientY
124
+ * @param element Element cần kiểm tra
125
+ * @param rect (Tùy chọn) Bounding rect đã có sẵn để tối ưu hiệu năng
126
+ * @returns true nếu chuột đang đè lên phần tử
127
+ */
77
128
  export const checkMouseOverInContainer = (mousePosition, element, rect) => {
78
129
  if (!element && !rect) {
79
130
  return false;
@@ -88,6 +139,11 @@ export const checkMouseOverInContainer = (mousePosition, element, rect) => {
88
139
  }
89
140
  return false;
90
141
  };
142
+ /**
143
+ * Tạo sự kiện Kéo-Thả (Drag) nâng cao dựa trên RxJS cho một Element.
144
+ * @param config Cấu hình sự kiện (element, các hàm callback, destroyRef)
145
+ * @returns Observable phát ra sự kiện mousemove trong quá trình kéo
146
+ */
91
147
  export const getDragEventByElement = (config) => {
92
148
  let timer;
93
149
  const addClass = () => {
@@ -123,4 +179,4 @@ export const getDragEventByElement = (config) => {
123
179
  }), takeUntil(mouseup));
124
180
  return mouseDown.pipe(mergeMap((e) => (config.isStartWithMouseDownEvent ? mousemove.pipe(startWith(e)) : mousemove)), takeUntilDestroyed(config.destroyRef), finalize(removeClass));
125
181
  };
126
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dom.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/dom.ts"],"names":[],"mappings":"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","sourcesContent":["import { DestroyRef } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { IBoundingClientRect } from '@libs-ui/interfaces-types';\n\nimport Bowser from 'bowser';\nimport { finalize, fromEvent, mergeMap, startWith, takeUntil, tap } from 'rxjs';\nimport { set } from './helpers';\n\nlet parser: DOMParser | null = null;\nlet bowserParser: Bowser.Parser.Parser | null = null;\n\nexport const getDeviceInfo = () => {\n  if (!bowserParser) {\n    bowserParser = Bowser.getParser(window.navigator.userAgent);\n  }\n  return bowserParser;\n};\n\nexport const isTouchDevice = (): boolean => ('onpointerdown' in navigator && navigator.maxTouchPoints > 0) || 'ontouchstart' in window || (window.matchMedia?.('(pointer: coarse)').matches ?? false);\n\nexport const getEventNameHandleClick = (() => {\n  if (isTouchDevice()) {\n    return 'pointerdown';\n  }\n  const deviceInfo = getDeviceInfo();\n\n  return deviceInfo.isPlatform('desktop') ? 'click' : 'touchstart';\n})();\n\nexport const getDocumentByString = (htmlStr: string): Document => {\n  if (!parser) {\n    parser = new DOMParser();\n  }\n  return parser.parseFromString(htmlStr, 'text/html');\n};\n\nexport const cloneIBoundingClientRect = (rect: IBoundingClientRect): IBoundingClientRect => {\n  return {\n    top: rect['top'],\n    left: rect['left'],\n    width: rect['width'],\n    height: rect['height'],\n    bottom: rect['bottom'],\n  };\n};\n\nexport const setStylesElement = (el: HTMLElement, styles: Record<string, string>): void => {\n  Object.keys(styles).forEach((key: string) => set(el, `style.${key}`, styles[key]));\n};\n\nexport const getViewport = (win: Window = window): { width: number; height: number } => {\n  const doc = win.document.documentElement,\n    body = win.document.getElementsByTagName('body')[0],\n    w = win.innerWidth || doc.clientWidth || body.clientWidth,\n    h = win.innerHeight || doc.clientHeight || body.clientHeight;\n\n  return { width: w, height: h };\n};\n\nexport const setCaretPosition = (element: HTMLTextAreaElement | HTMLInputElement, position: number): void => {\n  if (!element || !element.setSelectionRange) {\n    return;\n  }\n  element.focus();\n  element.setSelectionRange(position, position);\n};\n\nexport const checkViewInScreen = (container: HTMLElement, element: HTMLElement, elementScroll?: HTMLElement, maxTopLeft?: { top?: number; left?: number }): boolean => {\n  if (!container || !element) {\n    return false;\n  }\n\n  const rectContainer = cloneIBoundingClientRect(container.getBoundingClientRect());\n\n  if (elementScroll) {\n    const rectElementScroll = elementScroll.getBoundingClientRect();\n\n    rectContainer['left'] = rectElementScroll.left;\n    rectContainer['top'] = rectElementScroll.top;\n  }\n\n  const rectElement = element.getBoundingClientRect();\n  const { left, top } = rectElement;\n  const topContainer = rectContainer['top'];\n  const leftContainer = rectContainer['left'];\n  const widthContainer = rectContainer['width'];\n  const heightContainer = rectContainer['height'];\n  const scrollTopContainer = container.scrollTop;\n  const scrollLeftContainer = container.scrollLeft;\n  const limitTop = topContainer + scrollTopContainer + heightContainer;\n  const limitLeft = leftContainer + scrollLeftContainer + widthContainer;\n\n  const maxTopItem = top + (maxTopLeft?.top || 0);\n  const maxLeftItem = left + (maxTopLeft?.left || 0);\n\n  if (topContainer <= maxTopItem && maxTopItem <= limitTop && leftContainer <= maxLeftItem && maxLeftItem <= limitLeft) {\n    return true;\n  }\n\n  return false;\n};\n\nexport const checkMouseOverInContainer = (mousePosition: { clientX: number; clientY: number }, element?: HTMLElement, rect?: DOMRect): boolean => {\n  if (!element && !rect) {\n    return false;\n  }\n  const rectElement = rect || element?.getBoundingClientRect() || ({} as DOMRect);\n  const { left, top, width, height } = rectElement;\n  const limitLeft = left + width;\n  const limitTop = top + height;\n  const { clientX, clientY } = mousePosition;\n  if (left <= clientX && clientX <= limitLeft && top <= clientY && clientY <= limitTop) {\n    return true;\n  }\n\n  return false;\n};\n\nexport const getDragEventByElement = (config: {\n  elementMouseDown: HTMLElement;\n  functionMouseDown?: (event: MouseEvent) => void;\n  isStartWithMouseDownEvent?: boolean;\n  elementMouseMove?: HTMLElement;\n  functionMouseMove?: (event: MouseEvent) => void;\n  elementMouseUp?: HTMLElement;\n  functionMouseUp?: (event: MouseEvent) => void;\n  destroyRef: DestroyRef;\n}) => {\n  let timer: ReturnType<typeof setTimeout>;\n  const addClass = () => {\n    timer = setTimeout(() => {\n      document.body.classList.add('!select-none');\n      document.body.classList.add('!cursor-grabbing');\n    }, 250);\n  };\n  const removeClass = () => {\n    document.body.classList.remove('!select-none');\n    document.body.classList.remove('!cursor-grabbing');\n    clearTimeout(timer);\n  };\n  fromEvent(config.elementMouseDown, 'mouseleave')\n    .pipe(\n      tap(() => document.body.classList.remove('cursor-pointer')),\n      takeUntilDestroyed(config.destroyRef)\n    )\n    .subscribe();\n  fromEvent(config.elementMouseDown, 'mouseenter')\n    .pipe(\n      tap(() => document.body.classList.add('cursor-pointer')),\n      takeUntilDestroyed(config.destroyRef)\n    )\n    .subscribe();\n\n  const mouseDown = fromEvent<MouseEvent>(config.elementMouseDown, 'mousedown').pipe(\n    tap((e) => {\n      e.stopPropagation();\n      addClass();\n      config.functionMouseDown?.(e);\n    })\n  );\n  const mouseup = fromEvent<MouseEvent>(config.elementMouseUp || document, 'mouseup').pipe(\n    tap((e) => {\n      e.stopPropagation();\n      removeClass();\n      config.functionMouseUp?.(e);\n    })\n  );\n\n  const mousemove = fromEvent<MouseEvent>(config.elementMouseMove || document, 'mousemove').pipe(\n    tap((e) => {\n      e.stopPropagation();\n      config.functionMouseMove?.(e);\n    }),\n    takeUntil(mouseup)\n  );\n\n  return mouseDown.pipe(\n    mergeMap((e: MouseEvent) => (config.isStartWithMouseDownEvent ? mousemove.pipe(startWith(e)) : mousemove)),\n    takeUntilDestroyed(config.destroyRef),\n    finalize(removeClass)\n  );\n};\n"]}
182
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dom.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/dom.ts"],"names":[],"mappings":"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","sourcesContent":["import { DestroyRef } from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { IBoundingClientRect } from '@libs-ui/interfaces-types';\n\nimport Bowser from 'bowser';\nimport { finalize, fromEvent, mergeMap, startWith, takeUntil, tap } from 'rxjs';\nimport { set } from './helpers';\n\nlet parser: DOMParser | null = null;\nlet bowserParser: Bowser.Parser.Parser | null = null;\n\n/**\n * Lấy thông tin thiết bị và trình duyệt (Sử dụng thư viện Bowser).\n * @returns Đối tượng Parser chứa thông tin OS, Browser, Platform...\n */\nexport const getDeviceInfo = () => {\n  if (!bowserParser) {\n    bowserParser = Bowser.getParser(window.navigator.userAgent);\n  }\n  return bowserParser;\n};\n\n/**\n * Kiểm tra xem thiết bị hiện tại có hỗ trợ cảm ứng (Touch) không.\n * @returns true nếu là thiết bị cảm ứng\n */\nexport const isTouchDevice = (): boolean => ('onpointerdown' in navigator && navigator.maxTouchPoints > 0) || 'ontouchstart' in window || (window.matchMedia?.('(pointer: coarse)').matches ?? false);\n\n/**\n * Xác định tên sự kiện click phù hợp cho thiết bị (click cho desktop, pointerdown cho touch).\n */\nexport const getEventNameHandleClick = (() => {\n  if (isTouchDevice()) {\n    return 'pointerdown';\n  }\n  const deviceInfo = getDeviceInfo();\n\n  return deviceInfo.isPlatform('desktop') ? 'click' : 'touchstart';\n})();\n\n/**\n * Chuyển đổi một chuỗi HTML thành đối tượng Document.\n * @param htmlStr Chuỗi HTML cần parse\n * @returns Đối tượng Document tương ứng\n */\nexport const getDocumentByString = (htmlStr: string): Document => {\n  if (!parser) {\n    parser = new DOMParser();\n  }\n  return parser.parseFromString(htmlStr, 'text/html');\n};\n\n/**\n * Sao chép các thuộc tính cơ bản của IBoundingClientRect.\n * @param rect Đối tượng IBoundingClientRect gốc\n * @returns Đối tượng IBoundingClientRect đã được copy\n */\nexport const cloneIBoundingClientRect = (rect: IBoundingClientRect): IBoundingClientRect => {\n  return {\n    top: rect['top'],\n    left: rect['left'],\n    width: rect['width'],\n    height: rect['height'],\n    bottom: rect['bottom'],\n  };\n};\n\n/**\n * Gán nhiều thuộc tính CSS style cho một Element.\n * @param el Phần tử HTML cần gán style\n * @param styles Đối tượng chứa các style (ví dụ: { color: 'red', display: 'block' })\n */\nexport const setStylesElement = (el: HTMLElement, styles: Record<string, string>): void => {\n  Object.keys(styles).forEach((key: string) => set(el, `style.${key}`, styles[key]));\n};\n\n/**\n * Lấy kích thước Viewport hiển thị của cửa sổ trình duyệt.\n * @param win Đối tượng Window (mặc định là window)\n * @returns Đối tượng chứa width và height\n */\nexport const getViewport = (win: Window = window): { width: number; height: number } => {\n  const doc = win.document.documentElement,\n    body = win.document.getElementsByTagName('body')[0],\n    w = win.innerWidth || doc.clientWidth || body.clientWidth,\n    h = win.innerHeight || doc.clientHeight || body.clientHeight;\n\n  return { width: w, height: h };\n};\n\n/**\n * Thiết lập vị trí con trỏ (caret) cho thẻ Input hoặc Textarea.\n * @param element Thẻ input hoặc textarea\n * @param position Vị trí cần đặt con trỏ\n */\nexport const setCaretPosition = (element: HTMLTextAreaElement | HTMLInputElement, position: number): void => {\n  if (!element || !element.setSelectionRange) {\n    return;\n  }\n  element.focus();\n  element.setSelectionRange(position, position);\n};\n\n/**\n * Kiểm tra xem một phần tử có đang hiển thị trong vùng nhìn thấy (viewport) của một container không.\n * @param container Element cha chứa vùng scroll\n * @param element Element cần kiểm tra\n * @param elementScroll (Tùy chọn) Element đại diện cho vùng scroll nếu khác container\n * @param maxTopLeft (Tùy chọn) Offset bổ sung để kiểm tra\n * @returns true nếu nằm trong vùng nhìn thấy\n */\nexport const checkViewInScreen = (container: HTMLElement, element: HTMLElement, elementScroll?: HTMLElement, maxTopLeft?: { top?: number; left?: number }): boolean => {\n  if (!container || !element) {\n    return false;\n  }\n\n  const rectContainer = cloneIBoundingClientRect(container.getBoundingClientRect());\n\n  if (elementScroll) {\n    const rectElementScroll = elementScroll.getBoundingClientRect();\n\n    rectContainer['left'] = rectElementScroll.left;\n    rectContainer['top'] = rectElementScroll.top;\n  }\n\n  const rectElement = element.getBoundingClientRect();\n  const { left, top } = rectElement;\n  const topContainer = rectContainer['top'];\n  const leftContainer = rectContainer['left'];\n  const widthContainer = rectContainer['width'];\n  const heightContainer = rectContainer['height'];\n  const scrollTopContainer = container.scrollTop;\n  const scrollLeftContainer = container.scrollLeft;\n  const limitTop = topContainer + scrollTopContainer + heightContainer;\n  const limitLeft = leftContainer + scrollLeftContainer + widthContainer;\n\n  const maxTopItem = top + (maxTopLeft?.top || 0);\n  const maxLeftItem = left + (maxTopLeft?.left || 0);\n\n  if (topContainer <= maxTopItem && maxTopItem <= limitTop && leftContainer <= maxLeftItem && maxLeftItem <= limitLeft) {\n    return true;\n  }\n\n  return false;\n};\n\n/**\n * Kiểm tra xem tọa độ chuột hiện tại có nằm trong vùng của một Element không.\n * @param mousePosition Đối tượng chứa clientX, clientY\n * @param element Element cần kiểm tra\n * @param rect (Tùy chọn) Bounding rect đã có sẵn để tối ưu hiệu năng\n * @returns true nếu chuột đang đè lên phần tử\n */\nexport const checkMouseOverInContainer = (mousePosition: { clientX: number; clientY: number }, element?: HTMLElement, rect?: DOMRect): boolean => {\n  if (!element && !rect) {\n    return false;\n  }\n  const rectElement = rect || element?.getBoundingClientRect() || ({} as DOMRect);\n  const { left, top, width, height } = rectElement;\n  const limitLeft = left + width;\n  const limitTop = top + height;\n  const { clientX, clientY } = mousePosition;\n  if (left <= clientX && clientX <= limitLeft && top <= clientY && clientY <= limitTop) {\n    return true;\n  }\n\n  return false;\n};\n\n/**\n * Tạo sự kiện Kéo-Thả (Drag) nâng cao dựa trên RxJS cho một Element.\n * @param config Cấu hình sự kiện (element, các hàm callback, destroyRef)\n * @returns Observable phát ra sự kiện mousemove trong quá trình kéo\n */\nexport const getDragEventByElement = (config: {\n  elementMouseDown: HTMLElement;\n  functionMouseDown?: (event: MouseEvent) => void;\n  isStartWithMouseDownEvent?: boolean;\n  elementMouseMove?: HTMLElement;\n  functionMouseMove?: (event: MouseEvent) => void;\n  elementMouseUp?: HTMLElement;\n  functionMouseUp?: (event: MouseEvent) => void;\n  destroyRef: DestroyRef;\n}) => {\n  let timer: ReturnType<typeof setTimeout>;\n  const addClass = () => {\n    timer = setTimeout(() => {\n      document.body.classList.add('!select-none');\n      document.body.classList.add('!cursor-grabbing');\n    }, 250);\n  };\n  const removeClass = () => {\n    document.body.classList.remove('!select-none');\n    document.body.classList.remove('!cursor-grabbing');\n    clearTimeout(timer);\n  };\n  fromEvent(config.elementMouseDown, 'mouseleave')\n    .pipe(\n      tap(() => document.body.classList.remove('cursor-pointer')),\n      takeUntilDestroyed(config.destroyRef)\n    )\n    .subscribe();\n  fromEvent(config.elementMouseDown, 'mouseenter')\n    .pipe(\n      tap(() => document.body.classList.add('cursor-pointer')),\n      takeUntilDestroyed(config.destroyRef)\n    )\n    .subscribe();\n\n  const mouseDown = fromEvent<MouseEvent>(config.elementMouseDown, 'mousedown').pipe(\n    tap((e) => {\n      e.stopPropagation();\n      addClass();\n      config.functionMouseDown?.(e);\n    })\n  );\n  const mouseup = fromEvent<MouseEvent>(config.elementMouseUp || document, 'mouseup').pipe(\n    tap((e) => {\n      e.stopPropagation();\n      removeClass();\n      config.functionMouseUp?.(e);\n    })\n  );\n\n  const mousemove = fromEvent<MouseEvent>(config.elementMouseMove || document, 'mousemove').pipe(\n    tap((e) => {\n      e.stopPropagation();\n      config.functionMouseMove?.(e);\n    }),\n    takeUntil(mouseup)\n  );\n\n  return mouseDown.pipe(\n    mergeMap((e: MouseEvent) => (config.isStartWithMouseDownEvent ? mousemove.pipe(startWith(e)) : mousemove)),\n    takeUntilDestroyed(config.destroyRef),\n    finalize(removeClass)\n  );\n};\n"]}
@@ -1,4 +1,11 @@
1
1
  import { convertBase64ToBlob } from './base64';
2
+ /**
3
+ * Tải file từ URL bằng XMLHttpRequest.
4
+ * Khác với `downloadFileByUrl`, hàm này sử dụng XHR để tải file, hữu ích hơn
5
+ * trong các trường hợp cần header tùy chỉnh hoặc gọi API nội bộ.
6
+ * @param fileUrl URL của file cần tải
7
+ * @param filename Tên file sẽ được lưu về máy
8
+ */
2
9
  export const downloadFileByUrlUseXmlRequest = (fileUrl, filename) => {
3
10
  const xmlRequest = new XMLHttpRequest();
4
11
  xmlRequest.open('GET', fileUrl, true);
@@ -13,6 +20,13 @@ export const downloadFileByUrlUseXmlRequest = (fileUrl, filename) => {
13
20
  };
14
21
  xmlRequest.send();
15
22
  };
23
+ /**
24
+ * Tải file từ URL bằng cách tạo thẻ `<a>` ảo và kích hoạt sự kiện click.
25
+ * Hỗ trợ chế độ mở file trên tab mới (onlyOpen) hoặc tải xuống thực sự.
26
+ * @param fileUrl URL trực tiếp của file hoặc Blob URL
27
+ * @param filename Tên file sẽ được lưu khi tải về
28
+ * @param onlyOpen Nếu `true`, chỉ mở file trong tab mới thay vì tải về
29
+ */
16
30
  export const downloadFileByUrl = async (fileUrl, filename, onlyOpen) => {
17
31
  const downloadFileElement = document.createElement('a');
18
32
  if (!onlyOpen) {
@@ -28,6 +42,13 @@ export const downloadFileByUrl = async (fileUrl, filename, onlyOpen) => {
28
42
  downloadFileElement.click();
29
43
  downloadFileElement.remove();
30
44
  };
45
+ /**
46
+ * Tải ảnh từ một phần tử `<img>` dựa trên thuộc tính `src` (hỗ trợ Base64 Data URL).
47
+ * Thường dùng để cho phép người dùng lưu ảnh barcode, QR code về máy.
48
+ * @param imageElement Phần tử `<img>` cần lấy ảnh
49
+ * @param typeFileDownload Loại MIME của file (mặc định: 'image/png')
50
+ * @param nameFile Tên file sẽ được lưu (mặc định: 'barcode')
51
+ */
31
52
  export const downloadImageFromELement = (imageElement, typeFileDownload, nameFile) => {
32
53
  const parentElement = imageElement?.src;
33
54
  const blobData = convertBase64ToBlob(parentElement);
@@ -38,4 +59,4 @@ export const downloadImageFromELement = (imageElement, typeFileDownload, nameFil
38
59
  link.download = nameFile || 'barcode';
39
60
  link.click();
40
61
  };
41
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG93bmxvYWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzLXVpL3V0aWxzL3NyYy9kb3dubG9hZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFFL0MsTUFBTSxDQUFDLE1BQU0sOEJBQThCLEdBQUcsQ0FBQyxPQUFlLEVBQUUsUUFBZ0IsRUFBRSxFQUFFO0lBQ2xGLE1BQU0sVUFBVSxHQUFHLElBQUksY0FBYyxFQUFFLENBQUM7SUFFeEMsVUFBVSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQ3RDLFVBQVUsQ0FBQyxZQUFZLEdBQUcsTUFBTSxDQUFDO0lBQ2pDLFVBQVUsQ0FBQyxPQUFPLEdBQUcsR0FBRyxFQUFFO1FBQ3hCLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUU1RCxpQkFBaUIsQ0FBQyxHQUFHLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDbkMsQ0FBQyxDQUFDO0lBQ0YsVUFBVSxDQUFDLE1BQU0sR0FBRyxHQUFHLEVBQUU7UUFDdkIsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDLEdBQUcsQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBRTVELGlCQUFpQixDQUFDLEdBQUcsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUNuQyxDQUFDLENBQUM7SUFDRixVQUFVLENBQUMsSUFBSSxFQUFFLENBQUM7QUFDcEIsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQUcsS0FBSyxFQUFFLE9BQWUsRUFBRSxRQUFnQixFQUFFLFFBQWtCLEVBQUUsRUFBRTtJQUMvRixNQUFNLG1CQUFtQixHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFeEQsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ2QsTUFBTSxRQUFRLEdBQUcsTUFBTSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFFdEMsSUFBSSxRQUFRLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDaEIsTUFBTSxJQUFJLEdBQUcsTUFBTSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7WUFFbkMsT0FBTyxHQUFHLEdBQUcsQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDdEMsQ0FBQztJQUNILENBQUM7SUFDRCxtQkFBbUIsQ0FBQyxJQUFJLEdBQUcsT0FBTyxDQUFDO0lBQ25DLG1CQUFtQixDQUFDLFFBQVEsR0FBRyxRQUFRLENBQUM7SUFDeEMsbUJBQW1CLENBQUMsTUFBTSxHQUFHLFFBQVEsQ0FBQztJQUN0QyxtQkFBbUIsQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUM1QixtQkFBbUIsQ0FBQyxNQUFNLEVBQUUsQ0FBQztBQUMvQixDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSx3QkFBd0IsR0FBRyxDQUFDLFlBQThCLEVBQUUsZ0JBQXlCLEVBQUUsUUFBaUIsRUFBRSxFQUFFO0lBQ3ZILE1BQU0sYUFBYSxHQUFHLFlBQVksRUFBRSxHQUFHLENBQUM7SUFDeEMsTUFBTSxRQUFRLEdBQUcsbUJBQW1CLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDcEQsTUFBTSxJQUFJLEdBQUcsSUFBSSxJQUFJLENBQUMsQ0FBQyxRQUFRLENBQUMsRUFBRSxFQUFFLElBQUksRUFBRSxnQkFBZ0IsSUFBSSxXQUFXLEVBQUUsQ0FBQyxDQUFDO0lBQzdFLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdDLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFekMsSUFBSSxDQUFDLElBQUksR0FBRyxHQUFHLENBQUM7SUFDaEIsSUFBSSxDQUFDLFFBQVEsR0FBRyxRQUFRLElBQUksU0FBUyxDQUFDO0lBQ3RDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztBQUNmLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNvbnZlcnRCYXNlNjRUb0Jsb2IgfSBmcm9tICcuL2Jhc2U2NCc7XG5cbmV4cG9ydCBjb25zdCBkb3dubG9hZEZpbGVCeVVybFVzZVhtbFJlcXVlc3QgPSAoZmlsZVVybDogc3RyaW5nLCBmaWxlbmFtZTogc3RyaW5nKSA9PiB7XG4gIGNvbnN0IHhtbFJlcXVlc3QgPSBuZXcgWE1MSHR0cFJlcXVlc3QoKTtcblxuICB4bWxSZXF1ZXN0Lm9wZW4oJ0dFVCcsIGZpbGVVcmwsIHRydWUpO1xuICB4bWxSZXF1ZXN0LnJlc3BvbnNlVHlwZSA9ICdibG9iJztcbiAgeG1sUmVxdWVzdC5vbmVycm9yID0gKCkgPT4ge1xuICAgIGNvbnN0IHVybCA9IHdpbmRvdy5VUkwuY3JlYXRlT2JqZWN0VVJMKHhtbFJlcXVlc3QucmVzcG9uc2UpO1xuXG4gICAgZG93bmxvYWRGaWxlQnlVcmwodXJsLCBmaWxlbmFtZSk7XG4gIH07XG4gIHhtbFJlcXVlc3Qub25sb2FkID0gKCkgPT4ge1xuICAgIGNvbnN0IHVybCA9IHdpbmRvdy5VUkwuY3JlYXRlT2JqZWN0VVJMKHhtbFJlcXVlc3QucmVzcG9uc2UpO1xuXG4gICAgZG93bmxvYWRGaWxlQnlVcmwodXJsLCBmaWxlbmFtZSk7XG4gIH07XG4gIHhtbFJlcXVlc3Quc2VuZCgpO1xufTtcblxuZXhwb3J0IGNvbnN0IGRvd25sb2FkRmlsZUJ5VXJsID0gYXN5bmMgKGZpbGVVcmw6IHN0cmluZywgZmlsZW5hbWU6IHN0cmluZywgb25seU9wZW4/OiBib29sZWFuKSA9PiB7XG4gIGNvbnN0IGRvd25sb2FkRmlsZUVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdhJyk7XG5cbiAgaWYgKCFvbmx5T3Blbikge1xuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2goZmlsZVVybCk7XG5cbiAgICBpZiAocmVzcG9uc2Uub2spIHtcbiAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCByZXNwb25zZS5ibG9iKCk7XG5cbiAgICAgIGZpbGVVcmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGJsb2IpO1xuICAgIH1cbiAgfVxuICBkb3dubG9hZEZpbGVFbGVtZW50LmhyZWYgPSBmaWxlVXJsO1xuICBkb3dubG9hZEZpbGVFbGVtZW50LmRvd25sb2FkID0gZmlsZW5hbWU7XG4gIGRvd25sb2FkRmlsZUVsZW1lbnQudGFyZ2V0ID0gJ19ibGFuayc7XG4gIGRvd25sb2FkRmlsZUVsZW1lbnQuY2xpY2soKTtcbiAgZG93bmxvYWRGaWxlRWxlbWVudC5yZW1vdmUoKTtcbn07XG5cbmV4cG9ydCBjb25zdCBkb3dubG9hZEltYWdlRnJvbUVMZW1lbnQgPSAoaW1hZ2VFbGVtZW50OiBIVE1MSW1hZ2VFbGVtZW50LCB0eXBlRmlsZURvd25sb2FkPzogc3RyaW5nLCBuYW1lRmlsZT86IHN0cmluZykgPT4ge1xuICBjb25zdCBwYXJlbnRFbGVtZW50ID0gaW1hZ2VFbGVtZW50Py5zcmM7XG4gIGNvbnN0IGJsb2JEYXRhID0gY29udmVydEJhc2U2NFRvQmxvYihwYXJlbnRFbGVtZW50KTtcbiAgY29uc3QgYmxvYiA9IG5ldyBCbG9iKFtibG9iRGF0YV0sIHsgdHlwZTogdHlwZUZpbGVEb3dubG9hZCB8fCAnaW1hZ2UvcG5nJyB9KTtcbiAgY29uc3QgdXJsID0gd2luZG93LlVSTC5jcmVhdGVPYmplY3RVUkwoYmxvYik7XG4gIGNvbnN0IGxpbmsgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdhJyk7XG5cbiAgbGluay5ocmVmID0gdXJsO1xuICBsaW5rLmRvd25sb2FkID0gbmFtZUZpbGUgfHwgJ2JhcmNvZGUnO1xuICBsaW5rLmNsaWNrKCk7XG59O1xuIl19
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG93bmxvYWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzLXVpL3V0aWxzL3NyYy9kb3dubG9hZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFFL0M7Ozs7OztHQU1HO0FBQ0gsTUFBTSxDQUFDLE1BQU0sOEJBQThCLEdBQUcsQ0FBQyxPQUFlLEVBQUUsUUFBZ0IsRUFBRSxFQUFFO0lBQ2xGLE1BQU0sVUFBVSxHQUFHLElBQUksY0FBYyxFQUFFLENBQUM7SUFFeEMsVUFBVSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQ3RDLFVBQVUsQ0FBQyxZQUFZLEdBQUcsTUFBTSxDQUFDO0lBQ2pDLFVBQVUsQ0FBQyxPQUFPLEdBQUcsR0FBRyxFQUFFO1FBQ3hCLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUU1RCxpQkFBaUIsQ0FBQyxHQUFHLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDbkMsQ0FBQyxDQUFDO0lBQ0YsVUFBVSxDQUFDLE1BQU0sR0FBRyxHQUFHLEVBQUU7UUFDdkIsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDLEdBQUcsQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBRTVELGlCQUFpQixDQUFDLEdBQUcsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUNuQyxDQUFDLENBQUM7SUFDRixVQUFVLENBQUMsSUFBSSxFQUFFLENBQUM7QUFDcEIsQ0FBQyxDQUFDO0FBRUY7Ozs7OztHQU1HO0FBQ0gsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQUcsS0FBSyxFQUFFLE9BQWUsRUFBRSxRQUFnQixFQUFFLFFBQWtCLEVBQUUsRUFBRTtJQUMvRixNQUFNLG1CQUFtQixHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFeEQsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ2QsTUFBTSxRQUFRLEdBQUcsTUFBTSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFFdEMsSUFBSSxRQUFRLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDaEIsTUFBTSxJQUFJLEdBQUcsTUFBTSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7WUFFbkMsT0FBTyxHQUFHLEdBQUcsQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDdEMsQ0FBQztJQUNILENBQUM7SUFDRCxtQkFBbUIsQ0FBQyxJQUFJLEdBQUcsT0FBTyxDQUFDO0lBQ25DLG1CQUFtQixDQUFDLFFBQVEsR0FBRyxRQUFRLENBQUM7SUFDeEMsbUJBQW1CLENBQUMsTUFBTSxHQUFHLFFBQVEsQ0FBQztJQUN0QyxtQkFBbUIsQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUM1QixtQkFBbUIsQ0FBQyxNQUFNLEVBQUUsQ0FBQztBQUMvQixDQUFDLENBQUM7QUFFRjs7Ozs7O0dBTUc7QUFDSCxNQUFNLENBQUMsTUFBTSx3QkFBd0IsR0FBRyxDQUFDLFlBQThCLEVBQUUsZ0JBQXlCLEVBQUUsUUFBaUIsRUFBRSxFQUFFO0lBQ3ZILE1BQU0sYUFBYSxHQUFHLFlBQVksRUFBRSxHQUFHLENBQUM7SUFDeEMsTUFBTSxRQUFRLEdBQUcsbUJBQW1CLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDcEQsTUFBTSxJQUFJLEdBQUcsSUFBSSxJQUFJLENBQUMsQ0FBQyxRQUFRLENBQUMsRUFBRSxFQUFFLElBQUksRUFBRSxnQkFBZ0IsSUFBSSxXQUFXLEVBQUUsQ0FBQyxDQUFDO0lBQzdFLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdDLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFekMsSUFBSSxDQUFDLElBQUksR0FBRyxHQUFHLENBQUM7SUFDaEIsSUFBSSxDQUFDLFFBQVEsR0FBRyxRQUFRLElBQUksU0FBUyxDQUFDO0lBQ3RDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztBQUNmLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNvbnZlcnRCYXNlNjRUb0Jsb2IgfSBmcm9tICcuL2Jhc2U2NCc7XG5cbi8qKlxuICogVOG6o2kgZmlsZSB04burIFVSTCBi4bqxbmcgWE1MSHR0cFJlcXVlc3QuXG4gKiBLaMOhYyB24bubaSBgZG93bmxvYWRGaWxlQnlVcmxgLCBow6BtIG7DoHkgc+G7rSBk4bulbmcgWEhSIMSR4buDIHThuqNpIGZpbGUsIGjhu691IMOtY2ggaMahblxuICogdHJvbmcgY8OhYyB0csaw4budbmcgaOG7o3AgY+G6p24gaGVhZGVyIHTDuXkgY2jhu4luaCBob+G6t2MgZ+G7jWkgQVBJIG7hu5lpIGLhu5kuXG4gKiBAcGFyYW0gZmlsZVVybCBVUkwgY+G7p2EgZmlsZSBj4bqnbiB04bqjaVxuICogQHBhcmFtIGZpbGVuYW1lIFTDqm4gZmlsZSBz4bq9IMSRxrDhu6NjIGzGsHUgduG7gSBtw6F5XG4gKi9cbmV4cG9ydCBjb25zdCBkb3dubG9hZEZpbGVCeVVybFVzZVhtbFJlcXVlc3QgPSAoZmlsZVVybDogc3RyaW5nLCBmaWxlbmFtZTogc3RyaW5nKSA9PiB7XG4gIGNvbnN0IHhtbFJlcXVlc3QgPSBuZXcgWE1MSHR0cFJlcXVlc3QoKTtcblxuICB4bWxSZXF1ZXN0Lm9wZW4oJ0dFVCcsIGZpbGVVcmwsIHRydWUpO1xuICB4bWxSZXF1ZXN0LnJlc3BvbnNlVHlwZSA9ICdibG9iJztcbiAgeG1sUmVxdWVzdC5vbmVycm9yID0gKCkgPT4ge1xuICAgIGNvbnN0IHVybCA9IHdpbmRvdy5VUkwuY3JlYXRlT2JqZWN0VVJMKHhtbFJlcXVlc3QucmVzcG9uc2UpO1xuXG4gICAgZG93bmxvYWRGaWxlQnlVcmwodXJsLCBmaWxlbmFtZSk7XG4gIH07XG4gIHhtbFJlcXVlc3Qub25sb2FkID0gKCkgPT4ge1xuICAgIGNvbnN0IHVybCA9IHdpbmRvdy5VUkwuY3JlYXRlT2JqZWN0VVJMKHhtbFJlcXVlc3QucmVzcG9uc2UpO1xuXG4gICAgZG93bmxvYWRGaWxlQnlVcmwodXJsLCBmaWxlbmFtZSk7XG4gIH07XG4gIHhtbFJlcXVlc3Quc2VuZCgpO1xufTtcblxuLyoqXG4gKiBU4bqjaSBmaWxlIHThu6sgVVJMIGLhurFuZyBjw6FjaCB04bqhbyB0aOG6uyBgPGE+YCDhuqNvIHbDoCBrw61jaCBob+G6oXQgc+G7sSBraeG7h24gY2xpY2suXG4gKiBI4buXIHRy4bujIGNo4bq/IMSR4buZIG3hu58gZmlsZSB0csOqbiB0YWIgbeG7m2kgKG9ubHlPcGVuKSBob+G6t2MgdOG6o2kgeHXhu5FuZyB0aOG7sWMgc+G7sS5cbiAqIEBwYXJhbSBmaWxlVXJsIFVSTCB0cuG7sWMgdGnhur9wIGPhu6dhIGZpbGUgaG/hurdjIEJsb2IgVVJMXG4gKiBAcGFyYW0gZmlsZW5hbWUgVMOqbiBmaWxlIHPhur0gxJHGsOG7o2MgbMawdSBraGkgdOG6o2kgduG7gVxuICogQHBhcmFtIG9ubHlPcGVuIE7hur91IGB0cnVlYCwgY2jhu4kgbeG7nyBmaWxlIHRyb25nIHRhYiBt4bubaSB0aGF5IHbDrCB04bqjaSB24buBXG4gKi9cbmV4cG9ydCBjb25zdCBkb3dubG9hZEZpbGVCeVVybCA9IGFzeW5jIChmaWxlVXJsOiBzdHJpbmcsIGZpbGVuYW1lOiBzdHJpbmcsIG9ubHlPcGVuPzogYm9vbGVhbikgPT4ge1xuICBjb25zdCBkb3dubG9hZEZpbGVFbGVtZW50ID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnYScpO1xuXG4gIGlmICghb25seU9wZW4pIHtcbiAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGZldGNoKGZpbGVVcmwpO1xuXG4gICAgaWYgKHJlc3BvbnNlLm9rKSB7XG4gICAgICBjb25zdCBibG9iID0gYXdhaXQgcmVzcG9uc2UuYmxvYigpO1xuXG4gICAgICBmaWxlVXJsID0gVVJMLmNyZWF0ZU9iamVjdFVSTChibG9iKTtcbiAgICB9XG4gIH1cbiAgZG93bmxvYWRGaWxlRWxlbWVudC5ocmVmID0gZmlsZVVybDtcbiAgZG93bmxvYWRGaWxlRWxlbWVudC5kb3dubG9hZCA9IGZpbGVuYW1lO1xuICBkb3dubG9hZEZpbGVFbGVtZW50LnRhcmdldCA9ICdfYmxhbmsnO1xuICBkb3dubG9hZEZpbGVFbGVtZW50LmNsaWNrKCk7XG4gIGRvd25sb2FkRmlsZUVsZW1lbnQucmVtb3ZlKCk7XG59O1xuXG4vKipcbiAqIFThuqNpIOG6o25oIHThu6sgbeG7mXQgcGjhuqduIHThu60gYDxpbWc+YCBk4buxYSB0csOqbiB0aHXhu5ljIHTDrW5oIGBzcmNgICho4buXIHRy4bujIEJhc2U2NCBEYXRhIFVSTCkuXG4gKiBUaMaw4budbmcgZMO5bmcgxJHhu4MgY2hvIHBow6lwIG5nxrDhu51pIGTDuW5nIGzGsHUg4bqjbmggYmFyY29kZSwgUVIgY29kZSB24buBIG3DoXkuXG4gKiBAcGFyYW0gaW1hZ2VFbGVtZW50IFBo4bqnbiB04butIGA8aW1nPmAgY+G6p24gbOG6pXkg4bqjbmhcbiAqIEBwYXJhbSB0eXBlRmlsZURvd25sb2FkIExv4bqhaSBNSU1FIGPhu6dhIGZpbGUgKG3hurdjIMSR4buLbmg6ICdpbWFnZS9wbmcnKVxuICogQHBhcmFtIG5hbWVGaWxlIFTDqm4gZmlsZSBz4bq9IMSRxrDhu6NjIGzGsHUgKG3hurdjIMSR4buLbmg6ICdiYXJjb2RlJylcbiAqL1xuZXhwb3J0IGNvbnN0IGRvd25sb2FkSW1hZ2VGcm9tRUxlbWVudCA9IChpbWFnZUVsZW1lbnQ6IEhUTUxJbWFnZUVsZW1lbnQsIHR5cGVGaWxlRG93bmxvYWQ/OiBzdHJpbmcsIG5hbWVGaWxlPzogc3RyaW5nKSA9PiB7XG4gIGNvbnN0IHBhcmVudEVsZW1lbnQgPSBpbWFnZUVsZW1lbnQ/LnNyYztcbiAgY29uc3QgYmxvYkRhdGEgPSBjb252ZXJ0QmFzZTY0VG9CbG9iKHBhcmVudEVsZW1lbnQpO1xuICBjb25zdCBibG9iID0gbmV3IEJsb2IoW2Jsb2JEYXRhXSwgeyB0eXBlOiB0eXBlRmlsZURvd25sb2FkIHx8ICdpbWFnZS9wbmcnIH0pO1xuICBjb25zdCB1cmwgPSB3aW5kb3cuVVJMLmNyZWF0ZU9iamVjdFVSTChibG9iKTtcbiAgY29uc3QgbGluayA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2EnKTtcblxuICBsaW5rLmhyZWYgPSB1cmw7XG4gIGxpbmsuZG93bmxvYWQgPSBuYW1lRmlsZSB8fCAnYmFyY29kZSc7XG4gIGxpbmsuY2xpY2soKTtcbn07XG4iXX0=
package/esm2022/file.mjs CHANGED
@@ -1,11 +1,17 @@
1
1
  import { convertBase64ToBlob } from './base64';
2
2
  import { get, set } from './helpers';
3
3
  import { uuid } from './uuid';
4
+ /** Kiểm tra file có phải là ảnh không (image/*). */
4
5
  export const isTypeImage = (file) => (file.type.match(/image.*/) ? true : false);
6
+ /** Kiểm tra file có phải là video không (video/*). */
5
7
  export const isTypeVideo = (file) => (file.type.match(/video.*/) ? true : false);
8
+ /** Kiểm tra file có phải là audio không (audio/*). */
6
9
  export const isTypeAudio = (file) => (file.type.match(/audio.*/) ? true : false);
10
+ /** Kiểm tra giá trị có phải là đối tượng File không. */
7
11
  export const isTypeFile = (file) => (file instanceof File || Object.prototype.toString.call(file) === '[object File]' ? true : false);
12
+ /** Danh sách extension/MIME type của file Excel. */
8
13
  export const ExcelExtList = ['xls', 'xlsx', 'application/vnd.ms-excel', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'];
14
+ /** Danh sách extension/MIME type của tài liệu (Word, Excel, PDF, JSON...). */
9
15
  export const DocumentExtList = [
10
16
  'doc',
11
17
  'docx',
@@ -26,13 +32,40 @@ export const DocumentExtList = [
26
32
  'application/json',
27
33
  'application/xml',
28
34
  ];
35
+ /**
36
+ * Kiểm tra extension có thuộc danh sách tài liệu được chấp nhận không.
37
+ * @param ext Extension hoặc MIME type cần kiểm tra
38
+ * @param listExt Danh sách tùy chỉnh (mặc định là DocumentExtList)
39
+ */
29
40
  export const isIncludeDocumentExtList = (ext, listExt = DocumentExtList) => listExt.includes(ext) || listExt.includes(`application/${ext}`);
41
+ /** Danh sách extension/MIME type của ảnh. */
30
42
  export const ImageExtList = ['gif', 'jpg', 'jpeg', 'png', 'image/gif', 'image/jpeg', 'image/jpeg', 'image/png', 'webp'];
43
+ /**
44
+ * Kiểm tra extension có thuộc danh sách ảnh không.
45
+ * @param ext Extension hoặc MIME type cần kiểm tra
46
+ * @param listExt Danh sách tùy chỉnh (mặc định là ImageExtList)
47
+ */
31
48
  export const isIncludeImageExtList = (ext, listExt = ImageExtList) => listExt.includes(ext);
49
+ /** Danh sách extension/MIME type của video. */
32
50
  export const VideoExtList = ['mp4', 'mov', 'mpg', 'avi', 'wmv', 'video/mp4', 'video/quicktime', 'video/mpeg', 'video/x-msvideo', 'video/x-ms-wmv'];
51
+ /**
52
+ * Kiểm tra extension có thuộc danh sách video không.
53
+ * @param ext Extension hoặc MIME type cần kiểm tra
54
+ */
33
55
  export const isIncludeVideoExtList = (ext, listExt = VideoExtList) => listExt.includes(ext);
56
+ /** Danh sách extension/MIME type của audio. */
34
57
  export const AudioExtList = ['mp3', 'wav', 'ogg', 'aac', 'm4a', 'flac', 'wma', 'audio/mpeg', 'audio/wav', 'audio/ogg', 'audio/aac', 'audio/mp4', 'audio/flac', 'audio/x-ms-wma'];
58
+ /**
59
+ * Kiểm tra extension có thuộc danh sách audio không.
60
+ * @param ext Extension hoặc MIME type cần kiểm tra
61
+ */
35
62
  export const isIncludeAudioExtList = (ext, listExt = AudioExtList) => listExt.includes(ext);
63
+ /**
64
+ * Lấy phần mở rộng (extension) của một file.
65
+ * Hỗ trợ cả đối tượng File gốc của Browser và IFile (interface nội bộ).
66
+ * @param file Đối tượng File hoặc IFile
67
+ * @returns Extension dạng chữ thường (vd: "jpg", "pdf") hoặc undefined
68
+ */
36
69
  export const getFileExtension = (file) => {
37
70
  if (file instanceof File && file.type) {
38
71
  return (file.type.split('/').pop() || '').toLowerCase();
@@ -56,17 +89,37 @@ export const getFileExtension = (file) => {
56
89
  }
57
90
  return (dots.pop() || '').toLowerCase();
58
91
  };
92
+ /**
93
+ * Định dạng kích thước file dạng chuỗi dễ đọc (KB hoặc MB).
94
+ * @param size Kích thước file tính bằng byte
95
+ * @param toFixed Số chữ số thập phân (mặc định: 2)
96
+ * @returns Chuỗi định dạng (vd: " 1.50 MB", " 512.00 KB")
97
+ */
59
98
  export const getLabelBySizeFile = (size, toFixed = 2) => {
60
99
  if (size < 1024 * 1024) {
61
100
  return ` ${(size / 1024).toFixed(toFixed)} KB`;
62
101
  }
63
102
  return ` ${(size / (1024 * 1024)).toFixed(toFixed)} MB`;
64
103
  };
104
+ /**
105
+ * Chuyển đổi đối tượng Blob thành File.
106
+ * Tự động xác định phần mở rộng từ MIME type của Blob.
107
+ * @param blob Đối tượng Blob cần chuyển
108
+ * @param fileName Tên file mong muốn (mặc định: UUID ngẫu nhiên)
109
+ * @returns Đối tượng File tương ứng
110
+ */
65
111
  export const convertBlobToFile = (blob, fileName) => {
66
112
  const type = blob.type.split('/')[1];
67
113
  const name = fileName ? `${fileName.split('.')[0]}.${type}` : `${uuid()}.${type}`;
68
114
  return new File([blob], name, { type: blob.type, lastModified: Date.now() });
69
115
  };
116
+ /**
117
+ * Tải file từ URL và chuyển về đối tượng File.
118
+ * Sử dụng XHR + FileReader để đọc nội dung, sau đó chuyển Base64 → Blob → File.
119
+ * @param url URL của file cần tải
120
+ * @param fileName Tên file mong muốn (mặc định: timestamp)
121
+ * @returns Promise trả về File hoặc undefined nếu thất bại
122
+ */
70
123
  export const convertUrlToFile = (url, fileName) => {
71
124
  return new Promise((resolve, reject) => {
72
125
  const xhr = new XMLHttpRequest();
@@ -87,4 +140,4 @@ export const convertUrlToFile = (url, fileName) => {
87
140
  xhr.send();
88
141
  });
89
142
  };
90
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/file.ts"],"names":[],"mappings":"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","sourcesContent":["import { IFile } from '@libs-ui/interfaces-types';\nimport { convertBase64ToBlob } from './base64';\nimport { get, set } from './helpers';\nimport { uuid } from './uuid';\n\nexport const isTypeImage = (file: File | Blob) => (file.type.match(/image.*/) ? true : false);\nexport const isTypeVideo = (file: File | Blob) => (file.type.match(/video.*/) ? true : false);\nexport const isTypeAudio = (file: File | Blob) => (file.type.match(/audio.*/) ? true : false);\nexport const isTypeFile = (file: File) => (file instanceof File || Object.prototype.toString.call(file) === '[object File]' ? true : false);\n\nexport const ExcelExtList = ['xls', 'xlsx', 'application/vnd.ms-excel', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'];\nexport const DocumentExtList = [\n  'doc',\n  'docx',\n  'xls',\n  'xlsx',\n  'ppt',\n  'pptx',\n  'pdf',\n  'json',\n  'xml',\n  'application/msword',\n  'application/vnd.openxmlformats-officedocument.wordprocessingml.document',\n  'application/vnd.ms-excel',\n  'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',\n  'application/vnd.ms-powerpoint',\n  'application/vnd.openxmlformats-officedocument.presentationml.presentation',\n  'application/pdf',\n  'application/json',\n  'application/xml',\n];\nexport const isIncludeDocumentExtList = (ext: string, listExt: Array<string> = DocumentExtList) => listExt.includes(ext) || listExt.includes(`application/${ext}`);\n\nexport const ImageExtList = ['gif', 'jpg', 'jpeg', 'png', 'image/gif', 'image/jpeg', 'image/jpeg', 'image/png', 'webp'];\nexport const isIncludeImageExtList = (ext: string, listExt: Array<string> = ImageExtList) => listExt.includes(ext);\n\nexport const VideoExtList = ['mp4', 'mov', 'mpg', 'avi', 'wmv', 'video/mp4', 'video/quicktime', 'video/mpeg', 'video/x-msvideo', 'video/x-ms-wmv'];\nexport const isIncludeVideoExtList = (ext: string, listExt: Array<string> = VideoExtList) => listExt.includes(ext);\nexport const AudioExtList = ['mp3', 'wav', 'ogg', 'aac', 'm4a', 'flac', 'wma', 'audio/mpeg', 'audio/wav', 'audio/ogg', 'audio/aac', 'audio/mp4', 'audio/flac', 'audio/x-ms-wma'];\nexport const isIncludeAudioExtList = (ext: string, listExt: Array<string> = AudioExtList) => listExt.includes(ext);\n\nexport const getFileExtension = (file: IFile | File) => {\n  if (file instanceof File && file.type) {\n    return (file.type.split('/').pop() || '').toLowerCase();\n  }\n  file = file as IFile;\n  if (file.mimetype || file.file?.type) {\n    return file.mimetype || file.file?.type;\n  }\n\n  let fileName = file.name as string;\n  const url = get(file, 'url') || get(file, 'origin_url');\n  if (!fileName && !url) {\n    return;\n  }\n  if (!fileName && url) {\n    fileName = url.split('/').pop() || '';\n    set(file, 'name', fileName);\n  }\n  const dots = fileName.split('.');\n\n  if (!dots) {\n    return;\n  }\n\n  return (dots.pop() || '').toLowerCase();\n};\n\nexport const getLabelBySizeFile = (size: number, toFixed = 2) => {\n  if (size < 1024 * 1024) {\n    return ` ${(size / 1024).toFixed(toFixed)} KB`;\n  }\n\n  return ` ${(size / (1024 * 1024)).toFixed(toFixed)} MB`;\n};\n\nexport const convertBlobToFile = (blob: Blob, fileName?: string): File => {\n  const type = blob.type.split('/')[1];\n  const name = fileName ? `${fileName.split('.')[0]}.${type}` : `${uuid()}.${type}`;\n\n  return new File([blob], name, { type: blob.type, lastModified: Date.now() });\n};\n\nexport const convertUrlToFile = (url: string, fileName?: string) => {\n  return new Promise<File | undefined>((resolve, reject) => {\n    const xhr = new XMLHttpRequest();\n    xhr.onload = function () {\n      const reader = new FileReader();\n      reader.onloadend = () => {\n        const file = convertBase64ToBlob(reader.result as string);\n        resolve(convertBlobToFile(file, fileName || Date.now().toLocaleString()));\n      };\n      reader.onerror = () => resolve(undefined);\n      reader.readAsDataURL(xhr.response);\n    };\n    xhr.onerror = (err) => {\n      reject(err);\n    };\n    xhr.open('GET', url);\n    xhr.responseType = 'blob';\n    xhr.send();\n  });\n};\n"]}
143
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"file.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/file.ts"],"names":[],"mappings":"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","sourcesContent":["import { IFile } from '@libs-ui/interfaces-types';\nimport { convertBase64ToBlob } from './base64';\nimport { get, set } from './helpers';\nimport { uuid } from './uuid';\n\n/** Kiểm tra file có phải là ảnh không (image/*). */\nexport const isTypeImage = (file: File | Blob) => (file.type.match(/image.*/) ? true : false);\n/** Kiểm tra file có phải là video không (video/*). */\nexport const isTypeVideo = (file: File | Blob) => (file.type.match(/video.*/) ? true : false);\n/** Kiểm tra file có phải là audio không (audio/*). */\nexport const isTypeAudio = (file: File | Blob) => (file.type.match(/audio.*/) ? true : false);\n/** Kiểm tra giá trị có phải là đối tượng File không. */\nexport const isTypeFile = (file: File) => (file instanceof File || Object.prototype.toString.call(file) === '[object File]' ? true : false);\n\n/** Danh sách extension/MIME type của file Excel. */\nexport const ExcelExtList = ['xls', 'xlsx', 'application/vnd.ms-excel', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'];\n\n/** Danh sách extension/MIME type của tài liệu (Word, Excel, PDF, JSON...). */\nexport const DocumentExtList = [\n  'doc',\n  'docx',\n  'xls',\n  'xlsx',\n  'ppt',\n  'pptx',\n  'pdf',\n  'json',\n  'xml',\n  'application/msword',\n  'application/vnd.openxmlformats-officedocument.wordprocessingml.document',\n  'application/vnd.ms-excel',\n  'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',\n  'application/vnd.ms-powerpoint',\n  'application/vnd.openxmlformats-officedocument.presentationml.presentation',\n  'application/pdf',\n  'application/json',\n  'application/xml',\n];\n\n/**\n * Kiểm tra extension có thuộc danh sách tài liệu được chấp nhận không.\n * @param ext Extension hoặc MIME type cần kiểm tra\n * @param listExt Danh sách tùy chỉnh (mặc định là DocumentExtList)\n */\nexport const isIncludeDocumentExtList = (ext: string, listExt: Array<string> = DocumentExtList) => listExt.includes(ext) || listExt.includes(`application/${ext}`);\n\n/** Danh sách extension/MIME type của ảnh. */\nexport const ImageExtList = ['gif', 'jpg', 'jpeg', 'png', 'image/gif', 'image/jpeg', 'image/jpeg', 'image/png', 'webp'];\n\n/**\n * Kiểm tra extension có thuộc danh sách ảnh không.\n * @param ext Extension hoặc MIME type cần kiểm tra\n * @param listExt Danh sách tùy chỉnh (mặc định là ImageExtList)\n */\nexport const isIncludeImageExtList = (ext: string, listExt: Array<string> = ImageExtList) => listExt.includes(ext);\n\n/** Danh sách extension/MIME type của video. */\nexport const VideoExtList = ['mp4', 'mov', 'mpg', 'avi', 'wmv', 'video/mp4', 'video/quicktime', 'video/mpeg', 'video/x-msvideo', 'video/x-ms-wmv'];\n\n/**\n * Kiểm tra extension có thuộc danh sách video không.\n * @param ext Extension hoặc MIME type cần kiểm tra\n */\nexport const isIncludeVideoExtList = (ext: string, listExt: Array<string> = VideoExtList) => listExt.includes(ext);\n\n/** Danh sách extension/MIME type của audio. */\nexport const AudioExtList = ['mp3', 'wav', 'ogg', 'aac', 'm4a', 'flac', 'wma', 'audio/mpeg', 'audio/wav', 'audio/ogg', 'audio/aac', 'audio/mp4', 'audio/flac', 'audio/x-ms-wma'];\n\n/**\n * Kiểm tra extension có thuộc danh sách audio không.\n * @param ext Extension hoặc MIME type cần kiểm tra\n */\nexport const isIncludeAudioExtList = (ext: string, listExt: Array<string> = AudioExtList) => listExt.includes(ext);\n\n/**\n * Lấy phần mở rộng (extension) của một file.\n * Hỗ trợ cả đối tượng File gốc của Browser và IFile (interface nội bộ).\n * @param file Đối tượng File hoặc IFile\n * @returns Extension dạng chữ thường (vd: \"jpg\", \"pdf\") hoặc undefined\n */\nexport const getFileExtension = (file: IFile | File) => {\n  if (file instanceof File && file.type) {\n    return (file.type.split('/').pop() || '').toLowerCase();\n  }\n  file = file as IFile;\n  if (file.mimetype || file.file?.type) {\n    return file.mimetype || file.file?.type;\n  }\n\n  let fileName = file.name as string;\n  const url = get(file, 'url') || get(file, 'origin_url');\n  if (!fileName && !url) {\n    return;\n  }\n  if (!fileName && url) {\n    fileName = url.split('/').pop() || '';\n    set(file, 'name', fileName);\n  }\n  const dots = fileName.split('.');\n\n  if (!dots) {\n    return;\n  }\n\n  return (dots.pop() || '').toLowerCase();\n};\n\n/**\n * Định dạng kích thước file dạng chuỗi dễ đọc (KB hoặc MB).\n * @param size Kích thước file tính bằng byte\n * @param toFixed Số chữ số thập phân (mặc định: 2)\n * @returns Chuỗi định dạng (vd: \" 1.50 MB\", \" 512.00 KB\")\n */\nexport const getLabelBySizeFile = (size: number, toFixed = 2) => {\n  if (size < 1024 * 1024) {\n    return ` ${(size / 1024).toFixed(toFixed)} KB`;\n  }\n\n  return ` ${(size / (1024 * 1024)).toFixed(toFixed)} MB`;\n};\n\n/**\n * Chuyển đổi đối tượng Blob thành File.\n * Tự động xác định phần mở rộng từ MIME type của Blob.\n * @param blob Đối tượng Blob cần chuyển\n * @param fileName Tên file mong muốn (mặc định: UUID ngẫu nhiên)\n * @returns Đối tượng File tương ứng\n */\nexport const convertBlobToFile = (blob: Blob, fileName?: string): File => {\n  const type = blob.type.split('/')[1];\n  const name = fileName ? `${fileName.split('.')[0]}.${type}` : `${uuid()}.${type}`;\n\n  return new File([blob], name, { type: blob.type, lastModified: Date.now() });\n};\n\n/**\n * Tải file từ URL và chuyển về đối tượng File.\n * Sử dụng XHR + FileReader để đọc nội dung, sau đó chuyển Base64 → Blob → File.\n * @param url URL của file cần tải\n * @param fileName Tên file mong muốn (mặc định: timestamp)\n * @returns Promise trả về File hoặc undefined nếu thất bại\n */\nexport const convertUrlToFile = (url: string, fileName?: string) => {\n  return new Promise<File | undefined>((resolve, reject) => {\n    const xhr = new XMLHttpRequest();\n    xhr.onload = function () {\n      const reader = new FileReader();\n      reader.onloadend = () => {\n        const file = convertBase64ToBlob(reader.result as string);\n        resolve(convertBlobToFile(file, fileName || Date.now().toLocaleString()));\n      };\n      reader.onerror = () => resolve(undefined);\n      reader.readAsDataURL(xhr.response);\n    };\n    xhr.onerror = (err) => {\n      reject(err);\n    };\n    xhr.open('GET', url);\n    xhr.responseType = 'blob';\n    xhr.send();\n  });\n};\n"]}
@@ -1,6 +1,17 @@
1
1
  import { UtilsCache } from './cache';
2
2
  import { isNil } from './helpers';
3
3
  import { UtilsLanguageConstants } from './language';
4
+ /**
5
+ * Chuẩn hóa chuỗi số theo locale hiện tại về dạng số thực (parseable).
6
+ * - Với **EN**: Xóa dấu phẩy (`,`) — dấu phân tách hàng nghìn.
7
+ * - Với **VI**: Xóa dấu chấm (`.`) và đổi dấu phẩy (`,`) thành dấu chấm (`.`).
8
+ *
9
+ * @param value Chuỗi hoặc số cần chuẩn hóa
10
+ * @returns Chuỗi số đã được chuẩn hóa (không dấu phân cách locale)
11
+ * @example
12
+ * // EN: "1,234,567" → "1234567"
13
+ * // VI: "1.234.567,89" → "1234567.89"
14
+ */
4
15
  export const formatNumber = (value) => {
5
16
  const lang = UtilsCache.getLang();
6
17
  if (lang === UtilsLanguageConstants.EN) {
@@ -8,9 +19,26 @@ export const formatNumber = (value) => {
8
19
  }
9
20
  return `${value}`.replace(/[.]/g, '').replace(/[,]/g, '.');
10
21
  };
22
+ /**
23
+ * Định dạng và hiển thị số theo ngôn ngữ (VI/EN) với các tùy chọn linh hoạt.
24
+ * Xử lý số âm, phần thập phân, làm tròn và dấu phân cách hàng nghìn.
25
+ *
26
+ * @param value Giá trị số hoặc chuỗi số đầu vào
27
+ * @param acceptNegativeValue Có chấp nhận số âm không (false → trả về 0 nếu value <= 0)
28
+ * @param parseFixed Số chữ số thập phân muốn giữ lại (mặc định: 1)
29
+ * @param ignoreFormatSeparator Bỏ qua bước chuẩn hóa dấu phân cách (khi value đã là số thuần)
30
+ * @param ignoreParseFloat Bỏ qua bước làm tròn (giữ nguyên phần thập phân)
31
+ * @param lang Ngôn ngữ hiển thị ('vi' hoặc 'en'), mặc định lấy từ cache
32
+ * @returns Chuỗi số đã format theo locale (vd: "1.234.567,89" hoặc "1,234,567.89")
33
+ * @example
34
+ * // VI, 2 chữ số thập phân
35
+ * viewDataNumberByLanguage(1234567.891, true, 2) → "1.234.567,89"
36
+ * // EN, 2 chữ số thập phân
37
+ * viewDataNumberByLanguage(1234567.891, true, 2, false, false, 'en') → "1,234,567.89"
38
+ */
11
39
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
12
- export const viewDataNumberByLanguage = (value, acceptNegativeValue, parseFixed = 1, ignoreFormatSeparator = false, ignoreParseFloat = false) => {
13
- const lang = UtilsCache.getLang();
40
+ export const viewDataNumberByLanguage = (value, acceptNegativeValue, parseFixed = 1, ignoreFormatSeparator = false, ignoreParseFloat = false, lang) => {
41
+ lang = lang || UtilsCache.getLang();
14
42
  if (!`${value}`.trim()) {
15
43
  return '';
16
44
  }
@@ -63,4 +91,4 @@ export const viewDataNumberByLanguage = (value, acceptNegativeValue, parseFixed
63
91
  .replace(/\B(?=(\d{3})+(?!\d))/g, '.');
64
92
  }
65
93
  };
66
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybWF0LW51bWJlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2xpYnMtdWkvdXRpbHMvc3JjL2Zvcm1hdC1udW1iZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLFNBQVMsQ0FBQztBQUNyQyxPQUFPLEVBQUUsS0FBSyxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBQ2xDLE9BQU8sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLFlBQVksQ0FBQztBQUVwRCxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsQ0FBQyxLQUFzQixFQUFVLEVBQUU7SUFDN0QsTUFBTSxJQUFJLEdBQUcsVUFBVSxDQUFDLE9BQU8sRUFBRSxDQUFDO0lBRWxDLElBQUksSUFBSSxLQUFLLHNCQUFzQixDQUFDLEVBQUUsRUFBRSxDQUFDO1FBQ3ZDLE9BQU8sR0FBRyxLQUFLLEVBQUUsQ0FBQyxPQUFPLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxDQUFDO0lBQ3hDLENBQUM7SUFFRCxPQUFPLEdBQUcsS0FBSyxFQUFFLENBQUMsT0FBTyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsQ0FBQyxPQUFPLENBQUMsTUFBTSxFQUFFLEdBQUcsQ0FBQyxDQUFDO0FBQzdELENBQUMsQ0FBQztBQUVGLDhEQUE4RDtBQUM5RCxNQUFNLENBQUMsTUFBTSx3QkFBd0IsR0FBRyxDQUFDLEtBQXNCLEVBQUUsbUJBQXdDLEVBQUUsVUFBVSxHQUFHLENBQUMsRUFBRSxxQkFBcUIsR0FBRyxLQUFLLEVBQUUsZ0JBQWdCLEdBQUcsS0FBSyxFQUFPLEVBQUU7SUFDekwsTUFBTSxJQUFJLEdBQUcsVUFBVSxDQUFDLE9BQU8sRUFBRSxDQUFDO0lBRWxDLElBQUksQ0FBQyxHQUFHLEtBQUssRUFBRSxDQUFDLElBQUksRUFBRSxFQUFFLENBQUM7UUFDdkIsT0FBTyxFQUFFLENBQUM7SUFDWixDQUFDO0lBQ0QsSUFBSSxLQUFLLENBQUMsS0FB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QU8sQ0FBQyx1QkFBdUIsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUVsRSxPQUFPLEdBQUcsR0FBRyxHQUFHLFdBQVcsRUFBRSxDQUFDO0lBQ2hDLENBQUM7SUFDRCxJQUFJLEtBQUssS0FBSyxDQUFDLENBQUMsSUFBSSxVQUFVLElBQUksQ0FBQyxFQUFFLENBQUM7UUFDcEMsSUFBSSxJQUFJLEtBQUssc0JBQXNCLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDdkMsT0FBTyxLQUFLLENBQUMsUUFBUSxFQUFFLENBQUMsT0FBTyxDQUFDLHVCQUF1QixFQUFFLEdBQUcsQ0FBQyxDQUFDO1FBQ2hFLENBQUM7UUFFRCxPQUFPLEtBQUs7YUFDVCxRQUFRLEVBQUU7YUFDVixPQUFPLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQzthQUNqQixPQUFPLENBQUMsdUJBQXVCLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDM0MsQ0FBQztBQUNILENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFV0aWxzQ2FjaGUgfSBmcm9tICcuL2NhY2hlJztcbmltcG9ydCB7IGlzTmlsIH0gZnJvbSAnLi9oZWxwZXJzJztcbmltcG9ydCB7IFV0aWxzTGFuZ3VhZ2VDb25zdGFudHMgfSBmcm9tICcuL2xhbmd1YWdlJztcblxuZXhwb3J0IGNvbnN0IGZvcm1hdE51bWJlciA9ICh2YWx1ZTogc3RyaW5nIHwgbnVtYmVyKTogc3RyaW5nID0+IHtcbiAgY29uc3QgbGFuZyA9IFV0aWxzQ2FjaGUuZ2V0TGFuZygpO1xuXG4gIGlmIChsYW5nID09PSBVdGlsc0xhbmd1YWdlQ29uc3RhbnRzLkVOKSB7XG4gICAgcmV0dXJuIGAke3ZhbHVlfWAucmVwbGFjZSgvWyxdL2csICcnKTtcbiAgfVxuXG4gIHJldHVybiBgJHt2YWx1ZX1gLnJlcGxhY2UoL1suXS9nLCAnJykucmVwbGFjZSgvWyxdL2csICcuJyk7XG59O1xuXG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLWV4cGxpY2l0LWFueVxuZXhwb3J0IGNvbnN0IHZpZXdEYXRhTnVtYmVyQnlMYW5ndWFnZSA9ICh2YWx1ZTogc3RyaW5nIHwgbnVtYmVyLCBhY2NlcHROZWdhdGl2ZVZhbHVlOiBib29sZWFuIHwgdW5kZWZpbmVkLCBwYXJzZUZpeGVkID0gMSwgaWdub3JlRm9ybWF0U2VwYXJhdG9yID0gZmFsc2UsIGlnbm9yZVBhcnNlRmxvYXQgPSBmYWxzZSk6IGFueSA9PiB7XG4gIGNvbnN0IGxhbmcgPSBVdGlsc0NhY2hlLmdldExhbmcoKTtcblxuICBpZiAoIWAke3ZhbHVlfWAudHJpbSgpKSB7XG4gICAgcmV0dXJuICcnO1xuICB9XG4gIGlmIChpc05pbCh2YWx1ZSkgfHwgKCFhY2NlcHROZWdhdGl2ZVZhbHVlICYmICt2YWx1ZSA8PSAwKSkge1xuICAgIHJldHVybiAwO1xuICB9XG4gIGlmICh0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnICYmICFpZ25vcmVGb3JtYXRTZXBhcmF0b3IpIHtcbiAgICB2YWx1ZSA9IGZvcm1hdE51bWJlcih2YWx1ZSk7XG4gIH1cblxuICBjb25zdCBpc0Zsb2F0ID0gdmFsdWUudG9TdHJpbmcoKS5pbmNsdWRlcygnLicpO1xuICBjb25zdCBbaW50U3RyLCBmbG9hdFN0cl0gPSB2YWx1ZS50b1N0cmluZygpLnNwbGl0KCcuJyk7XG5cbiAgaWYgKCFmbG9hdFN0cj8ubGVuZ3RoKSB7XG4gICAgcGFyc2VGaXhlZCA9IDA7XG4gIH1cbiAgY29uc3Qgcm91bmQgPSAodmFsdWU6IG51bWJlciwgcGFyc2VGaXhlZDogbnVtYmVyKSA9PiB7XG4gICAgY29uc3QgZml4ZWQgPSBNYXRoLnBvdygxMCwgcGFyc2VGaXhlZCk7XG5cbiAgICByZXR1cm4gTWF0aC5yb3VuZCh2YWx1ZSAqIGZpeGVkKSAvIGZpeGVkO1xuICB9O1xuXG4gIGlmIChwYXJzZUZpeGVkID4gKGZsb2F0U3RyPy5sZW5ndGggfHwgMCkpIHtcbiAgICBjb25zdCBtYXhQYXJzZUZpeGVkID0gYWNjZXB0TmVnYXRpdmVWYWx1ZSAmJiArdmFsdWUgPCAwID8gMTcgOiAxNjtcbiAgICBjb25zdCBmaXhlZCA9IG1heFBhcnNlRml4ZWQgLSAoaW50U3RyPy5sZW5ndGggfHwgMCk7XG5cbiAgICBwYXJzZUZpeGVkID0gcGFyc2VGaXhlZCA8IGZpeGVkID8gcGFyc2VGaXhlZCA6IGZpeGVkO1xuICB9XG5cbiAgaWYgKGlzRmxvYXQgJiYgIWlnbm9yZVBhcnNlRmxvYXQpIHtcbiAgICB2YWx1ZSA9IHBhcnNlRmxvYXQocm91bmQoK3ZhbHVlLCBwYXJzZUZpeGVkKS50b0ZpeGVkKHBhcnNlRml4ZWQpKTtcbiAgfVxuXG4gIC8vIFjhu60gbMO9IGtow7RuZyBmb3JtYXQgZOG7ryBsaeG7h3UgcGjhuqduIHRo4bqtcCBwaMOiblxuICBjb25zdCB2YWx1ZVN0cmluZyA9IHZhbHVlLnRvU3RyaW5nKCk7XG4gIGNvbnN0IGluZGV4ID0gdmFsdWVTdHJpbmcuaW5kZXhPZignLicpO1xuXG4gIGlmIChpbmRleCAhPT0gLTEgJiYgKHBhcnNlRml4ZWQgPiAzIHx8IGlnbm9yZVBhcnNlRmxvYXQpKSB7XG4gICAgbGV0IGRlY2ltYWxQYXJ0ID0gdmFsdWVTdHJpbmcuc3Vic3RyaW5nKGluZGV4ICsgMSwgdmFsdWVTdHJpbmcubGVuZ3RoKTtcbiAgICBsZXQgaW50ID0gdmFsdWVTdHJpbmcuc3Vic3RyaW5nKDAsIGluZGV4ICsgMSk7XG5cbiAgICBpZiAoaWdub3JlUGFyc2VGbG9hdCkge1xuICAgICAgZGVjaW1hbFBhcnQgPSBkZWNpbWFsUGFydC5zdWJzdHJpbmcoMCwgcGFyc2VGaXhlZCk7XG4gICAgfVxuICAgIGlmIChsYW5nID09PSBVdGlsc0xhbmd1YWdlQ29uc3RhbnRzLkVOKSB7XG4gICAgICBpbnQgPSBpbnQucmVwbGFjZSgvXFxCKD89KFxcZHszfSkrKD8hXFxkKSkvZywgJywnKTtcblxuICAgICAgcmV0dXJuIGAke2ludH0ke2RlY2ltYWxQYXJ0fWA7XG4gICAgfVxuICAgIGludCA9IGludC5yZXBsYWNlKCcuJywgJywnKS5yZXBsYWNlKC9cXEIoPz0oXFxkezN9KSsoPyFcXGQpKS9nLCAnLicpO1xuXG4gICAgcmV0dXJuIGAke2ludH0ke2RlY2ltYWxQYXJ0fWA7XG4gIH1cbiAgaWYgKGluZGV4ID09PSAtMSB8fCBwYXJzZUZpeGVkIDw9IDMpIHtcbiAgICBpZiAobGFuZyA9PT0gVXRpbHNMYW5ndWFnZUNvbnN0YW50cy5FTikge1xuICAgICAgcmV0dXJuIHZhbHVlLnRvU3RyaW5nKCkucmVwbGFjZSgvXFxCKD89KFxcZHszfSkrKD8hXFxkKSkvZywgJywnKTtcbiAgICB9XG5cbiAgICByZXR1cm4gdmFsdWVcbiAgICAgIC50b1N0cmluZygpXG4gICAgICAucmVwbGFjZSgnLicsICcsJylcbiAgICAgIC5yZXBsYWNlKC9cXEIoPz0oXFxkezN9KSsoPyFcXGQpKS9nLCAnLicpO1xuICB9XG59O1xuIl19
94
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"format-number.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/format-number.ts"],"names":[],"mappings":"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","sourcesContent":["import { UtilsCache } from './cache';\nimport { isNil } from './helpers';\nimport { UtilsLanguageConstants } from './language';\n\n/**\n * Chuẩn hóa chuỗi số theo locale hiện tại về dạng số thực (parseable).\n * - Với **EN**: Xóa dấu phẩy (`,`) — dấu phân tách hàng nghìn.\n * - Với **VI**: Xóa dấu chấm (`.`) và đổi dấu phẩy (`,`) thành dấu chấm (`.`).\n *\n * @param value Chuỗi hoặc số cần chuẩn hóa\n * @returns Chuỗi số đã được chuẩn hóa (không dấu phân cách locale)\n * @example\n * // EN: \"1,234,567\" → \"1234567\"\n * // VI: \"1.234.567,89\" → \"1234567.89\"\n */\nexport const formatNumber = (value: string | number): string => {\n  const lang = UtilsCache.getLang();\n\n  if (lang === UtilsLanguageConstants.EN) {\n    return `${value}`.replace(/[,]/g, '');\n  }\n\n  return `${value}`.replace(/[.]/g, '').replace(/[,]/g, '.');\n};\n\n/**\n * Định dạng và hiển thị số theo ngôn ngữ (VI/EN) với các tùy chọn linh hoạt.\n * Xử lý số âm, phần thập phân, làm tròn và dấu phân cách hàng nghìn.\n *\n * @param value Giá trị số hoặc chuỗi số đầu vào\n * @param acceptNegativeValue Có chấp nhận số âm không (false → trả về 0 nếu value <= 0)\n * @param parseFixed Số chữ số thập phân muốn giữ lại (mặc định: 1)\n * @param ignoreFormatSeparator Bỏ qua bước chuẩn hóa dấu phân cách (khi value đã là số thuần)\n * @param ignoreParseFloat Bỏ qua bước làm tròn (giữ nguyên phần thập phân)\n * @param lang Ngôn ngữ hiển thị ('vi' hoặc 'en'), mặc định lấy từ cache\n * @returns Chuỗi số đã format theo locale (vd: \"1.234.567,89\" hoặc \"1,234,567.89\")\n * @example\n * // VI, 2 chữ số thập phân\n * viewDataNumberByLanguage(1234567.891, true, 2) → \"1.234.567,89\"\n * // EN, 2 chữ số thập phân\n * viewDataNumberByLanguage(1234567.891, true, 2, false, false, 'en') → \"1,234,567.89\"\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport const viewDataNumberByLanguage = (value: string | number, acceptNegativeValue: boolean | undefined, parseFixed = 1, ignoreFormatSeparator = false, ignoreParseFloat = false, lang?: string): any => {\n  lang = lang || UtilsCache.getLang();\n\n  if (!`${value}`.trim()) {\n    return '';\n  }\n  if (isNil(value) || (!acceptNegativeValue && +value <= 0)) {\n    return 0;\n  }\n  if (typeof value === 'string' && !ignoreFormatSeparator) {\n    value = formatNumber(value);\n  }\n\n  const isFloat = value.toString().includes('.');\n  const [intStr, floatStr] = value.toString().split('.');\n\n  if (!floatStr?.length) {\n    parseFixed = 0;\n  }\n  const round = (value: number, parseFixed: number) => {\n    const fixed = Math.pow(10, parseFixed);\n\n    return Math.round(value * fixed) / fixed;\n  };\n\n  if (parseFixed > (floatStr?.length || 0)) {\n    const maxParseFixed = acceptNegativeValue && +value < 0 ? 17 : 16;\n    const fixed = maxParseFixed - (intStr?.length || 0);\n\n    parseFixed = parseFixed < fixed ? parseFixed : fixed;\n  }\n\n  if (isFloat && !ignoreParseFloat) {\n    value = parseFloat(round(+value, parseFixed).toFixed(parseFixed));\n  }\n\n  // Xử lý không format dữ liệu phần thập phân\n  const valueString = value.toString();\n  const index = valueString.indexOf('.');\n\n  if (index !== -1 && (parseFixed > 3 || ignoreParseFloat)) {\n    let decimalPart = valueString.substring(index + 1, valueString.length);\n    let int = valueString.substring(0, index + 1);\n\n    if (ignoreParseFloat) {\n      decimalPart = decimalPart.substring(0, parseFixed);\n    }\n    if (lang === UtilsLanguageConstants.EN) {\n      int = int.replace(/\\B(?=(\\d{3})+(?!\\d))/g, ',');\n\n      return `${int}${decimalPart}`;\n    }\n    int = int.replace('.', ',').replace(/\\B(?=(\\d{3})+(?!\\d))/g, '.');\n\n    return `${int}${decimalPart}`;\n  }\n  if (index === -1 || parseFixed <= 3) {\n    if (lang === UtilsLanguageConstants.EN) {\n      return value.toString().replace(/\\B(?=(\\d{3})+(?!\\d))/g, ',');\n    }\n\n    return value\n      .toString()\n      .replace('.', ',')\n      .replace(/\\B(?=(\\d{3})+(?!\\d))/g, '.');\n  }\n};\n"]}