@libs-ui/utils 0.1.1-1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -0
- package/base64.d.ts +5 -0
- package/cache.d.ts +42 -0
- package/color.d.ts +11 -0
- package/communicate-micro.d.ts +16 -0
- package/constants.d.ts +10 -0
- package/crypto-3rd.d.ts +7 -0
- package/crypto.d.ts +8 -0
- package/dangerous-object.d.ts +79 -0
- package/date.d.ts +44 -0
- package/dom.d.ts +52 -0
- package/download.d.ts +3 -0
- package/esm2022/base64.mjs +43 -0
- package/esm2022/cache.mjs +388 -0
- package/esm2022/color.mjs +133 -0
- package/esm2022/communicate-micro.mjs +149 -0
- package/esm2022/constants.mjs +11 -0
- package/esm2022/crypto-3rd.mjs +38 -0
- package/esm2022/crypto.mjs +41 -0
- package/esm2022/dangerous-object.mjs +149 -0
- package/esm2022/date.mjs +191 -0
- package/esm2022/dom.mjs +256 -0
- package/esm2022/download.mjs +41 -0
- package/esm2022/file.mjs +90 -0
- package/esm2022/format-number.mjs +66 -0
- package/esm2022/format-text.mjs +149 -0
- package/esm2022/function-check-embed-frame.mjs +10 -0
- package/esm2022/get-smart-axis-scale.mjs +174 -0
- package/esm2022/helpers.mjs +651 -0
- package/esm2022/http-params.mjs +80 -0
- package/esm2022/index.mjs +30 -0
- package/esm2022/inject-token.mjs +5 -0
- package/esm2022/key-cache.mjs +31 -0
- package/esm2022/key-code.mjs +123 -0
- package/esm2022/language.mjs +70 -0
- package/esm2022/libs-ui-utils.mjs +5 -0
- package/esm2022/pattern.mjs +62 -0
- package/esm2022/random.mjs +42 -0
- package/esm2022/two-way-signal-object.mjs +131 -0
- package/esm2022/uri.mjs +25 -0
- package/esm2022/url-search-params.mjs +99 -0
- package/esm2022/uuid.mjs +18 -0
- package/esm2022/xss-filter.mjs +10 -0
- package/fesm2022/libs-ui-utils.mjs +3234 -0
- package/fesm2022/libs-ui-utils.mjs.map +1 -0
- package/file.d.ts +18 -0
- package/format-number.d.ts +2 -0
- package/format-text.d.ts +11 -0
- package/function-check-embed-frame.d.ts +2 -0
- package/get-smart-axis-scale.d.ts +34 -0
- package/helpers.d.ts +270 -0
- package/http-params.d.ts +37 -0
- package/index.d.ts +29 -0
- package/inject-token.d.ts +4 -0
- package/key-cache.d.ts +1 -0
- package/key-code.d.ts +122 -0
- package/language.d.ts +37 -0
- package/package.json +29 -0
- package/pattern.d.ts +20 -0
- package/random.d.ts +3 -0
- package/two-way-signal-object.d.ts +15 -0
- package/uri.d.ts +5 -0
- package/url-search-params.d.ts +25 -0
- package/uuid.d.ts +1 -0
- package/xss-filter.d.ts +3 -0
package/esm2022/date.mjs
ADDED
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
import { isSignal } from '@angular/core';
|
|
2
|
+
import dayjs from 'dayjs';
|
|
3
|
+
import 'dayjs/locale/en';
|
|
4
|
+
import 'dayjs/locale/vi';
|
|
5
|
+
import customParseFormat from 'dayjs/plugin/customParseFormat';
|
|
6
|
+
import localeData from 'dayjs/plugin/localeData';
|
|
7
|
+
import timezone from 'dayjs/plugin/timezone';
|
|
8
|
+
import updateLocale from 'dayjs/plugin/updateLocale';
|
|
9
|
+
import utc from 'dayjs/plugin/utc';
|
|
10
|
+
import { UtilsCache } from './cache';
|
|
11
|
+
dayjs.extend(localeData);
|
|
12
|
+
dayjs.extend(updateLocale);
|
|
13
|
+
dayjs.extend(utc);
|
|
14
|
+
dayjs.extend(timezone);
|
|
15
|
+
dayjs.extend(customParseFormat);
|
|
16
|
+
let timeZoneSetup = 'Asia/Ho_Chi_Minh';
|
|
17
|
+
export const setDefaultTimeZone = (localeZone = timeZoneSetup) => {
|
|
18
|
+
timeZoneSetup = localeZone;
|
|
19
|
+
dayjs.tz.setDefault(localeZone);
|
|
20
|
+
};
|
|
21
|
+
let functionFormatDate = undefined;
|
|
22
|
+
export const updateFunctionFormatDate = (functionCustom) => {
|
|
23
|
+
functionFormatDate = functionCustom;
|
|
24
|
+
};
|
|
25
|
+
/**
|
|
26
|
+
* @description Lấy đối tượng dayjs theo config
|
|
27
|
+
* @param config nếu không có config sẽ trả về đối tượng dayjs là thời gian hiện tại
|
|
28
|
+
* @param config.date thời gian cần lấy
|
|
29
|
+
* @param config.returnDayjsIfConfigDateNotExist true nếu muốn trả về đối tượng dayjs nếu config.date không có
|
|
30
|
+
* @param config.utc true nếu muốn lấy thời gian UTC
|
|
31
|
+
* @param config.formatOfDate định dạng thời gian đang được truyền vào
|
|
32
|
+
*/
|
|
33
|
+
export const getDayjs = (config) => {
|
|
34
|
+
// helper cast để tránh lặp lại kiểu điều kiện ở các return
|
|
35
|
+
const out = (v) => v;
|
|
36
|
+
if (!config) {
|
|
37
|
+
return out(dayjs().tz());
|
|
38
|
+
}
|
|
39
|
+
config.date = !config.date && config.returnDayjsIfConfigDateNotExist ? dayjs().tz() : config.date;
|
|
40
|
+
if (typeof config.date === 'number') {
|
|
41
|
+
config.date = dayjs.unix(config.date);
|
|
42
|
+
}
|
|
43
|
+
if (!config.date) {
|
|
44
|
+
return out(undefined);
|
|
45
|
+
}
|
|
46
|
+
let { date, utc, formatOfDate } = config;
|
|
47
|
+
while (isSignal(date)) {
|
|
48
|
+
date = date();
|
|
49
|
+
}
|
|
50
|
+
while (isSignal(utc)) {
|
|
51
|
+
utc = utc();
|
|
52
|
+
}
|
|
53
|
+
while (isSignal(formatOfDate)) {
|
|
54
|
+
formatOfDate = formatOfDate();
|
|
55
|
+
}
|
|
56
|
+
if (utc) {
|
|
57
|
+
if (formatOfDate) {
|
|
58
|
+
return out(dayjs(date, formatOfDate).utc());
|
|
59
|
+
}
|
|
60
|
+
const dateInputIsUTC = (dayjs.isDayjs(date) && date.isUTC()) || (typeof date === 'string' && date.includes('Z'));
|
|
61
|
+
if (dateInputIsUTC) {
|
|
62
|
+
return out(dayjs(date));
|
|
63
|
+
}
|
|
64
|
+
return out(dayjs(date).utc());
|
|
65
|
+
}
|
|
66
|
+
if (typeof date === 'string' && !date.includes('Z') && !date.includes('+')) {
|
|
67
|
+
return out(formatOfDate ? dayjs.tz(date, formatOfDate, config.localeZone || timeZoneSetup) : dayjs.tz(date, config.localeZone || timeZoneSetup));
|
|
68
|
+
}
|
|
69
|
+
return out((formatOfDate ? dayjs(date, formatOfDate) : dayjs(date)).tz());
|
|
70
|
+
};
|
|
71
|
+
/**
|
|
72
|
+
* @description Kiểm tra xem hai ngày có khác nhau không (khác ngày trong tuần)
|
|
73
|
+
* @param date1 ngày thứ nhất cần so sánh
|
|
74
|
+
* @param date2 ngày thứ hai cần so sánh
|
|
75
|
+
* @returns true nếu hai ngày khác nhau, false nếu giống nhau hoặc không thể so sánh
|
|
76
|
+
*/
|
|
77
|
+
export const isDifferenceDay = (date1, date2) => {
|
|
78
|
+
if (isDifferenceMonth(date1, date2)) {
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
const day1 = getDayjs({ date: date1 })?.day();
|
|
82
|
+
const day2 = getDayjs({ date: date2 })?.day();
|
|
83
|
+
return day1 !== day2;
|
|
84
|
+
};
|
|
85
|
+
/**
|
|
86
|
+
* @description Kiểm tra xem hai ngày có khác tháng không
|
|
87
|
+
* @param date1 ngày thứ nhất cần so sánh
|
|
88
|
+
* @param date2 ngày thứ hai cần so sánh
|
|
89
|
+
* @returns true nếu hai ngày khác tháng, false nếu cùng tháng hoặc không thể so sánh
|
|
90
|
+
*/
|
|
91
|
+
export const isDifferenceMonth = (date1, date2) => {
|
|
92
|
+
if (isDifferenceYear(date1, date2)) {
|
|
93
|
+
return true;
|
|
94
|
+
}
|
|
95
|
+
const month1 = getDayjs({ date: date1 })?.month();
|
|
96
|
+
const month2 = getDayjs({ date: date2 })?.month();
|
|
97
|
+
return month1 !== month2;
|
|
98
|
+
};
|
|
99
|
+
/**
|
|
100
|
+
* @description Kiểm tra xem hai ngày có khác năm không
|
|
101
|
+
* @param date1 ngày thứ nhất cần so sánh
|
|
102
|
+
* @param date2 ngày thứ hai cần so sánh
|
|
103
|
+
* @returns true nếu hai ngày khác năm, false nếu cùng năm hoặc không thể so sánh
|
|
104
|
+
*/
|
|
105
|
+
export const isDifferenceYear = (date1, date2) => {
|
|
106
|
+
const year1 = getDayjs({ date: date1 })?.year();
|
|
107
|
+
const year2 = getDayjs({ date: date2 })?.year();
|
|
108
|
+
return year1 !== year2;
|
|
109
|
+
};
|
|
110
|
+
/**
|
|
111
|
+
* @description Lấy ra chuỗi thời gian hiển thị theo định dạng và ngôn ngữ
|
|
112
|
+
* @param date thời gian cần định dạng
|
|
113
|
+
* @param format định dạng thời gian muốn lấy ra
|
|
114
|
+
* @param lang lấy theo ngôn ngữ
|
|
115
|
+
*/
|
|
116
|
+
export const formatDate = (date, formatOutput = 'YYYY/MM/DD HH:mm', lang, formatInput) => {
|
|
117
|
+
if (!date) {
|
|
118
|
+
return '';
|
|
119
|
+
}
|
|
120
|
+
lang = lang || UtilsCache.getLang();
|
|
121
|
+
if (functionFormatDate) {
|
|
122
|
+
return functionFormatDate(date, formatOutput, lang);
|
|
123
|
+
}
|
|
124
|
+
date = getDayjs({ date, formatOfDate: formatInput }).locale(lang);
|
|
125
|
+
if (lang === 'vi') {
|
|
126
|
+
dayjs.updateLocale('vi', {
|
|
127
|
+
monthsShort: 'Thg 1_Thg 2_Thg 3_Thg 4_Thg 5_Thg 6_Thg 7_Thg 8_Thg 9_Thg 10_Thg 11_Thg 12'.split('_'),
|
|
128
|
+
});
|
|
129
|
+
}
|
|
130
|
+
return date.format(getFormatData(getTypeByFormat(formatOutput), lang)) || '';
|
|
131
|
+
};
|
|
132
|
+
const getTypeByFormat = (format) => {
|
|
133
|
+
if (format === 'dm' || format === 'dmy' || format === 'dmy hm' || format === 'my') {
|
|
134
|
+
return format;
|
|
135
|
+
}
|
|
136
|
+
if (format === 'DD/MM/YYYY' || format === 'YYYY-MM-DD' || format === 'dd/MM/yyyy' || format === 'dd-MM-yyyy' || format === 'dd/mm/yyyy') {
|
|
137
|
+
return 'dmy';
|
|
138
|
+
}
|
|
139
|
+
if (format === 'MM-DD' || format === 'dd/MM' || format === 'dd/mm') {
|
|
140
|
+
return 'dm';
|
|
141
|
+
}
|
|
142
|
+
if (format === 'M/YYYY' || format === 'YYYY-MM' || format === 'MM/yyyy') {
|
|
143
|
+
return 'my';
|
|
144
|
+
}
|
|
145
|
+
if (format === 'YYYY/MM/DD hh:mm:ss' || format === 'dmy hms' || format === 'dd/mm/yyyy hh:mm:ss') {
|
|
146
|
+
return 'dmy hms';
|
|
147
|
+
}
|
|
148
|
+
if (format === 'hh:mm' || format === 'HH:mm') {
|
|
149
|
+
return 'hh:mm';
|
|
150
|
+
}
|
|
151
|
+
if (format === 'hh:mm A' || format === 'HH:mm A') {
|
|
152
|
+
return 'hh:mm A';
|
|
153
|
+
}
|
|
154
|
+
return 'dmy hm';
|
|
155
|
+
};
|
|
156
|
+
const getFormatData = (type, lang) => {
|
|
157
|
+
switch (type) {
|
|
158
|
+
case 'dm':
|
|
159
|
+
if (lang === 'vi') {
|
|
160
|
+
return 'D MMM';
|
|
161
|
+
}
|
|
162
|
+
return 'MMM D';
|
|
163
|
+
case 'dmy':
|
|
164
|
+
if (lang === 'vi') {
|
|
165
|
+
return 'D MMM, YYYY';
|
|
166
|
+
}
|
|
167
|
+
return 'MMM D, YYYY';
|
|
168
|
+
case 'dmy hm':
|
|
169
|
+
if (lang === 'vi') {
|
|
170
|
+
return 'D MMM, YYYY HH:mm';
|
|
171
|
+
}
|
|
172
|
+
return 'MMM D, YYYY HH:mm';
|
|
173
|
+
case 'my':
|
|
174
|
+
if (lang === 'vi') {
|
|
175
|
+
return 'MMM, YYYY';
|
|
176
|
+
}
|
|
177
|
+
return 'MMM YYYY ';
|
|
178
|
+
case 'dmy hms':
|
|
179
|
+
if (lang === 'vi') {
|
|
180
|
+
return 'D MMM, YYYY HH:mm:ss';
|
|
181
|
+
}
|
|
182
|
+
return 'MMM D, YYYY HH:mm:ss';
|
|
183
|
+
case 'hh:mm':
|
|
184
|
+
case 'HH:mm':
|
|
185
|
+
return 'HH:mm';
|
|
186
|
+
case 'hh:mm A':
|
|
187
|
+
case 'HH:mm A':
|
|
188
|
+
return 'HH:mm A';
|
|
189
|
+
}
|
|
190
|
+
};
|
|
191
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date.js","sourceRoot":"","sources":["../../../../libs-ui/utils/src/date.ts"],"names":[],"mappings":"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","sourcesContent":["import { isSignal } from '@angular/core';\nimport { GetDayjsConfig, GetDayjsReturn, NonNullableDate, TYPE_OBJECT } from '@libs-ui/interfaces-types';\nimport dayjs from 'dayjs';\nimport 'dayjs/locale/en';\nimport 'dayjs/locale/vi';\nimport customParseFormat from 'dayjs/plugin/customParseFormat';\nimport localeData from 'dayjs/plugin/localeData';\nimport timezone from 'dayjs/plugin/timezone';\nimport updateLocale from 'dayjs/plugin/updateLocale';\nimport utc from 'dayjs/plugin/utc';\nimport { UtilsCache } from './cache';\ndayjs.extend(localeData);\ndayjs.extend(updateLocale);\ndayjs.extend(utc);\ndayjs.extend(timezone);\ndayjs.extend(customParseFormat);\nlet timeZoneSetup = 'Asia/Ho_Chi_Minh';\n\nexport const setDefaultTimeZone = (localeZone = timeZoneSetup) => {\n  timeZoneSetup = localeZone;\n  dayjs.tz.setDefault(localeZone);\n};\n\nexport type TYPE_FUNCTION_FORMAT_DATE = (time: dayjs.ConfigType, formatOutput: string, lang?: string) => string;\nlet functionFormatDate: TYPE_FUNCTION_FORMAT_DATE | undefined = undefined;\n\nexport const updateFunctionFormatDate = (functionCustom: TYPE_FUNCTION_FORMAT_DATE) => {\n  functionFormatDate = functionCustom;\n};\n\n/**\n * @description Lấy đối tượng dayjs theo config\n * @param config nếu không có config sẽ trả về đối tượng dayjs là thời gian hiện tại\n * @param config.date thời gian cần lấy\n * @param config.returnDayjsIfConfigDateNotExist true nếu muốn trả về đối tượng dayjs nếu config.date không có\n * @param config.utc true nếu muốn lấy thời gian UTC\n * @param config.formatOfDate định dạng thời gian đang được truyền vào\n */\nexport const getDayjs = <C extends GetDayjsConfig | undefined = undefined>(config?: C): GetDayjsReturn<C> => {\n  // helper cast để tránh lặp lại kiểu điều kiện ở các return\n  const out = (v: unknown) => v as GetDayjsReturn<C>;\n  if (!config) {\n    return out(dayjs().tz());\n  }\n  config.date = !config.date && config.returnDayjsIfConfigDateNotExist ? dayjs().tz() : config.date;\n  if (typeof config.date === 'number') {\n    config.date = dayjs.unix(config.date);\n  }\n  if (!config.date) {\n    return out(undefined);\n  }\n  let { date, utc, formatOfDate } = config as TYPE_OBJECT;\n\n  while (isSignal(date)) {\n    date = date();\n  }\n  while (isSignal(utc)) {\n    utc = utc();\n  }\n  while (isSignal(formatOfDate)) {\n    formatOfDate = formatOfDate();\n  }\n\n  if (utc) {\n    if (formatOfDate) {\n      return out(dayjs(date, formatOfDate).utc());\n    }\n    const dateInputIsUTC = (dayjs.isDayjs(date) && date.isUTC()) || (typeof date === 'string' && (date as string).includes('Z'));\n\n    if (dateInputIsUTC) {\n      return out(dayjs(date));\n    }\n\n    return out(dayjs(date).utc());\n  }\n\n  if (typeof date === 'string' && !(date as string).includes('Z') && !(date as string).includes('+')) {\n    return out(formatOfDate ? dayjs.tz(date, formatOfDate, config.localeZone || timeZoneSetup) : dayjs.tz(date, config.localeZone || timeZoneSetup));\n  }\n\n  return out((formatOfDate ? dayjs(date, formatOfDate) : dayjs(date)).tz());\n};\n\n/**\n * @description Kiểm tra xem hai ngày có khác nhau không (khác ngày trong tuần)\n * @param date1 ngày thứ nhất cần so sánh\n * @param date2 ngày thứ hai cần so sánh\n * @returns true nếu hai ngày khác nhau, false nếu giống nhau hoặc không thể so sánh\n */\nexport const isDifferenceDay = (date1: NonNullableDate, date2: NonNullableDate) => {\n  if (isDifferenceMonth(date1, date2)) {\n    return true;\n  }\n  const day1 = getDayjs({ date: date1 })?.day();\n  const day2 = getDayjs({ date: date2 })?.day();\n\n  return day1 !== day2;\n};\n\n/**\n * @description Kiểm tra xem hai ngày có khác tháng không\n * @param date1 ngày thứ nhất cần so sánh\n * @param date2 ngày thứ hai cần so sánh\n * @returns true nếu hai ngày khác tháng, false nếu cùng tháng hoặc không thể so sánh\n */\nexport const isDifferenceMonth = (date1: NonNullableDate, date2: NonNullableDate) => {\n  if (isDifferenceYear(date1, date2)) {\n    return true;\n  }\n  const month1 = getDayjs({ date: date1 })?.month();\n  const month2 = getDayjs({ date: date2 })?.month();\n\n  return month1 !== month2;\n};\n\n/**\n * @description Kiểm tra xem hai ngày có khác năm không\n * @param date1 ngày thứ nhất cần so sánh\n * @param date2 ngày thứ hai cần so sánh\n * @returns true nếu hai ngày khác năm, false nếu cùng năm hoặc không thể so sánh\n */\nexport const isDifferenceYear = (date1: NonNullableDate, date2: NonNullableDate) => {\n  const year1 = getDayjs({ date: date1 })?.year();\n  const year2 = getDayjs({ date: date2 })?.year();\n\n  return year1 !== year2;\n};\n\n/**\n * @description Lấy ra chuỗi thời gian hiển thị theo định dạng và ngôn ngữ\n * @param date thời gian cần định dạng\n * @param format định dạng thời gian muốn lấy ra\n * @param lang lấy theo ngôn ngữ\n */\nexport const formatDate = (date: dayjs.ConfigType, formatOutput = 'YYYY/MM/DD HH:mm', lang?: string, formatInput?: string): string => {\n  if (!date) {\n    return '';\n  }\n  lang = lang || (UtilsCache.getLang() as string);\n  if (functionFormatDate) {\n    return functionFormatDate(date, formatOutput, lang);\n  }\n\n  date = getDayjs({ date, formatOfDate: formatInput }).locale(lang);\n  if (lang === 'vi') {\n    dayjs.updateLocale('vi', {\n      monthsShort: 'Thg 1_Thg 2_Thg 3_Thg 4_Thg 5_Thg 6_Thg 7_Thg 8_Thg 9_Thg 10_Thg 11_Thg 12'.split('_'),\n    });\n  }\n\n  return date.format(getFormatData(getTypeByFormat(formatOutput), lang)) || '';\n};\n\nconst getTypeByFormat = (format: string): 'dm' | 'dmy' | 'dmy hm' | 'my' | 'dmy hms' | 'hh:mm' | 'HH:mm' | 'hh:mm A' | 'HH:mm A' => {\n  if (format === 'dm' || format === 'dmy' || format === 'dmy hm' || format === 'my') {\n    return format;\n  }\n\n  if (format === 'DD/MM/YYYY' || format === 'YYYY-MM-DD' || format === 'dd/MM/yyyy' || format === 'dd-MM-yyyy' || format === 'dd/mm/yyyy') {\n    return 'dmy';\n  }\n\n  if (format === 'MM-DD' || format === 'dd/MM' || format === 'dd/mm') {\n    return 'dm';\n  }\n\n  if (format === 'M/YYYY' || format === 'YYYY-MM' || format === 'MM/yyyy') {\n    return 'my';\n  }\n\n  if (format === 'YYYY/MM/DD hh:mm:ss' || format === 'dmy hms' || format === 'dd/mm/yyyy hh:mm:ss') {\n    return 'dmy hms';\n  }\n\n  if (format === 'hh:mm' || format === 'HH:mm') {\n    return 'hh:mm';\n  }\n\n  if (format === 'hh:mm A' || format === 'HH:mm A') {\n    return 'hh:mm A';\n  }\n\n  return 'dmy hm';\n};\n\nconst getFormatData = (type: 'dm' | 'dmy' | 'dmy hm' | 'my' | 'dmy hms' | 'hh:mm' | 'hh:mm A' | 'HH:mm' | 'HH:mm A', lang: string) => {\n  switch (type) {\n    case 'dm':\n      if (lang === 'vi') {\n        return 'D MMM';\n      }\n\n      return 'MMM D';\n\n    case 'dmy':\n      if (lang === 'vi') {\n        return 'D MMM, YYYY';\n      }\n\n      return 'MMM D, YYYY';\n\n    case 'dmy hm':\n      if (lang === 'vi') {\n        return 'D MMM, YYYY HH:mm';\n      }\n\n      return 'MMM D, YYYY HH:mm';\n\n    case 'my':\n      if (lang === 'vi') {\n        return 'MMM, YYYY';\n      }\n\n      return 'MMM YYYY ';\n\n    case 'dmy hms':\n      if (lang === 'vi') {\n        return 'D MMM, YYYY HH:mm:ss';\n      }\n\n      return 'MMM D, YYYY HH:mm:ss';\n\n    case 'hh:mm':\n    case 'HH:mm':\n      return 'HH:mm';\n\n    case 'hh:mm A':\n    case 'HH:mm A':\n      return 'HH:mm A';\n  }\n};\n"]}
|
package/esm2022/dom.mjs
ADDED
|
@@ -0,0 +1,256 @@
|
|
|
1
|
+
import DeviceDetector from 'device-detector-js';
|
|
2
|
+
import Quill from 'quill';
|
|
3
|
+
import { finalize, fromEvent, lastValueFrom, mergeMap, startWith, takeUntil, tap, timer } from 'rxjs';
|
|
4
|
+
import { decodeEscapeHtml } from './format-text';
|
|
5
|
+
import { set } from './helpers';
|
|
6
|
+
const deviceDetector = new DeviceDetector();
|
|
7
|
+
const parser = new DOMParser();
|
|
8
|
+
const quill = new Quill(document.createElement('div'));
|
|
9
|
+
export const getDeviceInfo = () => {
|
|
10
|
+
return deviceDetector.parse(window.navigator.userAgent);
|
|
11
|
+
};
|
|
12
|
+
export const getPlatFromBrowser = () => {
|
|
13
|
+
const info = getDeviceInfo();
|
|
14
|
+
return `${info.client?.name} ${info.client?.version}`;
|
|
15
|
+
};
|
|
16
|
+
export const getEventNameHandleClick = (() => {
|
|
17
|
+
const deviceInfo = getDeviceInfo();
|
|
18
|
+
let nameEventHandleClick = deviceInfo.device?.type === 'desktop' ? 'click' : 'touchstart';
|
|
19
|
+
if ('onpointerdown' in window) {
|
|
20
|
+
nameEventHandleClick = 'pointerdown';
|
|
21
|
+
}
|
|
22
|
+
return nameEventHandleClick;
|
|
23
|
+
})();
|
|
24
|
+
export const getDocumentByString = (htmlStr) => {
|
|
25
|
+
return parser.parseFromString(htmlStr, 'text/html');
|
|
26
|
+
};
|
|
27
|
+
export const cloneIBoundingClientRect = (rect) => {
|
|
28
|
+
return {
|
|
29
|
+
top: rect['top'],
|
|
30
|
+
left: rect['left'],
|
|
31
|
+
width: rect['width'],
|
|
32
|
+
height: rect['height'],
|
|
33
|
+
bottom: rect['bottom'],
|
|
34
|
+
};
|
|
35
|
+
};
|
|
36
|
+
export const setStylesElement = (el, styles) => {
|
|
37
|
+
Object.keys(styles).forEach((key) => set(el, `style.${key}`, styles[key]));
|
|
38
|
+
};
|
|
39
|
+
export const getViewport = (win = window) => {
|
|
40
|
+
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
|
+
return { width: w, height: h };
|
|
42
|
+
};
|
|
43
|
+
export const setCaretPosition = (element, position) => {
|
|
44
|
+
if (!element || !element.setSelectionRange) {
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
element.focus();
|
|
48
|
+
element.setSelectionRange(position, position);
|
|
49
|
+
};
|
|
50
|
+
export const checkViewInScreen = (container, element, elementScroll, maxTopLeft) => {
|
|
51
|
+
if (!container || !element) {
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
const rectContainer = cloneIBoundingClientRect(container.getBoundingClientRect());
|
|
55
|
+
if (elementScroll) {
|
|
56
|
+
const rectElementScroll = elementScroll.getBoundingClientRect();
|
|
57
|
+
rectContainer['left'] = rectElementScroll.left;
|
|
58
|
+
rectContainer['top'] = rectElementScroll.top;
|
|
59
|
+
}
|
|
60
|
+
const rectElement = element.getBoundingClientRect();
|
|
61
|
+
const { left, top } = rectElement;
|
|
62
|
+
const topContainer = rectContainer['top'];
|
|
63
|
+
const leftContainer = rectContainer['left'];
|
|
64
|
+
const widthContainer = rectContainer['width'];
|
|
65
|
+
const heightContainer = rectContainer['height'];
|
|
66
|
+
const scrollTopContainer = container.scrollTop;
|
|
67
|
+
const scrollLeftContainer = container.scrollLeft;
|
|
68
|
+
const limitTop = topContainer + scrollTopContainer + heightContainer;
|
|
69
|
+
const limitLeft = leftContainer + scrollLeftContainer + widthContainer;
|
|
70
|
+
const maxTopItem = top + (maxTopLeft?.top || 0);
|
|
71
|
+
const maxLeftItem = left + (maxTopLeft?.left || 0);
|
|
72
|
+
if (topContainer <= maxTopItem && maxTopItem <= limitTop && leftContainer <= maxLeftItem && maxLeftItem <= limitLeft) {
|
|
73
|
+
return true;
|
|
74
|
+
}
|
|
75
|
+
return false;
|
|
76
|
+
};
|
|
77
|
+
export const checkMouseOverInContainer = (mousePosition, element, rect) => {
|
|
78
|
+
if (!element && !rect) {
|
|
79
|
+
return false;
|
|
80
|
+
}
|
|
81
|
+
const rectElement = rect || element?.getBoundingClientRect() || {};
|
|
82
|
+
const { left, top, width, height } = rectElement;
|
|
83
|
+
const limitLeft = left + width;
|
|
84
|
+
const limitTop = top + height;
|
|
85
|
+
const { clientX, clientY } = mousePosition;
|
|
86
|
+
if (left <= clientX && clientX <= limitLeft && top <= clientY && clientY <= limitTop) {
|
|
87
|
+
return true;
|
|
88
|
+
}
|
|
89
|
+
return false;
|
|
90
|
+
};
|
|
91
|
+
export const getDragEventByElement = (config) => {
|
|
92
|
+
let timer;
|
|
93
|
+
const addClass = () => {
|
|
94
|
+
timer = setTimeout(() => {
|
|
95
|
+
document.body.classList.add('!select-none');
|
|
96
|
+
document.body.classList.add('!cursor-grabbing');
|
|
97
|
+
}, 250);
|
|
98
|
+
};
|
|
99
|
+
const removeClass = () => {
|
|
100
|
+
document.body.classList.remove('!select-none');
|
|
101
|
+
document.body.classList.remove('!cursor-grabbing');
|
|
102
|
+
clearTimeout(timer);
|
|
103
|
+
};
|
|
104
|
+
fromEvent(config.elementMouseDown, 'mouseleave')
|
|
105
|
+
.pipe(tap(() => document.body.classList.remove('cursor-pointer')), takeUntil(config.onDestroy))
|
|
106
|
+
.subscribe();
|
|
107
|
+
fromEvent(config.elementMouseDown, 'mouseenter')
|
|
108
|
+
.pipe(tap(() => document.body.classList.add('cursor-pointer')), takeUntil(config.onDestroy))
|
|
109
|
+
.subscribe();
|
|
110
|
+
const mouseDown = fromEvent(config.elementMouseDown, 'mousedown').pipe(tap((e) => {
|
|
111
|
+
e.stopPropagation();
|
|
112
|
+
addClass();
|
|
113
|
+
config.functionMouseDown?.(e);
|
|
114
|
+
}));
|
|
115
|
+
const mouseup = fromEvent(config.elementMouseUp || document, 'mouseup').pipe(tap((e) => {
|
|
116
|
+
e.stopPropagation();
|
|
117
|
+
removeClass();
|
|
118
|
+
config.functionMouseUp?.(e);
|
|
119
|
+
}));
|
|
120
|
+
const mousemove = fromEvent(config.elementMouseMove || document, 'mousemove').pipe(tap((e) => {
|
|
121
|
+
e.stopPropagation();
|
|
122
|
+
config.functionMouseMove?.(e);
|
|
123
|
+
}), takeUntil(mouseup));
|
|
124
|
+
return mouseDown.pipe(mergeMap((e) => (config.isStartWithMouseDownEvent ? mousemove.pipe(startWith(e)) : mousemove)), takeUntil(config.onDestroy), finalize(removeClass));
|
|
125
|
+
};
|
|
126
|
+
export const getHTMLFromQuill = async (data, options) => {
|
|
127
|
+
const { replaceNewLineTo = '<br>', replaceTagBRTo, replaceTags, replaceBrToDiv } = options || {};
|
|
128
|
+
const delta = typeof data === 'string' ? await getDeltaFromHTML(data) : data;
|
|
129
|
+
if (options?.functionReplaceDelta) {
|
|
130
|
+
options.functionReplaceDelta(delta);
|
|
131
|
+
}
|
|
132
|
+
delta.ops.forEach((op) => {
|
|
133
|
+
if (op.insert) {
|
|
134
|
+
if (typeof op.insert === 'string') {
|
|
135
|
+
if (replaceNewLineTo) {
|
|
136
|
+
op.insert = op.insert.replace(/\n/g, replaceNewLineTo);
|
|
137
|
+
}
|
|
138
|
+
if (replaceTagBRTo) {
|
|
139
|
+
op.insert = op.insert.replace(/<br>/g, replaceTagBRTo);
|
|
140
|
+
}
|
|
141
|
+
if (replaceTags?.length) {
|
|
142
|
+
for (const tag of replaceTags) {
|
|
143
|
+
op.insert = op.insert.replace(new RegExp(`<${tag.tag}>`, 'g'), `<${tag.replaceTo}>`);
|
|
144
|
+
op.insert = op.insert.replace(new RegExp(`</${tag.tag}>`, 'g'), `</${tag.replaceTo}>`);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
});
|
|
150
|
+
quill.setContents(delta);
|
|
151
|
+
let htmlText = options?.getRootHtml ? quill.root.innerHTML : quill.root.firstElementChild?.innerHTML;
|
|
152
|
+
if (replaceBrToDiv) {
|
|
153
|
+
htmlText = convertHtmlToDivBlocks(htmlText || '');
|
|
154
|
+
}
|
|
155
|
+
return decodeEscapeHtml(htmlText || '');
|
|
156
|
+
};
|
|
157
|
+
export const convertHtmlToDivBlocks = (html) => {
|
|
158
|
+
const BREAK_TOKEN = '<<<BREAK>>>';
|
|
159
|
+
// Bước 1: thay <br> thành token tạm
|
|
160
|
+
const normalizedHtml = html.replace(/<br\s*\/?>/gi, BREAK_TOKEN);
|
|
161
|
+
// Bước 2: tách theo token
|
|
162
|
+
const parts = normalizedHtml.split(BREAK_TOKEN);
|
|
163
|
+
const parser = new DOMParser();
|
|
164
|
+
const divs = [];
|
|
165
|
+
for (const raw of parts) {
|
|
166
|
+
const trimmed = raw.trim();
|
|
167
|
+
if (!trimmed)
|
|
168
|
+
continue;
|
|
169
|
+
// parse mỗi phần nhỏ như một document riêng
|
|
170
|
+
const doc = parser.parseFromString(trimmed, 'text/html');
|
|
171
|
+
const body = doc.body;
|
|
172
|
+
// Lấy lại nội dung bên trong body
|
|
173
|
+
divs.push(`<div>${body.innerHTML}</div>`);
|
|
174
|
+
}
|
|
175
|
+
return divs.join('');
|
|
176
|
+
};
|
|
177
|
+
export const getDeltaFromHTML = async (html) => {
|
|
178
|
+
quill.root.innerHTML = html;
|
|
179
|
+
setTimeout(() => {
|
|
180
|
+
console.log(quill.getContents());
|
|
181
|
+
}, 1000);
|
|
182
|
+
await lastValueFrom(timer(1000));
|
|
183
|
+
return quill.getContents();
|
|
184
|
+
};
|
|
185
|
+
export const processPasteData = async (e, config) => {
|
|
186
|
+
const element = config.element;
|
|
187
|
+
const files = e.clipboardData?.files;
|
|
188
|
+
if (files?.length) {
|
|
189
|
+
e.preventDefault();
|
|
190
|
+
config.handlerPasteFile?.(files);
|
|
191
|
+
config.callBack?.('file');
|
|
192
|
+
return;
|
|
193
|
+
}
|
|
194
|
+
// Lưu selection TRƯỚC khi prevent default
|
|
195
|
+
const selection = window.getSelection();
|
|
196
|
+
let savedRange = null;
|
|
197
|
+
if (selection && selection.rangeCount > 0) {
|
|
198
|
+
const range = selection.getRangeAt(0);
|
|
199
|
+
// Chỉ lưu nếu range nằm trong contentText element
|
|
200
|
+
const container = range.commonAncestorContainer;
|
|
201
|
+
const isInContentElement = element.contains(container.nodeType === Node.TEXT_NODE ? container.parentNode : container);
|
|
202
|
+
if (isInContentElement) {
|
|
203
|
+
savedRange = range.cloneRange();
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
// Prevent default để tự xử lý paste
|
|
207
|
+
e.preventDefault();
|
|
208
|
+
// Sử dụng Quill để clean HTML content
|
|
209
|
+
const htmlContent = e.clipboardData?.getData('text/html') || '';
|
|
210
|
+
const plainText = e.clipboardData?.getData('text/plain') || '';
|
|
211
|
+
let contentToInsert = (plainText || '').replace(/\n/g, '<br>');
|
|
212
|
+
if (htmlContent) {
|
|
213
|
+
contentToInsert = await getHTMLFromQuill(htmlContent);
|
|
214
|
+
}
|
|
215
|
+
if (!contentToInsert) {
|
|
216
|
+
config.callBack?.('no-content');
|
|
217
|
+
return;
|
|
218
|
+
}
|
|
219
|
+
if (savedRange) {
|
|
220
|
+
insertContentWithRange(contentToInsert, savedRange, element);
|
|
221
|
+
config.callBack?.('range');
|
|
222
|
+
return;
|
|
223
|
+
}
|
|
224
|
+
element.innerHTML += contentToInsert;
|
|
225
|
+
config.callBack?.('content');
|
|
226
|
+
};
|
|
227
|
+
export const insertContentWithRange = (content, savedRange, element) => {
|
|
228
|
+
const selection = window.getSelection();
|
|
229
|
+
if (!selection) {
|
|
230
|
+
// Fallback: append vào cuối
|
|
231
|
+
element.innerHTML += content;
|
|
232
|
+
return;
|
|
233
|
+
}
|
|
234
|
+
// Restore selection
|
|
235
|
+
selection.removeAllRanges();
|
|
236
|
+
selection.addRange(savedRange);
|
|
237
|
+
// Xóa nội dung đã select (nếu có)
|
|
238
|
+
savedRange.deleteContents();
|
|
239
|
+
// Tạo document fragment từ HTML content
|
|
240
|
+
const tempDiv = document.createElement('div');
|
|
241
|
+
tempDiv.innerHTML = content;
|
|
242
|
+
const fragment = document.createDocumentFragment();
|
|
243
|
+
while (tempDiv.firstChild) {
|
|
244
|
+
fragment.appendChild(tempDiv.firstChild);
|
|
245
|
+
}
|
|
246
|
+
// Insert fragment tại vị trí range
|
|
247
|
+
savedRange.insertNode(fragment);
|
|
248
|
+
// Di chuyển cursor đến cuối nội dung vừa insert
|
|
249
|
+
if (fragment.lastChild) {
|
|
250
|
+
savedRange.setStartAfter(fragment.lastChild);
|
|
251
|
+
}
|
|
252
|
+
savedRange.collapse(true);
|
|
253
|
+
selection.removeAllRanges();
|
|
254
|
+
selection.addRange(savedRange);
|
|
255
|
+
};
|
|
256
|
+
//# 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":["/* eslint-disable @typescript-eslint/no-explicit-any */\nimport { IBoundingClientRect } from '@libs-ui/interfaces-types';\nimport DeviceDetector from 'device-detector-js';\nimport Quill from 'quill';\nimport Delta from 'quill-delta';\nimport { finalize, fromEvent, lastValueFrom, mergeMap, startWith, Subject, takeUntil, tap, timer } from 'rxjs';\nimport { decodeEscapeHtml } from './format-text';\nimport { set } from './helpers';\n\nconst deviceDetector = new DeviceDetector();\nconst parser = new DOMParser();\nconst quill = new Quill(document.createElement('div'));\n\nexport const getDeviceInfo = () => {\n  return deviceDetector.parse(window.navigator.userAgent);\n};\nexport const getPlatFromBrowser = () => {\n  const info = getDeviceInfo();\n  return `${info.client?.name} ${info.client?.version}`;\n};\n\nexport const getEventNameHandleClick = (() => {\n  const deviceInfo = getDeviceInfo();\n  let nameEventHandleClick = deviceInfo.device?.type === 'desktop' ? 'click' : 'touchstart';\n  if ('onpointerdown' in window) {\n    nameEventHandleClick = 'pointerdown';\n  }\n\n  return nameEventHandleClick;\n})();\n\nexport const getDocumentByString = (htmlStr: string): Document => {\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  onDestroy: Subject<any>;\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      takeUntil(config.onDestroy)\n    )\n    .subscribe();\n  fromEvent(config.elementMouseDown, 'mouseenter')\n    .pipe(\n      tap(() => document.body.classList.add('cursor-pointer')),\n      takeUntil(config.onDestroy)\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    takeUntil(config.onDestroy),\n    finalize(removeClass)\n  );\n};\n\nexport const getHTMLFromQuill = async (\n  data: Delta | string,\n  options?: {\n    getRootHtml?: boolean;\n    replaceNewLineTo?: string;\n    replaceTagBRTo?: string;\n    replaceBrToDiv?: boolean;\n    functionReplaceDelta?: (op: Delta) => void;\n    replaceTags?: {\n      tag: string;\n      replaceTo: string;\n    }[];\n  }\n): Promise<string> => {\n  const { replaceNewLineTo = '<br>', replaceTagBRTo, replaceTags, replaceBrToDiv } = options || {};\n  const delta = typeof data === 'string' ? await getDeltaFromHTML(data) : data;\n  if (options?.functionReplaceDelta) {\n    options.functionReplaceDelta(delta);\n  }\n\n  delta.ops.forEach((op) => {\n    if (op.insert) {\n      if (typeof op.insert === 'string') {\n        if (replaceNewLineTo) {\n          op.insert = op.insert.replace(/\\n/g, replaceNewLineTo);\n        }\n        if (replaceTagBRTo) {\n          op.insert = op.insert.replace(/<br>/g, replaceTagBRTo);\n        }\n\n        if (replaceTags?.length) {\n          for (const tag of replaceTags) {\n            op.insert = op.insert.replace(new RegExp(`<${tag.tag}>`, 'g'), `<${tag.replaceTo}>`) as string;\n            op.insert = op.insert.replace(new RegExp(`</${tag.tag}>`, 'g'), `</${tag.replaceTo}>`) as string;\n          }\n        }\n      }\n    }\n  });\n  quill.setContents(delta);\n  let htmlText = options?.getRootHtml ? quill.root.innerHTML : quill.root.firstElementChild?.innerHTML;\n  if (replaceBrToDiv) {\n    htmlText = convertHtmlToDivBlocks(htmlText || '');\n  }\n  return decodeEscapeHtml(htmlText || '');\n};\n\nexport const convertHtmlToDivBlocks = (html: string): string => {\n  const BREAK_TOKEN = '<<<BREAK>>>';\n  // Bước 1: thay <br> thành token tạm\n  const normalizedHtml = html.replace(/<br\\s*\\/?>/gi, BREAK_TOKEN);\n\n  // Bước 2: tách theo token\n  const parts = normalizedHtml.split(BREAK_TOKEN);\n\n  const parser = new DOMParser();\n  const divs: string[] = [];\n\n  for (const raw of parts) {\n    const trimmed = raw.trim();\n    if (!trimmed) continue;\n\n    // parse mỗi phần nhỏ như một document riêng\n    const doc = parser.parseFromString(trimmed, 'text/html');\n    const body = doc.body;\n\n    // Lấy lại nội dung bên trong body\n    divs.push(`<div>${body.innerHTML}</div>`);\n  }\n\n  return divs.join('');\n};\n\nexport const getDeltaFromHTML = async (html: string): Promise<Delta> => {\n  quill.root.innerHTML = html;\n  setTimeout(() => {\n    console.log(quill.getContents());\n  }, 1000);\n\n  await lastValueFrom(timer(1000));\n  return quill.getContents();\n};\n\nexport const processPasteData = async (\n  e: ClipboardEvent,\n  config: {\n    element: HTMLElement;\n    handlerPasteFile?: (files: FileList) => void;\n    callBack?: (positonReturn: 'file' | 'range' | 'content' | 'no-content') => void;\n  }\n) => {\n  const element = config.element;\n  const files = e.clipboardData?.files;\n  if (files?.length) {\n    e.preventDefault();\n    config.handlerPasteFile?.(files);\n    config.callBack?.('file');\n    return;\n  }\n  // Lưu selection TRƯỚC khi prevent default\n  const selection = window.getSelection();\n  let savedRange: Range | null = null;\n  if (selection && selection.rangeCount > 0) {\n    const range = selection.getRangeAt(0);\n    // Chỉ lưu nếu range nằm trong contentText element\n    const container = range.commonAncestorContainer;\n    const isInContentElement = element.contains(container.nodeType === Node.TEXT_NODE ? container.parentNode : container);\n    if (isInContentElement) {\n      savedRange = range.cloneRange();\n    }\n  }\n  // Prevent default để tự xử lý paste\n  e.preventDefault();\n  // Sử dụng Quill để clean HTML content\n  const htmlContent = e.clipboardData?.getData('text/html') || '';\n  const plainText = e.clipboardData?.getData('text/plain') || '';\n  let contentToInsert = (plainText || '').replace(/\\n/g, '<br>');\n  if (htmlContent) {\n    contentToInsert = await getHTMLFromQuill(htmlContent);\n  }\n\n  if (!contentToInsert) {\n    config.callBack?.('no-content');\n    return;\n  }\n\n  if (savedRange) {\n    insertContentWithRange(contentToInsert, savedRange, element);\n    config.callBack?.('range');\n    return;\n  }\n  element.innerHTML += contentToInsert;\n  config.callBack?.('content');\n};\n\nexport const insertContentWithRange = (content: string, savedRange: Range, element: HTMLElement): void => {\n  const selection = window.getSelection();\n  if (!selection) {\n    // Fallback: append vào cuối\n    element.innerHTML += content;\n    return;\n  }\n  // Restore selection\n  selection.removeAllRanges();\n  selection.addRange(savedRange);\n  // Xóa nội dung đã select (nếu có)\n  savedRange.deleteContents();\n  // Tạo document fragment từ HTML content\n  const tempDiv = document.createElement('div');\n  tempDiv.innerHTML = content;\n  const fragment = document.createDocumentFragment();\n  while (tempDiv.firstChild) {\n    fragment.appendChild(tempDiv.firstChild);\n  }\n  // Insert fragment tại vị trí range\n  savedRange.insertNode(fragment);\n  // Di chuyển cursor đến cuối nội dung vừa insert\n  if (fragment.lastChild) {\n    savedRange.setStartAfter(fragment.lastChild);\n  }\n  savedRange.collapse(true);\n  selection.removeAllRanges();\n  selection.addRange(savedRange);\n};\n"]}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { convertBase64ToBlob } from './base64';
|
|
2
|
+
export const downloadFileByUrlUseXmlRequest = (fileUrl, filename) => {
|
|
3
|
+
const xmlRequest = new XMLHttpRequest();
|
|
4
|
+
xmlRequest.open('GET', fileUrl, true);
|
|
5
|
+
xmlRequest.responseType = 'blob';
|
|
6
|
+
xmlRequest.onerror = () => {
|
|
7
|
+
const url = window.URL.createObjectURL(xmlRequest.response);
|
|
8
|
+
downloadFileByUrl(url, filename);
|
|
9
|
+
};
|
|
10
|
+
xmlRequest.onload = () => {
|
|
11
|
+
const url = window.URL.createObjectURL(xmlRequest.response);
|
|
12
|
+
downloadFileByUrl(url, filename);
|
|
13
|
+
};
|
|
14
|
+
xmlRequest.send();
|
|
15
|
+
};
|
|
16
|
+
export const downloadFileByUrl = async (fileUrl, filename, onlyOpen) => {
|
|
17
|
+
const downloadFileElement = document.createElement('a');
|
|
18
|
+
if (!onlyOpen) {
|
|
19
|
+
const response = await fetch(fileUrl);
|
|
20
|
+
if (response.ok) {
|
|
21
|
+
const blob = await response.blob();
|
|
22
|
+
fileUrl = URL.createObjectURL(blob);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
downloadFileElement.href = fileUrl;
|
|
26
|
+
downloadFileElement.download = filename;
|
|
27
|
+
downloadFileElement.target = '_blank';
|
|
28
|
+
downloadFileElement.click();
|
|
29
|
+
downloadFileElement.remove();
|
|
30
|
+
};
|
|
31
|
+
export const downloadImageFromELement = (imageElement, typeFileDownload, nameFile) => {
|
|
32
|
+
const parentElement = imageElement?.src;
|
|
33
|
+
const blobData = convertBase64ToBlob(parentElement);
|
|
34
|
+
const blob = new Blob([blobData], { type: typeFileDownload || 'image/png' });
|
|
35
|
+
const url = window.URL.createObjectURL(blob);
|
|
36
|
+
const link = document.createElement('a');
|
|
37
|
+
link.href = url;
|
|
38
|
+
link.download = nameFile || 'barcode';
|
|
39
|
+
link.click();
|
|
40
|
+
};
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZG93bmxvYWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzLXVpL3V0aWxzL3NyYy9kb3dubG9hZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxVQUFVLENBQUM7QUFFL0MsTUFBTSxDQUFDLE1BQU0sOEJBQThCLEdBQUcsQ0FBQyxPQUFlLEVBQUUsUUFBZ0IsRUFBRSxFQUFFO0lBQ2xGLE1BQU0sVUFBVSxHQUFHLElBQUksY0FBYyxFQUFFLENBQUM7SUFFeEMsVUFBVSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQ3RDLFVBQVUsQ0FBQyxZQUFZLEdBQUcsTUFBTSxDQUFDO0lBQ2pDLFVBQVUsQ0FBQyxPQUFPLEdBQUcsR0FBRyxFQUFFO1FBQ3hCLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLFVBQVUsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUU1RCxpQkFBaUIsQ0FBQyxHQUFHLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDbkMsQ0FBQyxDQUFDO0lBQ0YsVUFBVSxDQUFDLE1BQU0sR0FBRyxHQUFHLEVBQUU7UUFDdkIsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDLEdBQUcsQ0FBQyxlQUFlLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBRTVELGlCQUFpQixDQUFDLEdBQUcsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUNuQyxDQUFDLENBQUM7SUFDRixVQUFVLENBQUMsSUFBSSxFQUFFLENBQUM7QUFDcEIsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQUcsS0FBSyxFQUFFLE9BQWUsRUFBRSxRQUFnQixFQUFFLFFBQWtCLEVBQUUsRUFBRTtJQUMvRixNQUFNLG1CQUFtQixHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFeEQsSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDO1FBQ2QsTUFBTSxRQUFRLEdBQUcsTUFBTSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFFdEMsSUFBSSxRQUFRLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDaEIsTUFBTSxJQUFJLEdBQUcsTUFBTSxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUM7WUFFbkMsT0FBTyxHQUFHLEdBQUcsQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDdEMsQ0FBQztJQUNILENBQUM7SUFDRCxtQkFBbUIsQ0FBQyxJQUFJLEdBQUcsT0FBTyxDQUFDO0lBQ25DLG1CQUFtQixDQUFDLFFBQVEsR0FBRyxRQUFRLENBQUM7SUFDeEMsbUJBQW1CLENBQUMsTUFBTSxHQUFHLFFBQVEsQ0FBQztJQUN0QyxtQkFBbUIsQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUM1QixtQkFBbUIsQ0FBQyxNQUFNLEVBQUUsQ0FBQztBQUMvQixDQUFDLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSx3QkFBd0IsR0FBRyxDQUFDLFlBQThCLEVBQUUsZ0JBQXlCLEVBQUUsUUFBaUIsRUFBRSxFQUFFO0lBQ3ZILE1BQU0sYUFBYSxHQUFHLFlBQVksRUFBRSxHQUFHLENBQUM7SUFDeEMsTUFBTSxRQUFRLEdBQUcsbUJBQW1CLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDcEQsTUFBTSxJQUFJLEdBQUcsSUFBSSxJQUFJLENBQUMsQ0FBQyxRQUFRLENBQUMsRUFBRSxFQUFFLElBQUksRUFBRSxnQkFBZ0IsSUFBSSxXQUFXLEVBQUUsQ0FBQyxDQUFDO0lBQzdFLE1BQU0sR0FBRyxHQUFHLE1BQU0sQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdDLE1BQU0sSUFBSSxHQUFHLFFBQVEsQ0FBQyxhQUFhLENBQUMsR0FBRyxDQUFDLENBQUM7SUFFekMsSUFBSSxDQUFDLElBQUksR0FBRyxHQUFHLENBQUM7SUFDaEIsSUFBSSxDQUFDLFFBQVEsR0FBRyxRQUFRLElBQUksU0FBUyxDQUFDO0lBQ3RDLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztBQUNmLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNvbnZlcnRCYXNlNjRUb0Jsb2IgfSBmcm9tICcuL2Jhc2U2NCc7XG5cbmV4cG9ydCBjb25zdCBkb3dubG9hZEZpbGVCeVVybFVzZVhtbFJlcXVlc3QgPSAoZmlsZVVybDogc3RyaW5nLCBmaWxlbmFtZTogc3RyaW5nKSA9PiB7XG4gIGNvbnN0IHhtbFJlcXVlc3QgPSBuZXcgWE1MSHR0cFJlcXVlc3QoKTtcblxuICB4bWxSZXF1ZXN0Lm9wZW4oJ0dFVCcsIGZpbGVVcmwsIHRydWUpO1xuICB4bWxSZXF1ZXN0LnJlc3BvbnNlVHlwZSA9ICdibG9iJztcbiAgeG1sUmVxdWVzdC5vbmVycm9yID0gKCkgPT4ge1xuICAgIGNvbnN0IHVybCA9IHdpbmRvdy5VUkwuY3JlYXRlT2JqZWN0VVJMKHhtbFJlcXVlc3QucmVzcG9uc2UpO1xuXG4gICAgZG93bmxvYWRGaWxlQnlVcmwodXJsLCBmaWxlbmFtZSk7XG4gIH07XG4gIHhtbFJlcXVlc3Qub25sb2FkID0gKCkgPT4ge1xuICAgIGNvbnN0IHVybCA9IHdpbmRvdy5VUkwuY3JlYXRlT2JqZWN0VVJMKHhtbFJlcXVlc3QucmVzcG9uc2UpO1xuXG4gICAgZG93bmxvYWRGaWxlQnlVcmwodXJsLCBmaWxlbmFtZSk7XG4gIH07XG4gIHhtbFJlcXVlc3Quc2VuZCgpO1xufTtcblxuZXhwb3J0IGNvbnN0IGRvd25sb2FkRmlsZUJ5VXJsID0gYXN5bmMgKGZpbGVVcmw6IHN0cmluZywgZmlsZW5hbWU6IHN0cmluZywgb25seU9wZW4/OiBib29sZWFuKSA9PiB7XG4gIGNvbnN0IGRvd25sb2FkRmlsZUVsZW1lbnQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdhJyk7XG5cbiAgaWYgKCFvbmx5T3Blbikge1xuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2goZmlsZVVybCk7XG5cbiAgICBpZiAocmVzcG9uc2Uub2spIHtcbiAgICAgIGNvbnN0IGJsb2IgPSBhd2FpdCByZXNwb25zZS5ibG9iKCk7XG5cbiAgICAgIGZpbGVVcmwgPSBVUkwuY3JlYXRlT2JqZWN0VVJMKGJsb2IpO1xuICAgIH1cbiAgfVxuICBkb3dubG9hZEZpbGVFbGVtZW50LmhyZWYgPSBmaWxlVXJsO1xuICBkb3dubG9hZEZpbGVFbGVtZW50LmRvd25sb2FkID0gZmlsZW5hbWU7XG4gIGRvd25sb2FkRmlsZUVsZW1lbnQudGFyZ2V0ID0gJ19ibGFuayc7XG4gIGRvd25sb2FkRmlsZUVsZW1lbnQuY2xpY2soKTtcbiAgZG93bmxvYWRGaWxlRWxlbWVudC5yZW1vdmUoKTtcbn07XG5cbmV4cG9ydCBjb25zdCBkb3dubG9hZEltYWdlRnJvbUVMZW1lbnQgPSAoaW1hZ2VFbGVtZW50OiBIVE1MSW1hZ2VFbGVtZW50LCB0eXBlRmlsZURvd25sb2FkPzogc3RyaW5nLCBuYW1lRmlsZT86IHN0cmluZykgPT4ge1xuICBjb25zdCBwYXJlbnRFbGVtZW50ID0gaW1hZ2VFbGVtZW50Py5zcmM7XG4gIGNvbnN0IGJsb2JEYXRhID0gY29udmVydEJhc2U2NFRvQmxvYihwYXJlbnRFbGVtZW50KTtcbiAgY29uc3QgYmxvYiA9IG5ldyBCbG9iKFtibG9iRGF0YV0sIHsgdHlwZTogdHlwZUZpbGVEb3dubG9hZCB8fCAnaW1hZ2UvcG5nJyB9KTtcbiAgY29uc3QgdXJsID0gd2luZG93LlVSTC5jcmVhdGVPYmplY3RVUkwoYmxvYik7XG4gIGNvbnN0IGxpbmsgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdhJyk7XG5cbiAgbGluay5ocmVmID0gdXJsO1xuICBsaW5rLmRvd25sb2FkID0gbmFtZUZpbGUgfHwgJ2JhcmNvZGUnO1xuICBsaW5rLmNsaWNrKCk7XG59O1xuIl19
|