a2bei4-utils 1.0.3 → 1.0.5

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/dist/date.js CHANGED
@@ -67,133 +67,222 @@ function randomDateInRange(date1, date2) {
67
67
  return new Date(v1 + Math.floor(Math.random() * (v2 - v1 + 1)));
68
68
  }
69
69
 
70
+ //#region 持续时间相关
71
+
70
72
  /**
71
- * 计算两个时间之间的剩余/已过时长(天-时-分-秒),返回带补零的展示对象。
72
- *
73
- * @param {string|number|Date} originalTime - 原始时间(可转 Date 的任意值)
74
- * @param {Date} [currentTime=new Date()] - 基准时间,默认当前
75
- * @returns {{days:number,hours:string,minutes:string,seconds:string}}
73
+ * 时间持续对象(完整版本,包含年月日时分秒毫秒)
74
+ * @typedef {Object} DurationObject
75
+ * @property {number} years - 年数
76
+ * @property {number} months - 月数(0-11)
77
+ * @property {number} days - 天数(0-29,取决于 monthDays)
78
+ * @property {number} hours - 小时数(0-23)
79
+ * @property {number} minutes - 分钟数(0-59)
80
+ * @property {number} seconds - 秒数(0-59)
81
+ * @property {number} milliseconds - 毫秒数(0-999)
76
82
  */
77
- function calcTimeDifference(originalTime, currentTime = new Date()) {
78
- // 计算时间差(毫秒)
79
- const diffMs = currentTime - new Date(originalTime);
80
-
81
- // 转换为天、小时、分钟、秒
82
- const diffSeconds = Math.floor(diffMs / 1000);
83
- const days = Math.floor(diffSeconds / (3600 * 24));
84
- const hours = Math.floor((diffSeconds % (3600 * 24)) / 3600);
85
- const minutes = Math.floor((diffSeconds % 3600) / 60);
86
- const seconds = diffSeconds % 60;
87
83
 
88
- const padZero = (num) => String(num).padStart(2, "0");
84
+ /**
85
+ * 时间持续对象(最大单位为天)
86
+ * @typedef {Object} DurationMaxDayObject
87
+ * @property {number} days - 天数
88
+ * @property {number} hours - 小时数(0-23)
89
+ * @property {number} minutes - 分钟数(0-59)
90
+ * @property {number} seconds - 秒数(0-59)
91
+ * @property {number} milliseconds - 毫秒数(0-999)
92
+ */
89
93
 
90
- return {
91
- days,
92
- hours: padZero(hours),
93
- minutes: padZero(minutes),
94
- seconds: padZero(seconds)
95
- };
96
- }
94
+ /**
95
+ * 时间持续对象(最大单位为小时)
96
+ * @typedef {Object} DurationMaxHourObject
97
+ * @property {number} hours - 小时数
98
+ * @property {number} minutes - 分钟数(0-59)
99
+ * @property {number} seconds - 秒数(0-59)
100
+ * @property {number} milliseconds - 毫秒数(0-999)
101
+ */
97
102
 
98
103
  /**
99
- * 将总秒数格式化成人类可读的时间段文本。
100
- * 固定进制:1 年=365 天,1 月=30 天。
104
+ * 将毫秒转换为时间持续对象。
105
+ *
106
+ * @param {number} milliseconds - 毫秒数(非负整数)
107
+ * @param {Object} [options] - 选项对象。
108
+ * @param {number} [options.yearDays=365] - 一年的天数。
109
+ * @param {number} [options.monthDays=30] - 一月的天数。
110
+ * @returns {DurationObject} 时间持续对象
111
+ * @throws {TypeError} 当 milliseconds 不是有效数字
112
+ * @throws {RangeError} 当 milliseconds 为负数
113
+ * @throws {RangeError} 当 yearDays 或 monthDays 不是正整数
101
114
  *
102
- * @param {number} totalSeconds - 非负总秒数
103
- * @param {object} [options] - 格式化选项
104
- * @param {Partial<{year:string,month:string,day:string,hour:string,minute:string,second:string}>} [options.labels] - 各单位的自定义文本
105
- * @param {('year'|'month'|'day'|'hour'|'minute'|'second')} [options.maxUnit] - 最大输出单位
106
- * @param {('year'|'month'|'day'|'hour'|'minute'|'second')} [options.minUnit] - 最小输出单位
107
- * @param {boolean} [options.showZero] - 是否强制显示 0 秒
108
- * @returns {string} 拼接后的时长文本,如“1天 02小时 30分钟”
109
- * @throws {TypeError} 当 totalSeconds 为非数字或负数时抛出
115
+ * @example
116
+ * // 基本用法
117
+ * millisecond2Duration(42070000500);
118
+ * // 返回: { years: 1, months: 4, days: 1, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }
110
119
  */
