@libs-ui/utils 0.2.355-8 → 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.
- package/README.md +633 -2
- package/base64.d.ts +25 -0
- package/cache.d.ts +73 -0
- package/collection.d.ts +12 -0
- package/color.d.ts +37 -0
- package/communicate-micro.d.ts +34 -1
- package/crypto-3rd.d.ts +12 -0
- package/crypto.d.ts +17 -0
- package/dangerous-object.d.ts +8 -1
- package/data.d.ts +5 -3
- package/date.d.ts +12 -0
- package/dom.d.ts +56 -0
- package/download.d.ts +21 -0
- package/esm2022/base64.mjs +27 -2
- package/esm2022/cache.mjs +74 -1
- package/esm2022/collection.mjs +13 -1
- package/esm2022/color.mjs +38 -1
- package/esm2022/communicate-micro.mjs +37 -5
- package/esm2022/crypto-3rd.mjs +13 -1
- package/esm2022/crypto.mjs +18 -1
- package/esm2022/dangerous-object.mjs +11 -2
- package/esm2022/data.mjs +6 -4
- package/esm2022/date.mjs +13 -1
- package/esm2022/dom.mjs +57 -1
- package/esm2022/download.mjs +22 -1
- package/esm2022/file.mjs +54 -1
- package/esm2022/format-number.mjs +31 -3
- package/esm2022/format-text.mjs +73 -2
- package/esm2022/function-check-embed-frame.mjs +28 -1
- package/esm2022/pattern.mjs +83 -3
- package/esm2022/trace.mjs +8 -1
- package/esm2022/two-way-signal-object.mjs +14 -1
- package/esm2022/uri.mjs +22 -3
- package/esm2022/url-search-params.mjs +68 -1
- package/esm2022/url.mjs +10 -1
- package/esm2022/uuid.mjs +7 -1
- package/esm2022/xss-filter.mjs +14 -1
- package/fesm2022/libs-ui-utils.mjs +723 -21
- package/fesm2022/libs-ui-utils.mjs.map +1 -1
- package/file.d.ts +53 -0
- package/format-number.d.ts +29 -1
- package/format-text.d.ts +71 -0
- package/function-check-embed-frame.d.ts +22 -0
- package/package.json +2 -2
- package/pattern.d.ts +81 -1
- package/trace.d.ts +7 -0
- package/two-way-signal-object.d.ts +13 -0
- package/uri.d.ts +20 -0
- package/url-search-params.d.ts +67 -0
- package/url.d.ts +9 -0
- package/uuid.d.ts +6 -0
- 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"]}
|
package/esm2022/download.mjs
CHANGED
|
@@ -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,
|
|
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":"AACA,OAAO,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;AAC/C,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,MAAM,WAAW,CAAC;AACrC,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAE9B,MAAM,CAAC,MAAM,WAAW,GAAG,CAAC,IAAiB,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;AAC9F,MAAM,CAAC,MAAM,WAAW,GAAG,CAAC,IAAiB,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;AAC9F,MAAM,CAAC,MAAM,WAAW,GAAG,CAAC,IAAiB,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;AAC9F,MAAM,CAAC,MAAM,UAAU,GAAG,CAAC,IAAU,EAAE,EAAE,CAAC,CAAC,IAAI,YAAY,IAAI,IAAI,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;AAE5I,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,0BAA0B,EAAE,mEAAmE,CAAC,CAAC;AAC7I,MAAM,CAAC,MAAM,eAAe,GAAG;IAC7B,KAAK;IACL,MAAM;IACN,KAAK;IACL,MAAM;IACN,KAAK;IACL,MAAM;IACN,KAAK;IACL,MAAM;IACN,KAAK;IACL,oBAAoB;IACpB,yEAAyE;IACzE,0BAA0B;IAC1B,mEAAmE;IACnE,+BAA+B;IAC/B,2EAA2E;IAC3E,iBAAiB;IACjB,kBAAkB;IAClB,iBAAiB;CAClB,CAAC;AACF,MAAM,CAAC,MAAM,wBAAwB,GAAG,CAAC,GAAW,EAAE,UAAyB,eAAe,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,OAAO,CAAC,QAAQ,CAAC,eAAe,GAAG,EAAE,CAAC,CAAC;AAEnK,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,WAAW,EAAE,YAAY,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,CAAC,CAAC;AACxH,MAAM,CAAC,MAAM,qBAAqB,GAAG,CAAC,GAAW,EAAE,UAAyB,YAAY,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;AAEnH,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,WAAW,EAAE,iBAAiB,EAAE,YAAY,EAAE,iBAAiB,EAAE,gBAAgB,CAAC,CAAC;AACnJ,MAAM,CAAC,MAAM,qBAAqB,GAAG,CAAC,GAAW,EAAE,UAAyB,YAAY,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;AACnH,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,WAAW,EAAE,WAAW,EAAE,WAAW,EAAE,YAAY,EAAE,gBAAgB,CAAC,CAAC;AACjL,MAAM,CAAC,MAAM,qBAAqB,GAAG,CAAC,GAAW,EAAE,UAAyB,YAAY,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;AAEnH,MAAM,CAAC,MAAM,gBAAgB,GAAG,CAAC,IAAkB,EAAE,EAAE;IACrD,IAAI,IAAI,YAAY,IAAI,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;QACtC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;IAC1D,CAAC;IACD,IAAI,GAAG,IAAa,CAAC;IACrB,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC;QACrC,OAAO,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC;IAC1C,CAAC;IAED,IAAI,QAAQ,GAAG,IAAI,CAAC,IAAc,CAAC;IACnC,MAAM,GAAG,GAAG,GAAG,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,GAAG,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;IACxD,IAAI,CAAC,QAAQ,IAAI,CAAC,GAAG,EAAE,CAAC;QACtB,OAAO;IACT,CAAC;IACD,IAAI,CAAC,QAAQ,IAAI,GAAG,EAAE,CAAC;QACrB,QAAQ,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC;QACtC,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;IAC9B,CAAC;IACD,MAAM,IAAI,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAEjC,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO;IACT,CAAC;IAED,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC;AAC1C,CAAC,CAAC;AAEF,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAAC,IAAY,EAAE,OAAO,GAAG,CAAC,EAAE,EAAE;IAC9D,IAAI,IAAI,GAAG,IAAI,GAAG,IAAI,EAAE,CAAC;QACvB,OAAO,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC;IACjD,CAAC;IAED,OAAO,IAAI,CAAC,IAAI,GAAG,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC;AAC1D,CAAC,CAAC;AAEF,MAAM,CAAC,MAAM,iBAAiB,GAAG,CAAC,IAAU,EAAE,QAAiB,EAAQ,EAAE;IACvE,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IACrC,MAAM,IAAI,GAAG,QAAQ,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC,GAAG,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC;IAElF,OAAO,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,YAAY,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;AAC/E,CAAC,CAAC;AAEF,MAAM,CAAC,MAAM,gBAAgB,GAAG,CAAC,GAAW,EAAE,QAAiB,EAAE,EAAE;IACjE,OAAO,IAAI,OAAO,CAAmB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACvD,MAAM,GAAG,GAAG,IAAI,cAAc,EAAE,CAAC;QACjC,GAAG,CAAC,MAAM,GAAG;YACX,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAChC,MAAM,CAAC,SAAS,GAAG,GAAG,EAAE;gBACtB,MAAM,IAAI,GAAG,mBAAmB,CAAC,MAAM,CAAC,MAAgB,CAAC,CAAC;gBAC1D,OAAO,CAAC,iBAAiB,CAAC,IAAI,EAAE,QAAQ,IAAI,IAAI,CAAC,GAAG,EAAE,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;YAC5E,CAAC,CAAC;YACF,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;YAC1C,MAAM,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACrC,CAAC,CAAC;QACF,GAAG,CAAC,OAAO,GAAG,CAAC,GAAG,EAAE,EAAE;YACpB,MAAM,CAAC,GAAG,CAAC,CAAC;QACd,CAAC,CAAC;QACF,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;QACrB,GAAG,CAAC,YAAY,GAAG,MAAM,CAAC;QAC1B,GAAG,CAAC,IAAI,EAAE,CAAC;IACb,CAAC,CAAC,CAAC;AACL,CAAC,CAAC","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
|
-
|
|
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"]}
|