111
- function formatDuration(totalSeconds, options = {}) {
112
- if (typeof totalSeconds !== "number" || totalSeconds < 0 || !isFinite(totalSeconds)) {
113
- throw new TypeError("totalSeconds 必须是非负数字");
120
+ function millisecond2Duration(milliseconds, options = { yearDays: 365, monthDays: 30 }) {
121
+ // 参数验证
122
+ if (typeof milliseconds !== "number" || isNaN(milliseconds)) {
123
+ throw new TypeError("milliseconds must be a valid number");
124
+ }
125
+ if (milliseconds < 0) {
126
+ throw new RangeError("milliseconds must be a non-negative number");
114
127
  }
115
128
 
116
- // 1. 默认中文单位
117
- const DEFAULT_LABELS = {
118
- year: "年",
119
- month: "月",
120
- day: "天",
121
- hour: "小时",
122
- minute: "分钟",
123
- second: "秒"
124
- };
129
+ // 默认选项
130
+ const { yearDays = 365, monthDays = 30 } = options;
125
131
 
126
- // 2. 固定进制表(秒)
127
- const UNIT_TABLE = [
128
- { key: "year", seconds: 365 * 24 * 3600 },
129
- { key: "month", seconds: 30 * 24 * 3600 },
130
- { key: "day", seconds: 24 * 3600 },
131
- { key: "hour", seconds: 3600 },
132
- { key: "minute", seconds: 60 },
133
- { key: "second", seconds: 1 }
134
- ];
135
-
136
- // 3. 合并用户自定义文本
137
- const labels = Object.assign({}, DEFAULT_LABELS, options.labels);
138
-
139
- // 4. 根据 maxUnit / minUnit 截取
140
- let start = 0,
141
- end = UNIT_TABLE.length;
142
- if (options.maxUnit) {
143
- const idx = UNIT_TABLE.findIndex((u) => u.key === options.maxUnit);
144
- if (idx !== -1) start = idx;
132
+ // 选项验证
133
+ if (!Number.isInteger(yearDays) || yearDays <= 0) {
134
+ throw new RangeError("yearDays must be a positive integer");
145
135
  }
146
- if (options.minUnit) {
147
- const idx = UNIT_TABLE.findIndex((u) => u.key === options.minUnit);
148
- if (idx !== -1) end = idx + 1;
136
+ if (!Number.isInteger(monthDays) || monthDays <= 0) {
137
+ throw new RangeError("monthDays must be a positive integer");
149
138
  }
150
- const units = UNIT_TABLE.slice(start, end);
151
- if (!units.length) units.push(UNIT_TABLE[UNIT_TABLE.length - 1]); // 保底秒
152
139
 
153
- // 5. 逐级计算
154
- let rest = Math.floor(totalSeconds);
155
- const parts = [];
140
+ const totalMilliseconds = Math.floor(milliseconds);
141
+ const ms = totalMilliseconds % 1000;
142
+ const diffSeconds = Math.floor(totalMilliseconds / 1000);
156
143
 
157
- for (const { key, seconds } of units) {
158
- const val = Math.floor(rest / seconds);
159
- rest %= seconds;
144
+ const seconds = diffSeconds % 60;
145
+ const minutes = Math.floor(diffSeconds / 60) % 60;
146
+ const hours = Math.floor(diffSeconds / 3600) % 24;
160
147
 
161
- const shouldShow = val > 0 || (options.showZero && key === "second");
162
- if (shouldShow || (parts.length === 0 && rest === 0)) {
163
- parts.push(`${val}${labels[key]}`);
164
- }
148
+ // 计算年、月、日
149
+ const totalDays = Math.floor(diffSeconds / 3600 / 24);
150
+ const years = Math.floor(totalDays / yearDays);
151
+ const remainingDays = totalDays % yearDays;
152
+ const months = Math.floor(remainingDays / monthDays);
153
+ const days = remainingDays % monthDays;
154
+
155
+ return { years, months, days, hours, minutes, seconds, milliseconds: ms };
156
+ }
157
+
158
+ /**
159
+ * 将毫秒转换为时间持续对象(最大单位为天)。
160
+ * @param {number} milliseconds - 毫秒数(非负整数)
161
+ * @returns {DurationMaxDayObject} 包含天、小时、分钟、秒、毫秒的时间持续对象
162
+ * @throws {TypeError} 当 milliseconds 不是有效数字时抛出
163
+ * @throws {RangeError} 当 milliseconds 为负数时抛出
164
+ * @example
165
+ * // 返回 { days: 486, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }
166
+ * millisecond2DurationMaxDay(42070000500);
167
+ */
168
+ function millisecond2DurationMaxDay(milliseconds) {
169
+ if (typeof milliseconds !== "number" || isNaN(milliseconds)) {
170
+ throw new TypeError("milliseconds must be a valid number");
171
+ }
172
+ if (milliseconds < 0) {
173
+ throw new RangeError("milliseconds must be a non-negative number");
165
174
  }
166
175
 
167
- // 6. 兜底
168
- if (parts.length === 0) {
169
- parts.push(`0${labels[units[units.length - 1].key]}`);
176
+ const totalMilliseconds = Math.floor(milliseconds);
177
+ const ms = totalMilliseconds % 1000;
178
+ const diffSeconds = Math.floor(totalMilliseconds / 1000);
179
+
180
+ const seconds = diffSeconds % 60;
181
+ const minutes = Math.floor(diffSeconds / 60) % 60;
182
+ const hours = Math.floor(diffSeconds / 3600) % 24;
183
+ const days = Math.floor(diffSeconds / 3600 / 24);
184
+
185
+ return { days, hours, minutes, seconds, milliseconds: ms };
186
+ }
187
+
188
+ /**
189
+ * 将毫秒转换为时间持续对象(最大单位为小时)。
190
+ * @param {number} milliseconds - 毫秒数(非负整数)
191
+ * @returns {DurationMaxHourObject} 包含小时、分钟、秒、毫秒的时间持续对象
192
+ * @throws {TypeError} 当 milliseconds 不是有效数字时抛出
193
+ * @throws {RangeError} 当 milliseconds 为负数时抛出
194
+ * @example
195
+ * // 返回 { hours: 11686, minutes: 6, seconds: 40, milliseconds: 500 }
196
+ * millisecond2DurationMaxHour(42070000500);
197
+ */
198
+ function millisecond2DurationMaxHour(milliseconds) {
199
+ if (typeof milliseconds !== "number" || isNaN(milliseconds)) {
200
+ throw new TypeError("milliseconds must be a valid number");
170
201
  }
202
+ if (milliseconds < 0) {
203
+ throw new RangeError("milliseconds must be a non-negative number");
204
+ }
205
+
206
+ const totalMilliseconds = Math.floor(milliseconds);
207
+ const ms = totalMilliseconds % 1000;
208
+ const diffSeconds = Math.floor(totalMilliseconds / 1000);
209
+
210
+ const seconds = diffSeconds % 60;
211
+ const minutes = Math.floor(diffSeconds / 60) % 60;
212
+ const hours = Math.floor(diffSeconds / 3600);
171
213
 
172
- return parts.join("");
214
+ return { hours, minutes, seconds, milliseconds: ms };
173
215
  }
174
216
 
175
217
  /**
176
- * 快捷调用 {@link formatDuration},最大单位到“天”。
218
+ * 将秒转换为时间持续对象。
219
+ *
220
+ * @param {number} seconds - 秒数(非负整数)
221
+ * @param {Object} [options] - 选项对象。
222
+ * @param {number} [options.yearDays=365] - 一年的天数。
223
+ * @param {number} [options.monthDays=30] - 一月的天数。
224
+ * @returns {DurationObject} 时间持续对象
225
+ * @throws {TypeError} 当 seconds 不是有效数字
226
+ * @throws {RangeError} 当 seconds 为负数
227
+ * @throws {RangeError} 当 yearDays 或 monthDays 不是正整数
177
228
  *
178
- * @param {number} totalSeconds
179
- * @param {Omit<Parameters<typeof formatDuration>[1],'maxUnit'>} [options]
180
- * @returns {string}
229
+ * @example
230
+ * // 基本用法
231
+ * second2Duration(42070000.5);
232
+ * // 返回: { years: 1, months: 4, days: 1, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }
181
233
  */
182
- function formatDurationMaxDay(totalSeconds, options = {}) {
183
- return formatDuration(totalSeconds, { ...options, maxUnit: "day" });
234
+ function second2Duration(seconds, options = { yearDays: 365, monthDays: 30 }) {
235
+ if (typeof seconds !== "number" || isNaN(seconds)) {
236
+ throw new TypeError("seconds must be a valid number");
237
+ }
238
+ if (seconds < 0) {
239
+ throw new RangeError("seconds must be a non-negative number");
240
+ }
241
+ return millisecond2Duration(seconds * 1000, options);
184
242
  }
185
243
 
186
244
  /**
187
- * 快捷调用 {@link formatDuration},最大单位到“小时”。
188
- *
189
- * @param {number} totalSeconds
190
- * @param {Omit<Parameters<typeof formatDuration>[1],'maxUnit'>} [options]
191
- * @returns {string}
245
+ * 将秒转换为时间持续对象(最大单位为天)。
246
+ * @param {number} seconds - 秒数(非负整数)
247
+ * @returns {DurationMaxDayObject} 包含天、小时、分钟、秒、毫秒的时间持续对象
248
+ * @throws {TypeError} 当 seconds 不是有效数字时抛出
249
+ * @throws {RangeError} 当 seconds 为负数时抛出
250
+ * @example
251
+ * // 返回 { days: 486, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }
252
+ * second2DurationMaxDay(42070000.5);
192
253
  */
193
- function formatDurationMaxHour(totalSeconds, options = {}) {
194
- return formatDuration(totalSeconds, { ...options, maxUnit: "hour" });
254
+ function second2DurationMaxDay(seconds) {
255
+ if (typeof seconds !== "number" || isNaN(seconds)) {
256
+ throw new TypeError("seconds must be a valid number");
257
+ }
258
+ if (seconds < 0) {
259
+ throw new RangeError("seconds must be a non-negative number");
260
+ }
261
+ return millisecond2DurationMaxDay(seconds * 1000);
195
262
  }
196
263
 
264
+ /**
265
+ * 将秒转换为时间持续对象(最大单位为小时)。
266
+ * @param {number} seconds - 秒数(非负整数)
267
+ * @returns {DurationMaxHourObject} 包含小时、分钟、秒、毫秒的时间持续对象
268
+ * @throws {TypeError} 当 seconds 不是有效数字时抛出
269
+ * @throws {RangeError} 当 seconds 为负数时抛出
270
+ * @example
271
+ * // 返回 { hours: 11686, minutes: 6, seconds: 40, milliseconds: 500 }
272
+ * second2DurationMaxHour(42070000.5);
273
+ */
274
+ function second2DurationMaxHour(seconds) {
275
+ if (typeof seconds !== "number" || isNaN(seconds)) {
276
+ throw new TypeError("seconds must be a valid number");
277
+ }
278
+ if (seconds < 0) {
279
+ throw new RangeError("seconds must be a non-negative number");
280
+ }
281
+ return millisecond2DurationMaxHour(seconds * 1000);
282
+ }
283
+
284
+ //#endregion
285
+
197
286
  /**
198
287
  * 根据小时数返回对应的时间段名称。
199
288
  *
@@ -324,5 +413,5 @@ function formatTimeForLocale(
324
413
  return `${year}${locales.year}${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;
325
414
  }
326
415
 
327
- export { calcTimeDifference, formatDuration, formatDurationMaxDay, formatDurationMaxHour, formatTimeForLocale, getTimePeriodName, randomDateInRange, toDate };
416
+ export { formatTimeForLocale, getTimePeriodName, millisecond2Duration, millisecond2DurationMaxDay, millisecond2DurationMaxHour, randomDateInRange, second2Duration, second2DurationMaxDay, second2DurationMaxHour, toDate };
328
417
  //# sourceMappingURL=date.js.map
package/dist/date.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"date.js","sources":["../src/source/date.js"],"sourcesContent":["function getLocales_TimePeriod() {\r\n return {\r\n earlyMorning: \"凌晨\",\r\n morning: \"上午\",\r\n noon: \"中午\",\r\n afternoon: \"下午\",\r\n evening: \"晚上\"\r\n };\r\n}\r\n\r\n/**\r\n * 将任意值安全转换为 Date 对象。\r\n * - 数字/数字字符串:视为时间戳\r\n * - 字符串:尝试按 ISO/RFC 格式解析\r\n * - 对象:优先 valueOf(),再 toString()\r\n * - null / undefined / 无效值:返回 null\r\n *\r\n * @param {*} val - 待转换值\r\n * @returns {Date | null} 有效 Date 或 null\r\n */\r\nexport function toDate(val) {\r\n if (val == null) return null; // null / undefined\r\n if (val instanceof Date) return isNaN(val) ? null : val; // 已是 Date,但需排除 Invalid Date\r\n\r\n // 1. 数字或数字字符串 → 时间戳\r\n if (typeof val === \"number\" || (typeof val === \"string\" && /^-?\\d+(\\.\\d+)?$/.test(val.trim()))) {\r\n const d = new Date(+val);\r\n return isNaN(d) ? null : d;\r\n }\r\n\r\n // 2. 标准 ISO 8601 / RFC 2825 等合法字符串\r\n if (typeof val === \"string\") {\r\n const d = new Date(val);\r\n return isNaN(d) ? null : d; // 非法格式返回 null\r\n }\r\n\r\n // 3. 对象带 valueOf / toString\r\n if (typeof val === \"object\") {\r\n // 优先调用 valueOf(期望返回数字时间戳)\r\n const prim = val.valueOf ? val.valueOf() : Object.prototype.valueOf.call(val);\r\n if (typeof prim === \"number\" && !isNaN(prim)) {\r\n const d = new Date(prim);\r\n return isNaN(d) ? null : d;\r\n }\r\n // 兜底用字符串\r\n const str = val.toString ? val.toString() : String(val);\r\n const d = new Date(str);\r\n return isNaN(d) ? null : d;\r\n }\r\n\r\n // 4. 其余情况\r\n return null;\r\n}\r\n\r\n/**\r\n * 在闭区间 [date1, date2] 内随机生成一个日期(含首尾)。\r\n * 若顺序相反则自动交换。\r\n *\r\n * @param {Date} date1 - 起始日期\r\n * @param {Date} date2 - 结束日期\r\n * @returns {Date} 随机日期\r\n */\r\nexport function randomDateInRange(date1, date2) {\r\n let v1 = date1.getTime(),\r\n v2 = date2.getTime();\r\n if (v1 > v2) [v1, v2] = [v2, v1];\r\n return new Date(v1 + Math.floor(Math.random() * (v2 - v1 + 1)));\r\n}\r\n\r\n/**\r\n * 计算两个时间之间的剩余/已过时长(天-时-分-秒),返回带补零的展示对象。\r\n *\r\n * @param {string|number|Date} originalTime - 原始时间(可转 Date 的任意值)\r\n * @param {Date} [currentTime=new Date()] - 基准时间,默认当前\r\n * @returns {{days:number,hours:string,minutes:string,seconds:string}}\r\n */\r\nexport function calcTimeDifference(originalTime, currentTime = new Date()) {\r\n // 计算时间差(毫秒)\r\n const diffMs = currentTime - new Date(originalTime);\r\n\r\n // 转换为天、小时、分钟、秒\r\n const diffSeconds = Math.floor(diffMs / 1000);\r\n const days = Math.floor(diffSeconds / (3600 * 24));\r\n const hours = Math.floor((diffSeconds % (3600 * 24)) / 3600);\r\n const minutes = Math.floor((diffSeconds % 3600) / 60);\r\n const seconds = diffSeconds % 60;\r\n\r\n const padZero = (num) => String(num).padStart(2, \"0\");\r\n\r\n return {\r\n days,\r\n hours: padZero(hours),\r\n minutes: padZero(minutes),\r\n seconds: padZero(seconds)\r\n };\r\n}\r\n\r\n/**\r\n * 将总秒数格式化成人类可读的时间段文本。\r\n * 固定进制:1 年=365 天,1 月=30 天。\r\n *\r\n * @param {number} totalSeconds - 非负总秒数\r\n * @param {object} [options] - 格式化选项\r\n * @param {Partial<{year:string,month:string,day:string,hour:string,minute:string,second:string}>} [options.labels] - 各单位的自定义文本\r\n * @param {('year'|'month'|'day'|'hour'|'minute'|'second')} [options.maxUnit] - 最大输出单位\r\n * @param {('year'|'month'|'day'|'hour'|'minute'|'second')} [options.minUnit] - 最小输出单位\r\n * @param {boolean} [options.showZero] - 是否强制显示 0 秒\r\n * @returns {string} 拼接后的时长文本,如“1天 02小时 30分钟”\r\n * @throws {TypeError} 当 totalSeconds 为非数字或负数时抛出\r\n */\r\nexport function formatDuration(totalSeconds, options = {}) {\r\n if (typeof totalSeconds !== \"number\" || totalSeconds < 0 || !isFinite(totalSeconds)) {\r\n throw new TypeError(\"totalSeconds 必须是非负数字\");\r\n }\r\n\r\n // 1. 默认中文单位\r\n const DEFAULT_LABELS = {\r\n year: \"年\",\r\n month: \"月\",\r\n day: \"天\",\r\n hour: \"小时\",\r\n minute: \"分钟\",\r\n second: \"秒\"\r\n };\r\n\r\n // 2. 固定进制表(秒)\r\n const UNIT_TABLE = [\r\n { key: \"year\", seconds: 365 * 24 * 3600 },\r\n { key: \"month\", seconds: 30 * 24 * 3600 },\r\n { key: \"day\", seconds: 24 * 3600 },\r\n { key: \"hour\", seconds: 3600 },\r\n { key: \"minute\", seconds: 60 },\r\n { key: \"second\", seconds: 1 }\r\n ];\r\n\r\n // 3. 合并用户自定义文本\r\n const labels = Object.assign({}, DEFAULT_LABELS, options.labels);\r\n\r\n // 4. 根据 maxUnit / minUnit 截取\r\n let start = 0,\r\n end = UNIT_TABLE.length;\r\n if (options.maxUnit) {\r\n const idx = UNIT_TABLE.findIndex((u) => u.key === options.maxUnit);\r\n if (idx !== -1) start = idx;\r\n }\r\n if (options.minUnit) {\r\n const idx = UNIT_TABLE.findIndex((u) => u.key === options.minUnit);\r\n if (idx !== -1) end = idx + 1;\r\n }\r\n const units = UNIT_TABLE.slice(start, end);\r\n if (!units.length) units.push(UNIT_TABLE[UNIT_TABLE.length - 1]); // 保底秒\r\n\r\n // 5. 逐级计算\r\n let rest = Math.floor(totalSeconds);\r\n const parts = [];\r\n\r\n for (const { key, seconds } of units) {\r\n const val = Math.floor(rest / seconds);\r\n rest %= seconds;\r\n\r\n const shouldShow = val > 0 || (options.showZero && key === \"second\");\r\n if (shouldShow || (parts.length === 0 && rest === 0)) {\r\n parts.push(`${val}${labels[key]}`);\r\n }\r\n }\r\n\r\n // 6. 兜底\r\n if (parts.length === 0) {\r\n parts.push(`0${labels[units[units.length - 1].key]}`);\r\n }\r\n\r\n return parts.join(\"\");\r\n}\r\n\r\n/**\r\n * 快捷调用 {@link formatDuration},最大单位到“天”。\r\n *\r\n * @param {number} totalSeconds\r\n * @param {Omit<Parameters<typeof formatDuration>[1],'maxUnit'>} [options]\r\n * @returns {string}\r\n */\r\nexport function formatDurationMaxDay(totalSeconds, options = {}) {\r\n return formatDuration(totalSeconds, { ...options, maxUnit: \"day\" });\r\n}\r\n\r\n/**\r\n * 快捷调用 {@link formatDuration},最大单位到“小时”。\r\n *\r\n * @param {number} totalSeconds\r\n * @param {Omit<Parameters<typeof formatDuration>[1],'maxUnit'>} [options]\r\n * @returns {string}\r\n */\r\nexport function formatDurationMaxHour(totalSeconds, options = {}) {\r\n return formatDuration(totalSeconds, { ...options, maxUnit: \"hour\" });\r\n}\r\n\r\n/**\r\n * 根据小时数返回对应的时间段名称。\r\n *\r\n * @param {number} hour - 24 小时制的小时(0-23)\r\n * @param {object} [locales] - 自定义时段文案\r\n * @param {string} [locales.earlyMorning=\"凌晨\"] - 00-05\r\n * @param {string} [locales.morning=\"上午\"] - 06-11\r\n * @param {string} [locales.noon=\"中午\"] - 12-13\r\n * @param {string} [locales.afternoon=\"下午\"] - 14-17\r\n * @param {string} [locales.evening=\"晚上\"] - 18-23\r\n * @returns {string} 时段名称\r\n * @throws {RangeError} 当 hour 不在 0-23 范围时抛出\r\n */\r\nexport function getTimePeriodName(hour, locales = getLocales_TimePeriod()) {\r\n if (!Number.isInteger(hour) || hour < 0 || hour > 23) {\r\n throw new RangeError(\"hour 必须是 0-23 的整数\");\r\n }\r\n if (hour >= 0 && hour < 6) return locales.earlyMorning;\r\n if (hour < 12) return locales.morning;\r\n if (hour < 14) return locales.noon;\r\n if (hour < 18) return locales.afternoon;\r\n return locales.evening;\r\n}\r\n\r\n/**\r\n * 格式化时间戳为本地化的时间字符串。\r\n *\r\n * @param {number} timestamp - 要格式化的时间戳(毫秒)。\r\n * @param {Object} [locales] - 本地化配置对象,包含时间相关的本地化字符串。\r\n * @param {string} [locales.justNow='刚刚'] - 表示刚刚过去的时间。\r\n * @param {string} [locales.today='今天'] - 表示今天。\r\n * @param {string} [locales.yesterday='昨天'] - 表示昨天。\r\n * @param {string} [locales.beforeYesterday='前天'] - 表示前天。\r\n * @param {string} [locales.year='年'] - 年的单位。\r\n * @param {string} [locales.month='月'] - 月的单位。\r\n * @param {string} [locales.day='日'] - 日的单位。\r\n * @param {Object} [locales.timePeriod] - 一天中不同时间段的本地化字符串。\r\n * @param {string} [locales.timePeriod.earlyMorning='凌晨'] - 凌晨。\r\n * @param {string} [locales.timePeriod.morning='上午'] - 上午。\r\n * @param {string} [locales.timePeriod.noon='中午'] - 中午。\r\n * @param {string} [locales.timePeriod.afternoon='下午'] - 下午。\r\n * @param {string} [locales.timePeriod.evening='晚上'] - 晚上。\r\n * @param {Array<string>} [locales.weekDays=['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六']] - 星期几的本地化字符串数组。\r\n *\r\n * @returns {string} - 格式化后的时间字符串。\r\n */\r\nexport function formatTimeForLocale(\r\n timestamp,\r\n locales = {\r\n justNow: \"刚刚\",\r\n today: \"今天\",\r\n yesterday: \"昨天\",\r\n beforeYesterday: \"前天\",\r\n year: \"年\",\r\n month: \"月\",\r\n day: \"日\",\r\n timePeriod: getLocales_TimePeriod(),\r\n weekDays: [\"星期日\", \"星期一\", \"星期二\", \"星期三\", \"星期四\", \"星期五\", \"星期六\"]\r\n }\r\n) {\r\n const now = new Date();\r\n const messageDate = new Date(timestamp);\r\n\r\n const nowTime = now.getTime();\r\n const msgTime = messageDate.getTime();\r\n const diff = nowTime - msgTime;\r\n const diffMinutes = Math.floor(diff / (1000 * 60));\r\n\r\n const year = messageDate.getFullYear();\r\n const month = messageDate.getMonth() + 1;\r\n const day = messageDate.getDate();\r\n const hour = messageDate.getHours();\r\n const minute = messageDate.getMinutes().toString().padStart(2, \"0\");\r\n\r\n // 刚刚:1分钟内\r\n if (diffMinutes < 1) {\r\n return locales.justNow;\r\n }\r\n\r\n // 计算自然天数差(按日期而非时间差)\r\n const nowDate = new Date(now.getFullYear(), now.getMonth(), now.getDate());\r\n const msgDate = new Date(year, month - 1, day);\r\n const diffDays = Math.floor((nowDate.getTime() - msgDate.getTime()) / (1000 * 60 * 60 * 24));\r\n\r\n const period = getTimePeriodName(hour, locales.timePeriod);\r\n const timeStr = `${hour}:${minute}`;\r\n\r\n // 今天\r\n if (diffDays === 0) {\r\n return `${locales.today} ${period}${timeStr}`;\r\n }\r\n\r\n // 昨天\r\n if (diffDays === 1) {\r\n return `${locales.yesterday} ${period}${timeStr}`;\r\n }\r\n\r\n // 前天\r\n if (diffDays === 2) {\r\n return `${locales.beforeYesterday} ${period}${timeStr}`;\r\n }\r\n\r\n // 本周内(周一到周日,且不是今天/昨天/前天)\r\n // 获取本周一的日期\r\n const nowDay = now.getDay() || 7; // 周日转为7\r\n const msgDay = messageDate.getDay() || 7;\r\n const mondayOfThisWeek = new Date(nowDate.getTime() - (nowDay - 1) * 24 * 60 * 60 * 1000);\r\n const mondayOfThatWeek = new Date(msgDate.getTime() - (msgDay - 1) * 24 * 60 * 60 * 1000);\r\n\r\n if (mondayOfThisWeek.getTime() === mondayOfThatWeek.getTime() && diffDays < 7) {\r\n const weekDayName = locales.weekDays[messageDate.getDay()];\r\n return `${weekDayName}${period} ${timeStr}`;\r\n }\r\n\r\n // 本月内(非本周)\r\n const nowYear = now.getFullYear();\r\n const nowMonth = now.getMonth();\r\n if (year === nowYear && messageDate.getMonth() === nowMonth) {\r\n return `${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;\r\n }\r\n\r\n // 本年内(非本月)\r\n if (year === nowYear) {\r\n return `${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;\r\n }\r\n\r\n // 其他(非本年)\r\n return `${year}${locales.year}${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;\r\n}\r\n"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"date.js","sources":["../src/source/date.js"],"sourcesContent":["function getLocales_TimePeriod() {\r\n return {\r\n earlyMorning: \"凌晨\",\r\n morning: \"上午\",\r\n noon: \"中午\",\r\n afternoon: \"下午\",\r\n evening: \"晚上\"\r\n };\r\n}\r\n\r\n/**\r\n * 将任意值安全转换为 Date 对象。\r\n * - 数字/数字字符串:视为时间戳\r\n * - 字符串:尝试按 ISO/RFC 格式解析\r\n * - 对象:优先 valueOf(),再 toString()\r\n * - null / undefined / 无效值:返回 null\r\n *\r\n * @param {*} val - 待转换值\r\n * @returns {Date | null} 有效 Date 或 null\r\n */\r\nexport function toDate(val) {\r\n if (val == null) return null; // null / undefined\r\n if (val instanceof Date) return isNaN(val) ? null : val; // 已是 Date,但需排除 Invalid Date\r\n\r\n // 1. 数字或数字字符串 → 时间戳\r\n if (typeof val === \"number\" || (typeof val === \"string\" && /^-?\\d+(\\.\\d+)?$/.test(val.trim()))) {\r\n const d = new Date(+val);\r\n return isNaN(d) ? null : d;\r\n }\r\n\r\n // 2. 标准 ISO 8601 / RFC 2825 等合法字符串\r\n if (typeof val === \"string\") {\r\n const d = new Date(val);\r\n return isNaN(d) ? null : d; // 非法格式返回 null\r\n }\r\n\r\n // 3. 对象带 valueOf / toString\r\n if (typeof val === \"object\") {\r\n // 优先调用 valueOf(期望返回数字时间戳)\r\n const prim = val.valueOf ? val.valueOf() : Object.prototype.valueOf.call(val);\r\n if (typeof prim === \"number\" && !isNaN(prim)) {\r\n const d = new Date(prim);\r\n return isNaN(d) ? null : d;\r\n }\r\n // 兜底用字符串\r\n const str = val.toString ? val.toString() : String(val);\r\n const d = new Date(str);\r\n return isNaN(d) ? null : d;\r\n }\r\n\r\n // 4. 其余情况\r\n return null;\r\n}\r\n\r\n/**\r\n * 在闭区间 [date1, date2] 内随机生成一个日期(含首尾)。\r\n * 若顺序相反则自动交换。\r\n *\r\n * @param {Date} date1 - 起始日期\r\n * @param {Date} date2 - 结束日期\r\n * @returns {Date} 随机日期\r\n */\r\nexport function randomDateInRange(date1, date2) {\r\n let v1 = date1.getTime(),\r\n v2 = date2.getTime();\r\n if (v1 > v2) [v1, v2] = [v2, v1];\r\n return new Date(v1 + Math.floor(Math.random() * (v2 - v1 + 1)));\r\n}\r\n\r\n//#region 持续时间相关\r\n\r\n/**\r\n * 时间持续对象(完整版本,包含年月日时分秒毫秒)\r\n * @typedef {Object} DurationObject\r\n * @property {number} years - 年数\r\n * @property {number} months - 月数(0-11)\r\n * @property {number} days - 天数(0-29,取决于 monthDays)\r\n * @property {number} hours - 小时数(0-23)\r\n * @property {number} minutes - 分钟数(0-59)\r\n * @property {number} seconds - 秒数(0-59)\r\n * @property {number} milliseconds - 毫秒数(0-999)\r\n */\r\n\r\n/**\r\n * 时间持续对象(最大单位为天)\r\n * @typedef {Object} DurationMaxDayObject\r\n * @property {number} days - 天数\r\n * @property {number} hours - 小时数(0-23)\r\n * @property {number} minutes - 分钟数(0-59)\r\n * @property {number} seconds - 秒数(0-59)\r\n * @property {number} milliseconds - 毫秒数(0-999)\r\n */\r\n\r\n/**\r\n * 时间持续对象(最大单位为小时)\r\n * @typedef {Object} DurationMaxHourObject\r\n * @property {number} hours - 小时数\r\n * @property {number} minutes - 分钟数(0-59)\r\n * @property {number} seconds - 秒数(0-59)\r\n * @property {number} milliseconds - 毫秒数(0-999)\r\n */\r\n\r\n/**\r\n * 将毫秒转换为时间持续对象。\r\n *\r\n * @param {number} milliseconds - 毫秒数(非负整数)\r\n * @param {Object} [options] - 选项对象。\r\n * @param {number} [options.yearDays=365] - 一年的天数。\r\n * @param {number} [options.monthDays=30] - 一月的天数。\r\n * @returns {DurationObject} 时间持续对象\r\n * @throws {TypeError} 当 milliseconds 不是有效数字\r\n * @throws {RangeError} 当 milliseconds 为负数\r\n * @throws {RangeError} 当 yearDays 或 monthDays 不是正整数\r\n *\r\n * @example\r\n * // 基本用法\r\n * millisecond2Duration(42070000500);\r\n * // 返回: { years: 1, months: 4, days: 1, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }\r\n */\r\nexport function millisecond2Duration(milliseconds, options = { yearDays: 365, monthDays: 30 }) {\r\n // 参数验证\r\n if (typeof milliseconds !== \"number\" || isNaN(milliseconds)) {\r\n throw new TypeError(\"milliseconds must be a valid number\");\r\n }\r\n if (milliseconds < 0) {\r\n throw new RangeError(\"milliseconds must be a non-negative number\");\r\n }\r\n\r\n // 默认选项\r\n const { yearDays = 365, monthDays = 30 } = options;\r\n\r\n // 选项验证\r\n if (!Number.isInteger(yearDays) || yearDays <= 0) {\r\n throw new RangeError(\"yearDays must be a positive integer\");\r\n }\r\n if (!Number.isInteger(monthDays) || monthDays <= 0) {\r\n throw new RangeError(\"monthDays must be a positive integer\");\r\n }\r\n\r\n const totalMilliseconds = Math.floor(milliseconds);\r\n const ms = totalMilliseconds % 1000;\r\n const diffSeconds = Math.floor(totalMilliseconds / 1000);\r\n\r\n const seconds = diffSeconds % 60;\r\n const minutes = Math.floor(diffSeconds / 60) % 60;\r\n const hours = Math.floor(diffSeconds / 3600) % 24;\r\n\r\n // 计算年、月、日\r\n const totalDays = Math.floor(diffSeconds / 3600 / 24);\r\n const years = Math.floor(totalDays / yearDays);\r\n const remainingDays = totalDays % yearDays;\r\n const months = Math.floor(remainingDays / monthDays);\r\n const days = remainingDays % monthDays;\r\n\r\n return { years, months, days, hours, minutes, seconds, milliseconds: ms };\r\n}\r\n\r\n/**\r\n * 将毫秒转换为时间持续对象(最大单位为天)。\r\n * @param {number} milliseconds - 毫秒数(非负整数)\r\n * @returns {DurationMaxDayObject} 包含天、小时、分钟、秒、毫秒的时间持续对象\r\n * @throws {TypeError} 当 milliseconds 不是有效数字时抛出\r\n * @throws {RangeError} 当 milliseconds 为负数时抛出\r\n * @example\r\n * // 返回 { days: 486, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }\r\n * millisecond2DurationMaxDay(42070000500);\r\n */\r\nexport function millisecond2DurationMaxDay(milliseconds) {\r\n if (typeof milliseconds !== \"number\" || isNaN(milliseconds)) {\r\n throw new TypeError(\"milliseconds must be a valid number\");\r\n }\r\n if (milliseconds < 0) {\r\n throw new RangeError(\"milliseconds must be a non-negative number\");\r\n }\r\n\r\n const totalMilliseconds = Math.floor(milliseconds);\r\n const ms = totalMilliseconds % 1000;\r\n const diffSeconds = Math.floor(totalMilliseconds / 1000);\r\n\r\n const seconds = diffSeconds % 60;\r\n const minutes = Math.floor(diffSeconds / 60) % 60;\r\n const hours = Math.floor(diffSeconds / 3600) % 24;\r\n const days = Math.floor(diffSeconds / 3600 / 24);\r\n\r\n return { days, hours, minutes, seconds, milliseconds: ms };\r\n}\r\n\r\n/**\r\n * 将毫秒转换为时间持续对象(最大单位为小时)。\r\n * @param {number} milliseconds - 毫秒数(非负整数)\r\n * @returns {DurationMaxHourObject} 包含小时、分钟、秒、毫秒的时间持续对象\r\n * @throws {TypeError} 当 milliseconds 不是有效数字时抛出\r\n * @throws {RangeError} 当 milliseconds 为负数时抛出\r\n * @example\r\n * // 返回 { hours: 11686, minutes: 6, seconds: 40, milliseconds: 500 }\r\n * millisecond2DurationMaxHour(42070000500);\r\n */\r\nexport function millisecond2DurationMaxHour(milliseconds) {\r\n if (typeof milliseconds !== \"number\" || isNaN(milliseconds)) {\r\n throw new TypeError(\"milliseconds must be a valid number\");\r\n }\r\n if (milliseconds < 0) {\r\n throw new RangeError(\"milliseconds must be a non-negative number\");\r\n }\r\n\r\n const totalMilliseconds = Math.floor(milliseconds);\r\n const ms = totalMilliseconds % 1000;\r\n const diffSeconds = Math.floor(totalMilliseconds / 1000);\r\n\r\n const seconds = diffSeconds % 60;\r\n const minutes = Math.floor(diffSeconds / 60) % 60;\r\n const hours = Math.floor(diffSeconds / 3600);\r\n\r\n return { hours, minutes, seconds, milliseconds: ms };\r\n}\r\n\r\n/**\r\n * 将秒转换为时间持续对象。\r\n *\r\n * @param {number} seconds - 秒数(非负整数)\r\n * @param {Object} [options] - 选项对象。\r\n * @param {number} [options.yearDays=365] - 一年的天数。\r\n * @param {number} [options.monthDays=30] - 一月的天数。\r\n * @returns {DurationObject} 时间持续对象\r\n * @throws {TypeError} 当 seconds 不是有效数字\r\n * @throws {RangeError} 当 seconds 为负数\r\n * @throws {RangeError} 当 yearDays 或 monthDays 不是正整数\r\n *\r\n * @example\r\n * // 基本用法\r\n * second2Duration(42070000.5);\r\n * // 返回: { years: 1, months: 4, days: 1, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }\r\n */\r\nexport function second2Duration(seconds, options = { yearDays: 365, monthDays: 30 }) {\r\n if (typeof seconds !== \"number\" || isNaN(seconds)) {\r\n throw new TypeError(\"seconds must be a valid number\");\r\n }\r\n if (seconds < 0) {\r\n throw new RangeError(\"seconds must be a non-negative number\");\r\n }\r\n return millisecond2Duration(seconds * 1000, options);\r\n}\r\n\r\n/**\r\n * 将秒转换为时间持续对象(最大单位为天)。\r\n * @param {number} seconds - 秒数(非负整数)\r\n * @returns {DurationMaxDayObject} 包含天、小时、分钟、秒、毫秒的时间持续对象\r\n * @throws {TypeError} 当 seconds 不是有效数字时抛出\r\n * @throws {RangeError} 当 seconds 为负数时抛出\r\n * @example\r\n * // 返回 { days: 486, hours: 22, minutes: 6, seconds: 40, milliseconds: 500 }\r\n * second2DurationMaxDay(42070000.5);\r\n */\r\nexport function second2DurationMaxDay(seconds) {\r\n if (typeof seconds !== \"number\" || isNaN(seconds)) {\r\n throw new TypeError(\"seconds must be a valid number\");\r\n }\r\n if (seconds < 0) {\r\n throw new RangeError(\"seconds must be a non-negative number\");\r\n }\r\n return millisecond2DurationMaxDay(seconds * 1000);\r\n}\r\n\r\n/**\r\n * 将秒转换为时间持续对象(最大单位为小时)。\r\n * @param {number} seconds - 秒数(非负整数)\r\n * @returns {DurationMaxHourObject} 包含小时、分钟、秒、毫秒的时间持续对象\r\n * @throws {TypeError} 当 seconds 不是有效数字时抛出\r\n * @throws {RangeError} 当 seconds 为负数时抛出\r\n * @example\r\n * // 返回 { hours: 11686, minutes: 6, seconds: 40, milliseconds: 500 }\r\n * second2DurationMaxHour(42070000.5);\r\n */\r\nexport function second2DurationMaxHour(seconds) {\r\n if (typeof seconds !== \"number\" || isNaN(seconds)) {\r\n throw new TypeError(\"seconds must be a valid number\");\r\n }\r\n if (seconds < 0) {\r\n throw new RangeError(\"seconds must be a non-negative number\");\r\n }\r\n return millisecond2DurationMaxHour(seconds * 1000);\r\n}\r\n\r\n//#endregion\r\n\r\n/**\r\n * 根据小时数返回对应的时间段名称。\r\n *\r\n * @param {number} hour - 24 小时制的小时(0-23)\r\n * @param {object} [locales] - 自定义时段文案\r\n * @param {string} [locales.earlyMorning=\"凌晨\"] - 00-05\r\n * @param {string} [locales.morning=\"上午\"] - 06-11\r\n * @param {string} [locales.noon=\"中午\"] - 12-13\r\n * @param {string} [locales.afternoon=\"下午\"] - 14-17\r\n * @param {string} [locales.evening=\"晚上\"] - 18-23\r\n * @returns {string} 时段名称\r\n * @throws {RangeError} 当 hour 不在 0-23 范围时抛出\r\n */\r\nexport function getTimePeriodName(hour, locales = getLocales_TimePeriod()) {\r\n if (!Number.isInteger(hour) || hour < 0 || hour > 23) {\r\n throw new RangeError(\"hour 必须是 0-23 的整数\");\r\n }\r\n if (hour >= 0 && hour < 6) return locales.earlyMorning;\r\n if (hour < 12) return locales.morning;\r\n if (hour < 14) return locales.noon;\r\n if (hour < 18) return locales.afternoon;\r\n return locales.evening;\r\n}\r\n\r\n/**\r\n * 格式化时间戳为本地化的时间字符串。\r\n *\r\n * @param {number} timestamp - 要格式化的时间戳(毫秒)。\r\n * @param {Object} [locales] - 本地化配置对象,包含时间相关的本地化字符串。\r\n * @param {string} [locales.justNow='刚刚'] - 表示刚刚过去的时间。\r\n * @param {string} [locales.today='今天'] - 表示今天。\r\n * @param {string} [locales.yesterday='昨天'] - 表示昨天。\r\n * @param {string} [locales.beforeYesterday='前天'] - 表示前天。\r\n * @param {string} [locales.year='年'] - 年的单位。\r\n * @param {string} [locales.month='月'] - 月的单位。\r\n * @param {string} [locales.day='日'] - 日的单位。\r\n * @param {Object} [locales.timePeriod] - 一天中不同时间段的本地化字符串。\r\n * @param {string} [locales.timePeriod.earlyMorning='凌晨'] - 凌晨。\r\n * @param {string} [locales.timePeriod.morning='上午'] - 上午。\r\n * @param {string} [locales.timePeriod.noon='中午'] - 中午。\r\n * @param {string} [locales.timePeriod.afternoon='下午'] - 下午。\r\n * @param {string} [locales.timePeriod.evening='晚上'] - 晚上。\r\n * @param {Array<string>} [locales.weekDays=['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六']] - 星期几的本地化字符串数组。\r\n *\r\n * @returns {string} - 格式化后的时间字符串。\r\n */\r\nexport function formatTimeForLocale(\r\n timestamp,\r\n locales = {\r\n justNow: \"刚刚\",\r\n today: \"今天\",\r\n yesterday: \"昨天\",\r\n beforeYesterday: \"前天\",\r\n year: \"年\",\r\n month: \"月\",\r\n day: \"日\",\r\n timePeriod: getLocales_TimePeriod(),\r\n weekDays: [\"星期日\", \"星期一\", \"星期二\", \"星期三\", \"星期四\", \"星期五\", \"星期六\"]\r\n }\r\n) {\r\n const now = new Date();\r\n const messageDate = new Date(timestamp);\r\n\r\n const nowTime = now.getTime();\r\n const msgTime = messageDate.getTime();\r\n const diff = nowTime - msgTime;\r\n const diffMinutes = Math.floor(diff / (1000 * 60));\r\n\r\n const year = messageDate.getFullYear();\r\n const month = messageDate.getMonth() + 1;\r\n const day = messageDate.getDate();\r\n const hour = messageDate.getHours();\r\n const minute = messageDate.getMinutes().toString().padStart(2, \"0\");\r\n\r\n // 刚刚:1分钟内\r\n if (diffMinutes < 1) {\r\n return locales.justNow;\r\n }\r\n\r\n // 计算自然天数差(按日期而非时间差)\r\n const nowDate = new Date(now.getFullYear(), now.getMonth(), now.getDate());\r\n const msgDate = new Date(year, month - 1, day);\r\n const diffDays = Math.floor((nowDate.getTime() - msgDate.getTime()) / (1000 * 60 * 60 * 24));\r\n\r\n const period = getTimePeriodName(hour, locales.timePeriod);\r\n const timeStr = `${hour}:${minute}`;\r\n\r\n // 今天\r\n if (diffDays === 0) {\r\n return `${locales.today} ${period}${timeStr}`;\r\n }\r\n\r\n // 昨天\r\n if (diffDays === 1) {\r\n return `${locales.yesterday} ${period}${timeStr}`;\r\n }\r\n\r\n // 前天\r\n if (diffDays === 2) {\r\n return `${locales.beforeYesterday} ${period}${timeStr}`;\r\n }\r\n\r\n // 本周内(周一到周日,且不是今天/昨天/前天)\r\n // 获取本周一的日期\r\n const nowDay = now.getDay() || 7; // 周日转为7\r\n const msgDay = messageDate.getDay() || 7;\r\n const mondayOfThisWeek = new Date(nowDate.getTime() - (nowDay - 1) * 24 * 60 * 60 * 1000);\r\n const mondayOfThatWeek = new Date(msgDate.getTime() - (msgDay - 1) * 24 * 60 * 60 * 1000);\r\n\r\n if (mondayOfThisWeek.getTime() === mondayOfThatWeek.getTime() && diffDays < 7) {\r\n const weekDayName = locales.weekDays[messageDate.getDay()];\r\n return `${weekDayName}${period} ${timeStr}`;\r\n }\r\n\r\n // 本月内(非本周)\r\n const nowYear = now.getFullYear();\r\n const nowMonth = now.getMonth();\r\n if (year === nowYear && messageDate.getMonth() === nowMonth) {\r\n return `${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;\r\n }\r\n\r\n // 本年内(非本月)\r\n if (year === nowYear) {\r\n return `${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;\r\n }\r\n\r\n // 其他(非本年)\r\n return `${year}${locales.year}${month}${locales.month}${day}${locales.day} ${period}${timeStr}`;\r\n}\r\n"],"names":[],"mappings":"AAAA,SAAS,qBAAqB,GAAG;AACjC,IAAI,OAAO;AACX,QAAQ,YAAY,EAAE,IAAI;AAC1B,QAAQ,OAAO,EAAE,IAAI;AACrB,QAAQ,IAAI,EAAE,IAAI;AAClB,QAAQ,SAAS,EAAE,IAAI;AACvB,QAAQ,OAAO,EAAE,IAAI;AACrB,KAAK,CAAC;AACN,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,MAAM,CAAC,GAAG,EAAE;AAC5B,IAAI,IAAI,GAAG,IAAI,IAAI,EAAE,OAAO,IAAI,CAAC;AACjC,IAAI,IAAI,GAAG,YAAY,IAAI,EAAE,OAAO,KAAK,CAAC,GAAG,CAAC,GAAG,IAAI,GAAG,GAAG,CAAC;AAC5D;AACA;AACA,IAAI,IAAI,OAAO,GAAG,KAAK,QAAQ,KAAK,OAAO,GAAG,KAAK,QAAQ,IAAI,iBAAiB,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE;AACpG,QAAQ,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;AACjC,QAAQ,OAAO,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC;AACnC,IAAI,CAAC;AACL;AACA;AACA,IAAI,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE;AACjC,QAAQ,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC;AAChC,QAAQ,OAAO,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC;AACnC,IAAI,CAAC;AACL;AACA;AACA,IAAI,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE;AACjC;AACA,QAAQ,MAAM,IAAI,GAAG,GAAG,CAAC,OAAO,GAAG,GAAG,CAAC,OAAO,EAAE,GAAG,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AACtF,QAAQ,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE;AACtD,YAAY,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC;AACrC,YAAY,OAAO,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC;AACvC,QAAQ,CAAC;AACT;AACA,QAAQ,MAAM,GAAG,GAAG,GAAG,CAAC,QAAQ,GAAG,GAAG,CAAC,QAAQ,EAAE,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;AAChE,QAAQ,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC;AAChC,QAAQ,OAAO,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC;AACnC,IAAI,CAAC;AACL;AACA;AACA,IAAI,OAAO,IAAI,CAAC;AAChB,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,iBAAiB,CAAC,KAAK,EAAE,KAAK,EAAE;AAChD,IAAI,IAAI,EAAE,GAAG,KAAK,CAAC,OAAO,EAAE;AAC5B,QAAQ,EAAE,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC;AAC7B,IAAI,IAAI,EAAE,GAAG,EAAE,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;AACrC,IAAI,OAAO,IAAI,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACpE,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,oBAAoB,CAAC,YAAY,EAAE,OAAO,GAAG,EAAE,QAAQ,EAAE,GAAG,EAAE,SAAS,EAAE,EAAE,EAAE,EAAE;AAC/F;AACA,IAAI,IAAI,OAAO,YAAY,KAAK,QAAQ,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE;AACjE,QAAQ,MAAM,IAAI,SAAS,CAAC,qCAAqC,CAAC,CAAC;AACnE,IAAI,CAAC;AACL,IAAI,IAAI,YAAY,GAAG,CAAC,EAAE;AAC1B,QAAQ,MAAM,IAAI,UAAU,CAAC,4CAA4C,CAAC,CAAC;AAC3E,IAAI,CAAC;AACL;AACA;AACA,IAAI,MAAM,EAAE,QAAQ,GAAG,GAAG,EAAE,SAAS,GAAG,EAAE,EAAE,GAAG,OAAO,CAAC;AACvD;AACA;AACA,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,QAAQ,IAAI,CAAC,EAAE;AACtD,QAAQ,MAAM,IAAI,UAAU,CAAC,qCAAqC,CAAC,CAAC;AACpE,IAAI,CAAC;AACL,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,SAAS,IAAI,CAAC,EAAE;AACxD,QAAQ,MAAM,IAAI,UAAU,CAAC,sCAAsC,CAAC,CAAC;AACrE,IAAI,CAAC;AACL;AACA,IAAI,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;AACvD,IAAI,MAAM,EAAE,GAAG,iBAAiB,GAAG,IAAI,CAAC;AACxC,IAAI,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAAC;AAC7D;AACA,IAAI,MAAM,OAAO,GAAG,WAAW,GAAG,EAAE,CAAC;AACrC,IAAI,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;AACtD,IAAI,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;AACtD;AACA;AACA,IAAI,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC;AAC1D,IAAI,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,QAAQ,CAAC,CAAC;AACnD,IAAI,MAAM,aAAa,GAAG,SAAS,GAAG,QAAQ,CAAC;AAC/C,IAAI,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,aAAa,GAAG,SAAS,CAAC,CAAC;AACzD,IAAI,MAAM,IAAI,GAAG,aAAa,GAAG,SAAS,CAAC;AAC3C;AACA,IAAI,OAAO,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;AAC9E,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,0BAA0B,CAAC,YAAY,EAAE;AACzD,IAAI,IAAI,OAAO,YAAY,KAAK,QAAQ,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE;AACjE,QAAQ,MAAM,IAAI,SAAS,CAAC,qCAAqC,CAAC,CAAC;AACnE,IAAI,CAAC;AACL,IAAI,IAAI,YAAY,GAAG,CAAC,EAAE;AAC1B,QAAQ,MAAM,IAAI,UAAU,CAAC,4CAA4C,CAAC,CAAC;AAC3E,IAAI,CAAC;AACL;AACA,IAAI,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;AACvD,IAAI,MAAM,EAAE,GAAG,iBAAiB,GAAG,IAAI,CAAC;AACxC,IAAI,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAAC;AAC7D;AACA,IAAI,MAAM,OAAO,GAAG,WAAW,GAAG,EAAE,CAAC;AACrC,IAAI,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;AACtD,IAAI,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;AACtD,IAAI,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC;AACrD;AACA,IAAI,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;AAC/D,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,2BAA2B,CAAC,YAAY,EAAE;AAC1D,IAAI,IAAI,OAAO,YAAY,KAAK,QAAQ,IAAI,KAAK,CAAC,YAAY,CAAC,EAAE;AACjE,QAAQ,MAAM,IAAI,SAAS,CAAC,qCAAqC,CAAC,CAAC;AACnE,IAAI,CAAC;AACL,IAAI,IAAI,YAAY,GAAG,CAAC,EAAE;AAC1B,QAAQ,MAAM,IAAI,UAAU,CAAC,4CAA4C,CAAC,CAAC;AAC3E,IAAI,CAAC;AACL;AACA,IAAI,MAAM,iBAAiB,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;AACvD,IAAI,MAAM,EAAE,GAAG,iBAAiB,GAAG,IAAI,CAAC;AACxC,IAAI,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAAC;AAC7D;AACA,IAAI,MAAM,OAAO,GAAG,WAAW,GAAG,EAAE,CAAC;AACrC,IAAI,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,EAAE,CAAC,GAAG,EAAE,CAAC;AACtD,IAAI,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,GAAG,IAAI,CAAC,CAAC;AACjD;AACA,IAAI,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;AACzD,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,eAAe,CAAC,OAAO,EAAE,OAAO,GAAG,EAAE,QAAQ,EAAE,GAAG,EAAE,SAAS,EAAE,EAAE,EAAE,EAAE;AACrF,IAAI,IAAI,OAAO,OAAO,KAAK,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,EAAE;AACvD,QAAQ,MAAM,IAAI,SAAS,CAAC,gCAAgC,CAAC,CAAC;AAC9D,IAAI,CAAC;AACL,IAAI,IAAI,OAAO,GAAG,CAAC,EAAE;AACrB,QAAQ,MAAM,IAAI,UAAU,CAAC,uCAAuC,CAAC,CAAC;AACtE,IAAI,CAAC;AACL,IAAI,OAAO,oBAAoB,CAAC,OAAO,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC;AACzD,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,qBAAqB,CAAC,OAAO,EAAE;AAC/C,IAAI,IAAI,OAAO,OAAO,KAAK,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,EAAE;AACvD,QAAQ,MAAM,IAAI,SAAS,CAAC,gCAAgC,CAAC,CAAC;AAC9D,IAAI,CAAC;AACL,IAAI,IAAI,OAAO,GAAG,CAAC,EAAE;AACrB,QAAQ,MAAM,IAAI,UAAU,CAAC,uCAAuC,CAAC,CAAC;AACtE,IAAI,CAAC;AACL,IAAI,OAAO,0BAA0B,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC;AACtD,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,sBAAsB,CAAC,OAAO,EAAE;AAChD,IAAI,IAAI,OAAO,OAAO,KAAK,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,EAAE;AACvD,QAAQ,MAAM,IAAI,SAAS,CAAC,gCAAgC,CAAC,CAAC;AAC9D,IAAI,CAAC;AACL,IAAI,IAAI,OAAO,GAAG,CAAC,EAAE;AACrB,QAAQ,MAAM,IAAI,UAAU,CAAC,uCAAuC,CAAC,CAAC;AACtE,IAAI,CAAC;AACL,IAAI,OAAO,2BAA2B,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC;AACvD,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,iBAAiB,CAAC,IAAI,EAAE,OAAO,GAAG,qBAAqB,EAAE,EAAE;AAC3E,IAAI,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,IAAI,GAAG,CAAC,IAAI,IAAI,GAAG,EAAE,EAAE;AAC1D,QAAQ,MAAM,IAAI,UAAU,CAAC,mBAAmB,CAAC,CAAC;AAClD,IAAI,CAAC;AACL,IAAI,IAAI,IAAI,IAAI,CAAC,IAAI,IAAI,GAAG,CAAC,EAAE,OAAO,OAAO,CAAC,YAAY,CAAC;AAC3D,IAAI,IAAI,IAAI,GAAG,EAAE,EAAE,OAAO,OAAO,CAAC,OAAO,CAAC;AAC1C,IAAI,IAAI,IAAI,GAAG,EAAE,EAAE,OAAO,OAAO,CAAC,IAAI,CAAC;AACvC,IAAI,IAAI,IAAI,GAAG,EAAE,EAAE,OAAO,OAAO,CAAC,SAAS,CAAC;AAC5C,IAAI,OAAO,OAAO,CAAC,OAAO,CAAC;AAC3B,CAAC;AACD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACO,SAAS,mBAAmB;AACnC,IAAI,SAAS;AACb,IAAI,OAAO,GAAG;AACd,QAAQ,OAAO,EAAE,IAAI;AACrB,QAAQ,KAAK,EAAE,IAAI;AACnB,QAAQ,SAAS,EAAE,IAAI;AACvB,QAAQ,eAAe,EAAE,IAAI;AAC7B,QAAQ,IAAI,EAAE,GAAG;AACjB,QAAQ,KAAK,EAAE,GAAG;AAClB,QAAQ,GAAG,EAAE,GAAG;AAChB,QAAQ,UAAU,EAAE,qBAAqB,EAAE;AAC3C,QAAQ,QAAQ,EAAE,CAAC,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC;AACnE,KAAK;AACL,EAAE;AACF,IAAI,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;AAC3B,IAAI,MAAM,WAAW,GAAG,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC;AAC5C;AACA,IAAI,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,EAAE,CAAC;AAClC,IAAI,MAAM,OAAO,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;AAC1C,IAAI,MAAM,IAAI,GAAG,OAAO,GAAG,OAAO,CAAC;AACnC,IAAI,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,IAAI,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC;AACvD;AACA,IAAI,MAAM,IAAI,GAAG,WAAW,CAAC,WAAW,EAAE,CAAC;AAC3C,IAAI,MAAM,KAAK,GAAG,WAAW,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;AAC7C,IAAI,MAAM,GAAG,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC;AACtC,IAAI,MAAM,IAAI,GAAG,WAAW,CAAC,QAAQ,EAAE,CAAC;AACxC,IAAI,MAAM,MAAM,GAAG,WAAW,CAAC,UAAU,EAAE,CAAC,QAAQ,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;AACxE;AACA;AACA,IAAI,IAAI,WAAW,GAAG,CAAC,EAAE;AACzB,QAAQ,OAAO,OAAO,CAAC,OAAO,CAAC;AAC/B,IAAI,CAAC;AACL;AACA;AACA,IAAI,MAAM,OAAO,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,WAAW,EAAE,EAAE,GAAG,CAAC,QAAQ,EAAE,EAAE,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;AAC/E,IAAI,MAAM,OAAO,GAAG,IAAI,IAAI,CAAC,IAAI,EAAE,KAAK,GAAG,CAAC,EAAE,GAAG,CAAC,CAAC;AACnD,IAAI,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,OAAO,EAAE,KAAK,IAAI,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC,CAAC;AACjG;AACA,IAAI,MAAM,MAAM,GAAG,iBAAiB,CAAC,IAAI,EAAE,OAAO,CAAC,UAAU,CAAC,CAAC;AAC/D,IAAI,MAAM,OAAO,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC;AACxC;AACA;AACA,IAAI,IAAI,QAAQ,KAAK,CAAC,EAAE;AACxB,QAAQ,OAAO,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AACtD,IAAI,CAAC;AACL;AACA;AACA,IAAI,IAAI,QAAQ,KAAK,CAAC,EAAE;AACxB,QAAQ,OAAO,CAAC,EAAE,OAAO,CAAC,SAAS,CAAC,CAAC,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AAC1D,IAAI,CAAC;AACL;AACA;AACA,IAAI,IAAI,QAAQ,KAAK,CAAC,EAAE;AACxB,QAAQ,OAAO,CAAC,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AAChE,IAAI,CAAC;AACL;AACA;AACA;AACA,IAAI,MAAM,MAAM,GAAG,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;AACrC,IAAI,MAAM,MAAM,GAAG,WAAW,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;AAC7C,IAAI,MAAM,gBAAgB,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;AAC9F,IAAI,MAAM,gBAAgB,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;AAC9F;AACA,IAAI,IAAI,gBAAgB,CAAC,OAAO,EAAE,KAAK,gBAAgB,CAAC,OAAO,EAAE,IAAI,QAAQ,GAAG,CAAC,EAAE;AACnF,QAAQ,MAAM,WAAW,GAAG,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC;AACnE,QAAQ,OAAO,CAAC,EAAE,WAAW,CAAC,EAAE,MAAM,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AACpD,IAAI,CAAC;AACL;AACA;AACA,IAAI,MAAM,OAAO,GAAG,GAAG,CAAC,WAAW,EAAE,CAAC;AACtC,IAAI,MAAM,QAAQ,GAAG,GAAG,CAAC,QAAQ,EAAE,CAAC;AACpC,IAAI,IAAI,IAAI,KAAK,OAAO,IAAI,WAAW,CAAC,QAAQ,EAAE,KAAK,QAAQ,EAAE;AACjE,QAAQ,OAAO,CAAC,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,EAAE,GAAG,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AAClF,IAAI,CAAC;AACL;AACA;AACA,IAAI,IAAI,IAAI,KAAK,OAAO,EAAE;AAC1B,QAAQ,OAAO,CAAC,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,EAAE,GAAG,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AAClF,IAAI,CAAC;AACL;AACA;AACA,IAAI,OAAO,CAAC,EAAE,IAAI,CAAC,EAAE,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,CAAC,EAAE,OAAO,CAAC,KAAK,CAAC,EAAE,GAAG,CAAC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AACpG;;;;"}
package/dist/menu.cjs ADDED
@@ -0,0 +1,202 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * 处理数据库菜单项,生成 UI 菜单树和路由配置。
5
+ *
6
+ * @param {Array<Object>} menuItems - 原始菜单项数组,树形结构
7
+ * @param {Object} [options] - 配置选项
8
+ * @param {string} [options.idKey='id'] - 数据源 ID 键名
9
+ * @param {string} [options.codeKey='code'] - 数据源编码键名, 用于拼接路由名称和路由路径
10
+ * @param {string} [options.labelKey='text'] - 数据源标签键名
11
+ * @param {string} [options.childrenKey='children'] - 数据源子节点键名
12
+ * @param {string} [options.extendKey='extend'] - 数据源扩展对象键名
13
+ * @param {string} [options.menuIdKey='key'] - 输出菜单 ID 键名
14
+ * @param {string} [options.menuLabelKey='label'] - 输出菜单标签键名
15
+ * @param {string} [options.menuChildrenKey='children'] - 输出菜单子节点键名
16
+ * @param {string} [options.menuExtendKey='extend'] - 输出菜单扩展对象键名
17
+ * @param {string} [options.routeNameKey='name'] - 路由名称键名
18
+ * @param {string} [options.routePathKey='path'] - 路由路径键名
19
+ * @param {string} [options.routeMetaKey='meta'] - 路由元数据键名
20
+ * @param {Function} [options.handleNodeItem] - 节点处理钩子,参数:(nodeSimple) => void
21
+ * @param {Function} [options.handleMenuItem] - 菜单项处理钩子,参数:(uiMenuItem, nodeSimple) => void
22
+ * @param {Function} [options.handleRouteItem] - 路由项处理钩子,参数:(routeItem, nodeSimple) => void
23
+ *
24
+ * @returns {Object} 处理结果
25
+ * @returns {Array<Object>} returns.uiMenuItems - UI 菜单树形结构数组
26
+ * @returns {Array<Object>} returns.routeItems - 扁平化路由配置数组
27
+ * @returns {Map<string, Object>} returns.nodeMap - 节点 ID 到节点数据的映射表
28
+ *
29
+ * @example
30
+ * const menuItems = [
31
+ * { id: '1', code: 'system', text: '系统管理', children: [
32
+ * { id: '1-1', code: 'user', text: '用户管理' }
33
+ * ]}
34
+ * ];
35
+ *
36
+ * const result = handleDbMenuItems(menuItems, {
37
+ * handleRouteItem: (route, node) => {
38
+ * route.component = () => import(`./views/${node.code}.vue`);
39
+ * }
40
+ * });
41
+ *
42
+ * // result.uiMenuItems: [{ key: '1', label: '系统管理', children: [...] }]
43
+ * // result.routeItems: [{ name: 'system.user', path: '/system/user', meta: { id: '1-1' } }]
44
+ * // result.nodeMap: Map { '1' => {...}, '1-1' => {...} }
45
+ */
46
+ function handleDbMenuItems(menuItems, options = {}) {
47
+ // 1. 统一配置项,简化调用
48
+ const {
49
+ // 数据源键名
50
+ idKey = "id",
51
+ codeKey = "code",
52
+ labelKey = "text",
53
+ childrenKey = "children",
54
+ extendKey = "extend",
55
+
56
+ // 输出目标键名
57
+ menuIdKey = "key",
58
+ menuLabelKey = "label",
59
+ menuChildrenKey = "children",
60
+ menuExtendKey = "extend",
61
+
62
+ routeNameKey = "name",
63
+ routePathKey = "path",
64
+ routeMetaKey = "meta",
65
+
66
+ // 钩子函数
67
+ handleNodeItem,
68
+ handleMenuItem,
69
+ handleRouteItem
70
+ } = options;
71
+
72
+ const uiMenuItems = [];
73
+ const routeItems = [];
74
+ const nodeMap = new Map(); // id -> node (纯净的节点数据)
75
+ const nodePathMap = new Map(); // id -> [{ code, id }] (仅存储路径计算所需的轻量数据)
76
+
77
+ /* ---------- 1. 建立索引:扁平化节点 & 构建路径索引 ---------- */
78
+ // 使用栈进行深度优先遍历,记录路径
79
+ const stack = [];
80
+ if (Array.isArray(menuItems)) {
81
+ menuItems.forEach((n) => stack.push({ node: n, parentPath: [] }));
82
+ }
83
+
84
+ while (stack.length) {
85
+ const { node, parentPath } = stack.pop();
86
+ const idValue = node[idKey];
87
+
88
+ // 构建当前节点的路径片段
89
+ const currentPathSegment = { [idKey]: idValue };
90
+ const fullPath = [...parentPath, currentPathSegment];
91
+
92
+ // 存储路径信息用于后续路由生成
93
+ nodePathMap.set(idValue, fullPath);
94
+
95
+ // 创建轻量级的节点对象存入 nodeMap
96
+ // 注意:这里先创建基础结构,路由信息在第二步补充
97
+ const nodeSimple = {
98
+ ...node,
99
+ [extendKey]: {}, // 初始化扩展对象
100
+ [childrenKey]: null // 去除原始 children,避免引用污染
101
+ };
102
+ nodeMap.set(idValue, nodeSimple);
103
+
104
+ // 处理子节点
105
+ const childrenNodes = node[childrenKey];
106
+ if (Array.isArray(childrenNodes)) {
107
+ // 逆序压栈,保持原序
108
+ for (let i = childrenNodes.length - 1; i >= 0; i--) {
109
+ stack.push({ node: childrenNodes[i], parentPath: fullPath });
110
+ }
111
+ }
112
+ }
113
+
114
+ /* ---------- 2. 构建树形结构 & 生成路由 ---------- */
115
+ const buildStack = [];
116
+ if (Array.isArray(menuItems)) {
117
+ for (let i = menuItems.length - 1; i >= 0; i--) {
118
+ buildStack.push({ node: menuItems[i], parentUi: undefined });
119
+ }
120
+ }
121
+
122
+ while (buildStack.length) {
123
+ const { node, parentUi } = buildStack.pop();
124
+ const idValue = node[idKey];
125
+
126
+ // 从 Map 中取出之前创建好的节点对象
127
+ const nodeSimple = nodeMap.get(idValue);
128
+
129
+ // 2.1 构建 UI 菜单项
130
+ const uiMenuItem = {
131
+ [menuIdKey]: idValue,
132
+ [menuLabelKey]: node[labelKey],
133
+ [menuExtendKey]: {} // UI 菜单专用扩展
134
+ };
135
+
136
+ // 处理子节点
137
+ const childrenNodes = node[childrenKey];
138
+ const hasChildren = Array.isArray(childrenNodes) && childrenNodes.length > 0;
139
+
140
+ if (hasChildren) {
141
+ // 有子节点 -> 继续压栈,传递当前 uiMenuItem 作为父级
142
+ for (let i = childrenNodes.length - 1; i >= 0; i--) {
143
+ buildStack.push({ node: childrenNodes[i], parentUi: uiMenuItem });
144
+ }
145
+ } else {
146
+ // 2.2 无子节点 -> 生成路由配置
147
+
148
+ // 提取 code 和 id 数组
149
+ const codePaths = [],
150
+ keyPaths = [];
151
+ nodePathMap.get(idValue).forEach((item) => {
152
+ const idValue = item[idKey];
153
+ const nodeSimple = nodeMap.get(idValue);
154
+ keyPaths.push(idValue);
155
+ codePaths.push(nodeSimple[codeKey]);
156
+ });
157
+
158
+ const routeName = codePaths.join(".");
159
+ const routePath = "/" + codePaths.join("/");
160
+
161
+ const routeItem = {
162
+ [routeNameKey]: routeName,
163
+ [routePathKey]: routePath,
164
+ [routeMetaKey]: { [idKey]: idValue }
165
+ };
166
+
167
+ // 执行路由钩子
168
+ if (typeof handleRouteItem === "function") {
169
+ handleRouteItem(routeItem, nodeSimple);
170
+ }
171
+ routeItems.push(routeItem);
172
+
173
+ // 补充扩展信息
174
+ uiMenuItem[menuExtendKey].routeName = routeName;
175
+ uiMenuItem[menuExtendKey].keyPaths = keyPaths;
176
+
177
+ nodeSimple[extendKey].routeName = routeName;
178
+ nodeSimple[extendKey].keyPaths = keyPaths;
179
+ }
180
+
181
+ // 2.3 挂载到父级 UI 或根数组
182
+ if (parentUi) {
183
+ // 逻辑赋值,确保 children 数组存在
184
+ (parentUi[menuChildrenKey] || (parentUi[menuChildrenKey] = [])).push(uiMenuItem);
185
+ } else {
186
+ if (typeof handleMenuItem === "function") {
187
+ handleMenuItem(uiMenuItem, nodeSimple);
188
+ }
189
+ uiMenuItems.push(uiMenuItem);
190
+ }
191
+
192
+ // 执行节点钩子
193
+ if (typeof handleNodeItem === "function") {
194
+ handleNodeItem(nodeSimple);
195
+ }
196
+ }
197
+
198
+ return { uiMenuItems, routeItems, nodeMap };
199
+ }
200
+
201
+ exports.handleDbMenuItems = handleDbMenuItems;
202
+ //# sourceMappingURL=menu.cjs.map