@cloudcome/utils-core 0.0.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.
Files changed (107) hide show
  1. package/CHANGELOG.md +52 -0
  2. package/LICENSE +21 -0
  3. package/package.json +277 -6
  4. package/src/array.ts +312 -0
  5. package/src/async.ts +379 -0
  6. package/src/base64.ts +20 -0
  7. package/src/cache.ts +146 -0
  8. package/src/color/contrast.ts +20 -0
  9. package/src/color/distance.ts +28 -0
  10. package/src/color/helpers.ts +23 -0
  11. package/src/color/hex-hsl.ts +11 -0
  12. package/src/color/hex-hsv.ts +28 -0
  13. package/src/color/hex-hwb.ts +31 -0
  14. package/src/color/hex-rgb.ts +39 -0
  15. package/src/color/hsl-lighten.ts +15 -0
  16. package/src/color/hsv-brighten.ts +17 -0
  17. package/src/color/luminance.ts +17 -0
  18. package/src/color/mix.ts +26 -0
  19. package/src/color/rgb-hsl.ts +53 -0
  20. package/src/color/rgb-hsv.ts +52 -0
  21. package/src/color/rgb-hwb.ts +56 -0
  22. package/src/color/rgb-lab.ts +33 -0
  23. package/src/color/rgb-whiter.ts +22 -0
  24. package/src/color/rgb-xyz.ts +62 -0
  25. package/src/color/types.ts +65 -0
  26. package/src/color/xyz-lab.ts +54 -0
  27. package/src/color.ts +19 -0
  28. package/src/crypto/md5.mjs +357 -0
  29. package/src/crypto/sha1.mjs +300 -0
  30. package/src/crypto/sha256.mjs +310 -0
  31. package/src/crypto/sha512.mjs +459 -0
  32. package/src/crypto.ts +60 -0
  33. package/src/date/const.ts +6 -0
  34. package/src/date/core.ts +162 -0
  35. package/src/date/days.ts +51 -0
  36. package/src/date/is.ts +186 -0
  37. package/src/date/relative.ts +92 -0
  38. package/src/date/start-end.ts +246 -0
  39. package/src/date/timezone.ts +220 -0
  40. package/src/date/weeks.ts +100 -0
  41. package/src/date.ts +8 -0
  42. package/src/dict.ts +1 -0
  43. package/src/dts/global.d.ts +27 -0
  44. package/src/easing.ts +166 -0
  45. package/src/emitter.ts +117 -0
  46. package/src/enum.ts +171 -0
  47. package/src/env.ts +62 -0
  48. package/src/error.ts +31 -0
  49. package/src/exception.ts +68 -0
  50. package/src/fn.ts +197 -0
  51. package/src/index.ts +1 -0
  52. package/src/number.ts +236 -0
  53. package/src/object/each.ts +56 -0
  54. package/src/object/get-set.ts +273 -0
  55. package/src/object/is.ts +128 -0
  56. package/src/object/merge.ts +180 -0
  57. package/src/object/process.ts +80 -0
  58. package/src/object.ts +5 -0
  59. package/src/path.ts +188 -0
  60. package/src/promise.ts +111 -0
  61. package/src/qs.ts +119 -0
  62. package/src/regexp.ts +156 -0
  63. package/src/string.ts +146 -0
  64. package/src/time/from.ts +57 -0
  65. package/src/time/to.ts +106 -0
  66. package/src/time.ts +2 -0
  67. package/src/timer.ts +226 -0
  68. package/src/tree.ts +394 -0
  69. package/src/type.ts +197 -0
  70. package/src/types.ts +78 -0
  71. package/src/unique.ts +77 -0
  72. package/src/url.ts +93 -0
  73. package/src/version.ts +71 -0
  74. package/test/array.test.ts +332 -0
  75. package/test/async-real.test.ts +39 -0
  76. package/test/async.test.ts +375 -0
  77. package/test/base64.test.ts +32 -0
  78. package/test/cache.test.ts +83 -0
  79. package/test/color.test.ts +163 -0
  80. package/test/crypto.test.ts +34 -0
  81. package/test/date-tz.test.ts +206 -0
  82. package/test/date.test.ts +353 -0
  83. package/test/easing.test.ts +33 -0
  84. package/test/emitter.test.ts +71 -0
  85. package/test/enum.test.ts +113 -0
  86. package/test/env.test.ts +69 -0
  87. package/test/error.test.ts +58 -0
  88. package/test/exception.test.ts +43 -0
  89. package/test/fn.test.ts +263 -0
  90. package/test/helpers.ts +23 -0
  91. package/test/index.test.ts +6 -0
  92. package/test/number.test.ts +213 -0
  93. package/test/object.test.ts +309 -0
  94. package/test/path.test.ts +156 -0
  95. package/test/promise.test.ts +199 -0
  96. package/test/qs.test.ts +79 -0
  97. package/test/regexp.test.ts +97 -0
  98. package/test/string.test.ts +150 -0
  99. package/test/time.test.ts +214 -0
  100. package/test/timer.test.ts +114 -0
  101. package/test/tree.test.ts +348 -0
  102. package/test/type.test.ts +226 -0
  103. package/test/unique.test.ts +71 -0
  104. package/test/url.test.ts +136 -0
  105. package/test/version.test.ts +52 -0
  106. package/tsconfig.json +31 -0
  107. package/vite.config.mts +114 -0
@@ -0,0 +1,206 @@
1
+ import { dateFormat } from '@/date';
2
+ import { TzDate } from '@/date';
3
+
4
+ const gmtOrder = TzDate.getOrder();
5
+ console.log(`当前时区 GMT${gmtOrder > 0 ? '+' : ''}${gmtOrder}`);
6
+
7
+ describe('0 时区', () => {
8
+ const offset = TzDate.getOffset(0);
9
+
10
+ it('空入参', () => {
11
+ // 时间戳与时区无关
12
+ const tzNow = new TzDate({ offset }).getTime();
13
+ const now = new Date().getTime();
14
+ expect(tzNow).toBeLessThanOrEqual(now);
15
+ // 误差小于 10ms
16
+ expect(tzNow).toBeGreaterThanOrEqual(now - 10);
17
+ });
18
+
19
+ it('时间戳', () => {
20
+ // 时间戳与时区无关
21
+ const now = Date.now();
22
+ const td = new TzDate({ offset, timestamp: now });
23
+ const dt = new Date(now);
24
+
25
+ expect(td.getTime()).toBe(dt.getTime());
26
+ expect(td.getTimezoneOrder()).toBe(0);
27
+ });
28
+
29
+ it('年月日', () => {
30
+ const localOffset = new Date().getTimezoneOffset() * 60 * 1000;
31
+ const targetOffset = 0;
32
+ const value = [2025, 5, 3, 12, 34, 56, 789] as const;
33
+ const td = new TzDate({ offset, value });
34
+
35
+ expect(td.toISOString()).toEqual('2025-06-03T12:34:56.789Z');
36
+
37
+ // 0 时区与 UTC 时间戳一致
38
+ expect(td.getTime()).toBe(Date.UTC(...value) + targetOffset);
39
+ expect(td.getTime()).toBe(new Date(...value).getTime() - localOffset + targetOffset);
40
+
41
+ expect([
42
+ td.getFullYear(),
43
+ td.getMonth(),
44
+ td.getDate(),
45
+ td.getHours(),
46
+ td.getMinutes(),
47
+ td.getSeconds(),
48
+ td.getMilliseconds(),
49
+ ]).toEqual(value);
50
+
51
+ td.setFullYear(2024);
52
+ expect(td.getFullYear()).toBe(2024);
53
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
54
+
55
+ td.setMonth(5);
56
+ expect(td.getMonth()).toBe(5);
57
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
58
+
59
+ td.setDate(10);
60
+ expect(td.getDate()).toBe(10);
61
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
62
+
63
+ td.setHours(12);
64
+ expect(td.getHours()).toBe(12);
65
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
66
+
67
+ td.setMinutes(30);
68
+ expect(td.getMinutes()).toBe(30);
69
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
70
+
71
+ td.setSeconds(45);
72
+ expect(td.getSeconds()).toBe(45);
73
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
74
+
75
+ td.setMilliseconds(100);
76
+ expect(td.getMilliseconds()).toBe(100);
77
+ expect(td.getTime()).toEqual(getUtcTimestamp(td));
78
+ });
79
+ });
80
+
81
+ describe('东 8 时区', () => {
82
+ const offset = TzDate.getOffset(8);
83
+
84
+ it('空入参', () => {
85
+ // 时间戳与时区无关
86
+ const tzNow = new TzDate({ offset }).getTime();
87
+ const now = new Date().getTime();
88
+ expect(tzNow).toBeLessThanOrEqual(now);
89
+ // 误差小于 10ms
90
+ expect(tzNow).toBeGreaterThanOrEqual(now - 10);
91
+ });
92
+
93
+ it('时间戳', () => {
94
+ // 时间戳与时区无关
95
+ const now = Date.now();
96
+ const td = new TzDate({ offset, timestamp: now });
97
+ const dt = new Date(now);
98
+
99
+ expect(td.getTime()).toBe(dt.getTime());
100
+ expect(td.getTimezoneOrder()).toBe(8);
101
+ });
102
+
103
+ it('年月日', () => {
104
+ const localOffset = new Date().getTimezoneOffset() * 60 * 1000;
105
+ const targetOffset = offset * 60 * 1000;
106
+ const value = [2025, 5, 3, 12, 34, 56, 789] as const;
107
+ const td = new TzDate({ offset, value: [...value] });
108
+
109
+ expect(td.toISOString()).toEqual('2025-06-03T04:34:56.789Z');
110
+
111
+ // 0 时区与 UTC 时间戳一致
112
+ expect(td.getTime()).toBe(Date.UTC(...value) + targetOffset);
113
+ expect(td.getTime()).toBe(new Date(...value).getTime() - localOffset + targetOffset);
114
+
115
+ expect([
116
+ td.getFullYear(),
117
+ td.getMonth(),
118
+ td.getDate(),
119
+ td.getHours(),
120
+ td.getMinutes(),
121
+ td.getSeconds(),
122
+ td.getMilliseconds(),
123
+ ]).toEqual(value);
124
+
125
+ td.setFullYear(2024);
126
+ expect(td.getFullYear()).toBe(2024);
127
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
128
+
129
+ td.setMonth(5);
130
+ expect(td.getMonth()).toBe(5);
131
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
132
+
133
+ td.setDate(10);
134
+ expect(td.getDate()).toBe(10);
135
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
136
+
137
+ td.setHours(12);
138
+ expect(td.getHours()).toBe(12);
139
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
140
+
141
+ td.setMinutes(30);
142
+ expect(td.getMinutes()).toBe(30);
143
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
144
+
145
+ td.setSeconds(45);
146
+ expect(td.getSeconds()).toBe(45);
147
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
148
+
149
+ td.setMilliseconds(100);
150
+ expect(td.getMilliseconds()).toBe(100);
151
+ expect(td.getTime()).toEqual(getUtcTimestamp(td, offset));
152
+ });
153
+ });
154
+
155
+ describe('时区转换', () => {
156
+ it('时间戳', () => {
157
+ const timestamp = Date.now();
158
+ const gmt8Date = new TzDate({
159
+ offset: TzDate.getOffset(8),
160
+ timestamp,
161
+ });
162
+ const gmt0Date = TzDate.from(gmt8Date, 0);
163
+
164
+ console.log('gmt8:', dateFormat(gmt8Date));
165
+ console.log('gmt0:', dateFormat(gmt0Date));
166
+
167
+ let gmt8Hours = gmt8Date.getHours();
168
+ const gmt0Hours = gmt0Date.getHours();
169
+
170
+ // 不是同一天
171
+ if (gmt8Date.getDate() !== gmt0Date.getDate()) {
172
+ gmt8Hours += 24;
173
+ }
174
+
175
+ expect(gmt8Hours - gmt0Hours).toBe(8);
176
+ expect(gmt0Date.getTime()).toBe(gmt0Date.getTime());
177
+ });
178
+
179
+ it('日期', () => {
180
+ const value = [2024, 5, 10, 12, 30, 45, 100] as const;
181
+ const gmt8Td = new TzDate({
182
+ offset: TzDate.getOffset(8),
183
+ value,
184
+ });
185
+ const gmt0Td = TzDate.from(gmt8Td, 0);
186
+
187
+ expect(gmt8Td.getHours()).toBe(12);
188
+ expect(gmt0Td.getHours()).toBe(4);
189
+ expect(gmt0Td.getTime()).toBe(gmt0Td.getTime());
190
+ });
191
+ });
192
+
193
+ function getUtcTimestamp(td: TzDate, offset = 0) {
194
+ return (
195
+ Date.UTC(
196
+ td.getFullYear(),
197
+ td.getMonth(),
198
+ td.getDate(),
199
+ td.getHours(),
200
+ td.getMinutes(),
201
+ td.getSeconds(),
202
+ td.getMilliseconds(),
203
+ ) +
204
+ offset * 60 * 1000
205
+ );
206
+ }
@@ -0,0 +1,353 @@
1
+ import {
2
+ DATE_DAY_MS,
3
+ DATE_HOUR_MS,
4
+ DATE_MINUTE_MS,
5
+ DATE_SECOND_MS,
6
+ type TDateRelativeTemplates,
7
+ dateDaysInMonth,
8
+ dateDaysInYear,
9
+ dateEndInDay,
10
+ dateEndInHour,
11
+ dateEndInMinute,
12
+ dateEndInMonth,
13
+ dateEndInSecond,
14
+ dateEndInYear,
15
+ dateFormat,
16
+ dateParse,
17
+ dateRelative,
18
+ dateStartInDay,
19
+ dateStartInHour,
20
+ dateStartInMinute,
21
+ dateStartInMonth,
22
+ dateStartInSecond,
23
+ dateStartInYear,
24
+ isLeapYear,
25
+ isSameDateInDay,
26
+ isSameDateInHour,
27
+ isSameDateInMinute,
28
+ isSameDateInMonth,
29
+ isSameDateInSecond,
30
+ isSameDateInYear,
31
+ isValidDate,
32
+ } from '@/date';
33
+ import { describe, expect, it } from 'vitest';
34
+
35
+ describe('isValidDate', () => {
36
+ it('应正确判断有效的日期对象', () => {
37
+ expect(isValidDate(new Date())).toBe(true);
38
+ expect(isValidDate(new Date('2023-01-01'))).toBe(true);
39
+ });
40
+
41
+ it('应正确判断无效的日期对象', () => {
42
+ expect(isValidDate('2023-01-01')).toBe(false);
43
+ expect(isValidDate(Number.NaN)).toBe(false);
44
+ expect(isValidDate(123456789)).toBe(false);
45
+ });
46
+ });
47
+
48
+ describe('dateParse', () => {
49
+ it('应正确解析数值为日期对象', () => {
50
+ const date = dateParse(1672531200000);
51
+ expect(date.getFullYear()).toBe(2023);
52
+ expect(date.getMonth()).toBe(0);
53
+ expect(date.getDate()).toBe(1);
54
+ });
55
+
56
+ it('应正确解析字符串为日期对象', () => {
57
+ const date = dateParse('2023-01-01');
58
+ expect(date.getFullYear()).toBe(2023);
59
+ expect(date.getMonth()).toBe(0);
60
+ expect(date.getDate()).toBe(1);
61
+ });
62
+
63
+ it('应正确解析 Date 对象', () => {
64
+ const inputDate = new Date('2023-01-01');
65
+ const date = dateParse(inputDate);
66
+ expect(date.getTime()).toBe(inputDate.getTime());
67
+ });
68
+
69
+ it('应抛出 SyntaxError 当解析无效的日期字符串', () => {
70
+ expect(() => dateParse('invalid date')).toThrow(SyntaxError);
71
+ });
72
+ });
73
+
74
+ describe('dateStringify', () => {
75
+ it('应正确格式化日期为默认模板', () => {
76
+ const date = new Date('2023-01-01T00:00:00');
77
+ expect(dateFormat(date)).toBe('2023-01-01 00:00:00');
78
+ });
79
+
80
+ it('12/24 小时制', () => {
81
+ const date = new Date('2023-01-01T15:22:33');
82
+ expect(dateFormat(date, 'hh/HH')).toBe('03/15');
83
+ });
84
+
85
+ it('应正确格式化日期为自定义模板', () => {
86
+ const date = new Date('2023-01-01T12:34:56');
87
+ expect(dateFormat(date, 'YYYY/MM/DD HH:mm:ss')).toBe('2023/01/01 12:34:56');
88
+ expect(dateFormat(date, 'YYYY年MM月DD日')).toBe('2023年01月01日');
89
+ });
90
+
91
+ it('应正确处理数值和字符串作为日期值', () => {
92
+ expect(dateFormat(1672531200000, 'YYYY-MM-DD')).toBe('2023-01-01');
93
+ expect(dateFormat('2023-01-01', 'YYYY/MM/DD')).toBe('2023/01/01');
94
+ });
95
+ });
96
+
97
+ describe('dateRelative', () => {
98
+ const now = new Date(1999, 9, 9, 9, 9, 9, 9).getTime();
99
+
100
+ it('前时间', () => {
101
+ expect(dateRelative(now, now)).toEqual('刚刚');
102
+ expect(dateRelative(now - 1000 * 9, now)).toEqual('刚刚');
103
+ expect(dateRelative(now - 1000 * 10, now)).toEqual('10 秒前');
104
+ expect(dateRelative(now - 1000 * 59, now)).toEqual('59 秒前');
105
+ expect(dateRelative(now - 1000 * 59 - 999, now)).toEqual('59 秒前');
106
+ expect(dateRelative(now - 1000 * 60, now)).toEqual('1 分钟前');
107
+ expect(dateRelative(now - 1000 * 60 * 59, now)).toEqual('59 分钟前');
108
+ expect(dateRelative(now - 1000 * 60 * 60, now)).toEqual('1 小时前');
109
+ expect(dateRelative(now - 1000 * 60 * 60 * 23, now)).toEqual('23 小时前');
110
+ expect(dateRelative(now - 1000 * 60 * 60 * 24, now)).toEqual('昨天');
111
+ expect(dateRelative(now - 1000 * 60 * 60 * 24 * 2, now)).toEqual('前天');
112
+ expect(dateRelative(now - 1000 * 60 * 60 * 24 * 3, now)).toEqual('3 天前');
113
+ expect(dateRelative(now - 1000 * 60 * 60 * 24 * 29, now)).toEqual('29 天前');
114
+ expect(dateRelative(now - 1000 * 60 * 60 * 24 * 30, now)).toEqual('1999年09月09日');
115
+ });
116
+
117
+ it('后时间', () => {
118
+ expect(dateRelative(now, now)).toEqual('刚刚');
119
+ expect(dateRelative(now + 1000 * 9, now)).toEqual('刚刚');
120
+ expect(dateRelative(now + 1000 * 10, now)).toEqual('10 秒后');
121
+ expect(dateRelative(now + 1000 * 59, now)).toEqual('59 秒后');
122
+ expect(dateRelative(now + 1000 * 60, now)).toEqual('1 分钟后');
123
+ expect(dateRelative(now + 1000 * 60 * 59, now)).toEqual('59 分钟后');
124
+ expect(dateRelative(now + 1000 * 60 * 60, now)).toEqual('1 小时后');
125
+ expect(dateRelative(now + 1000 * 60 * 60 * 23, now)).toEqual('23 小时后');
126
+ expect(dateRelative(now + 1000 * 60 * 60 * 24, now)).toEqual('明天');
127
+ expect(dateRelative(now + 1000 * 60 * 60 * 24 * 2, now)).toEqual('后天');
128
+ expect(dateRelative(now + 1000 * 60 * 60 * 24 * 3, now)).toEqual('3 天后');
129
+ expect(dateRelative(now + 1000 * 60 * 60 * 24 * 29, now)).toEqual('29 天后');
130
+ expect(dateRelative(now + 1000 * 60 * 60 * 24 * 30, now)).toEqual('1999年11月08日');
131
+ });
132
+
133
+ it('自定义模板', () => {
134
+ const myTemplates: TDateRelativeTemplates = [
135
+ [1, 100, 'in {n} seconds'],
136
+ [0, Number.POSITIVE_INFINITY, 'YYYY-MM-DD HH:mm:ss'],
137
+ ];
138
+ expect(dateRelative(now - 1000 * 59, now, myTemplates)).toEqual('in 59 seconds');
139
+ expect(dateRelative(now - 1000 * 99, now, myTemplates)).toEqual('in 99 seconds');
140
+ expect(dateRelative(now - 1000 * 100, now, myTemplates)).toEqual('1999-10-09 09:07:29');
141
+ expect(dateRelative(new Date(), myTemplates)).toEqual('in 1 seconds');
142
+ });
143
+ });
144
+
145
+ describe('dateOfStart', () => {
146
+ it('应正确返回指定时间单位的起始时间', () => {
147
+ const date = new Date(2023, 5, 15, 12, 30, 45, 500); // 2023-06-15 12:30:45.500
148
+
149
+ // 测试秒级起始时间
150
+ const secondStart = dateStartInSecond(date);
151
+ expect(secondStart.getMilliseconds()).toBe(0);
152
+ expect(secondStart.getSeconds()).toBe(45);
153
+ expect(secondStart.getMinutes()).toBe(30);
154
+ expect(secondStart.getHours()).toBe(12);
155
+ expect(secondStart.getDate()).toBe(15);
156
+ expect(secondStart.getMonth()).toBe(5);
157
+ expect(secondStart.getFullYear()).toBe(2023);
158
+
159
+ // 测试分钟级起始时间
160
+ const minuteStart = dateStartInMinute(date);
161
+ expect(minuteStart.getMilliseconds()).toBe(0);
162
+ expect(minuteStart.getSeconds()).toBe(0);
163
+ expect(minuteStart.getMinutes()).toBe(30);
164
+ expect(minuteStart.getHours()).toBe(12);
165
+ expect(minuteStart.getDate()).toBe(15);
166
+ expect(minuteStart.getMonth()).toBe(5);
167
+ expect(minuteStart.getFullYear()).toBe(2023);
168
+
169
+ // 测试小时级起始时间
170
+ const hourStart = dateStartInHour(date);
171
+ expect(hourStart.getMilliseconds()).toBe(0);
172
+ expect(hourStart.getSeconds()).toBe(0);
173
+ expect(hourStart.getMinutes()).toBe(0);
174
+ expect(hourStart.getHours()).toBe(12);
175
+ expect(hourStart.getDate()).toBe(15);
176
+ expect(hourStart.getMonth()).toBe(5);
177
+ expect(hourStart.getFullYear()).toBe(2023);
178
+
179
+ // 测试天级起始时间
180
+ const dayStart = dateStartInDay(date);
181
+ expect(dayStart.getMilliseconds()).toBe(0);
182
+ expect(dayStart.getSeconds()).toBe(0);
183
+ expect(dayStart.getMinutes()).toBe(0);
184
+ expect(dayStart.getHours()).toBe(0);
185
+ expect(dayStart.getDate()).toBe(15);
186
+ expect(dayStart.getMonth()).toBe(5);
187
+ expect(dayStart.getFullYear()).toBe(2023);
188
+
189
+ // 测试月级起始时间
190
+ const monthStart = dateStartInMonth(date);
191
+ expect(monthStart.getMilliseconds()).toBe(0);
192
+ expect(monthStart.getSeconds()).toBe(0);
193
+ expect(monthStart.getMinutes()).toBe(0);
194
+ expect(monthStart.getHours()).toBe(0);
195
+ expect(monthStart.getDate()).toBe(1);
196
+ expect(monthStart.getMonth()).toBe(5);
197
+ expect(monthStart.getFullYear()).toBe(2023);
198
+
199
+ // 测试年级起始时间
200
+ const yearStart = dateStartInYear(date);
201
+ expect(yearStart.getMilliseconds()).toBe(0);
202
+ expect(yearStart.getSeconds()).toBe(0);
203
+ expect(yearStart.getMinutes()).toBe(0);
204
+ expect(yearStart.getHours()).toBe(0);
205
+ expect(yearStart.getDate()).toBe(1);
206
+ expect(yearStart.getMonth()).toBe(0);
207
+ expect(yearStart.getFullYear()).toBe(2023);
208
+ });
209
+ });
210
+
211
+ describe('dateOfEnd', () => {
212
+ it('应正确返回指定时间单位的结束时间', () => {
213
+ const date = new Date(2023, 1, 15, 12, 30, 45, 500); // 2023-02-15 12:30:45.500
214
+
215
+ // 测试秒级结束时间
216
+ const secondEnd = dateEndInSecond(date);
217
+ expect(secondEnd.getMilliseconds()).toBe(999);
218
+ expect(secondEnd.getSeconds()).toBe(45);
219
+ expect(secondEnd.getMinutes()).toBe(30);
220
+ expect(secondEnd.getHours()).toBe(12);
221
+ expect(secondEnd.getDate()).toBe(15);
222
+ expect(secondEnd.getMonth()).toBe(1);
223
+ expect(secondEnd.getFullYear()).toBe(2023);
224
+
225
+ // 测试分钟级结束时间
226
+ const minuteEnd = dateEndInMinute(date);
227
+ expect(minuteEnd.getMilliseconds()).toBe(999);
228
+ expect(minuteEnd.getSeconds()).toBe(59);
229
+ expect(minuteEnd.getMinutes()).toBe(30);
230
+ expect(minuteEnd.getHours()).toBe(12);
231
+ expect(minuteEnd.getDate()).toBe(15);
232
+ expect(minuteEnd.getMonth()).toBe(1);
233
+ expect(minuteEnd.getFullYear()).toBe(2023);
234
+
235
+ // 测试小时级结束时间
236
+ const hourEnd = dateEndInHour(date);
237
+ expect(hourEnd.getMilliseconds()).toBe(999);
238
+ expect(hourEnd.getSeconds()).toBe(59);
239
+ expect(hourEnd.getMinutes()).toBe(59);
240
+ expect(hourEnd.getHours()).toBe(12);
241
+ expect(hourEnd.getDate()).toBe(15);
242
+ expect(hourEnd.getMonth()).toBe(1);
243
+ expect(hourEnd.getFullYear()).toBe(2023);
244
+
245
+ // 测试天级结束时间
246
+ const dayEnd = dateEndInDay(date);
247
+ expect(dayEnd.getMilliseconds()).toBe(999);
248
+ expect(dayEnd.getSeconds()).toBe(59);
249
+ expect(dayEnd.getMinutes()).toBe(59);
250
+ expect(dayEnd.getHours()).toBe(23);
251
+ expect(dayEnd.getDate()).toBe(15);
252
+ expect(dayEnd.getMonth()).toBe(1);
253
+ expect(dayEnd.getFullYear()).toBe(2023);
254
+
255
+ // 测试月级结束时间
256
+ const monthEnd = dateEndInMonth(date);
257
+ expect(monthEnd.getMilliseconds()).toBe(999);
258
+ expect(monthEnd.getSeconds()).toBe(59);
259
+ expect(monthEnd.getMinutes()).toBe(59);
260
+ expect(monthEnd.getHours()).toBe(23);
261
+ expect(monthEnd.getDate()).toBe(28);
262
+ expect(monthEnd.getMonth()).toBe(1);
263
+ expect(monthEnd.getFullYear()).toBe(2023);
264
+
265
+ // 测试年级结束时间
266
+ const yearEnd = dateEndInYear(date);
267
+ expect(yearEnd.getMilliseconds()).toBe(999);
268
+ expect(yearEnd.getSeconds()).toBe(59);
269
+ expect(yearEnd.getMinutes()).toBe(59);
270
+ expect(yearEnd.getHours()).toBe(23);
271
+ expect(yearEnd.getDate()).toBe(31);
272
+ expect(yearEnd.getMonth()).toBe(11);
273
+ expect(yearEnd.getFullYear()).toBe(2023);
274
+ });
275
+ });
276
+
277
+ describe('dateDaysInMonth', () => {
278
+ it('应正确计算指定日期所在月的天数', () => {
279
+ expect(dateDaysInMonth(new Date('2023-02-15'))).toBe(28); // 非闰年2月
280
+ expect(dateDaysInMonth(new Date('2024-02-15'))).toBe(29); // 闰年2月
281
+ expect(dateDaysInMonth(new Date('2023-04-15'))).toBe(30); // 4月
282
+ expect(dateDaysInMonth(new Date('2023-07-15'))).toBe(31); // 7月
283
+ });
284
+
285
+ it('默认应计算指定日期所在月的天数', () => {
286
+ expect(dateDaysInMonth(new Date('2023-02-15'))).toBe(28); // 默认计算月天数
287
+ });
288
+ });
289
+
290
+ describe('dateDaysInYear', () => {
291
+ it('应正确计算指定日期所在年的天数', () => {
292
+ expect(dateDaysInYear(new Date('2023-02-15'))).toBe(365); // 非闰年
293
+ expect(dateDaysInYear(new Date('2024-02-15'))).toBe(366); // 闰年
294
+ });
295
+ });
296
+
297
+ describe('isSameDate', () => {
298
+ const date1 = new Date(2023, 5, 15, 12, 30, 45, 500); // 2023-06-15 12:30:45.500
299
+ const date2 = new Date(2023, 5, 15, 13, 30, 45, 500); // 2023-06-15 13:30:45.500
300
+ const date3 = new Date(2023, 5, 16, 12, 30, 45, 500); // 2023-06-16 12:30:45.500
301
+ const date4 = new Date(2023, 6, 15, 12, 30, 45, 500); // 2023-07-15 12:30:45.500
302
+ const date5 = new Date(2024, 5, 15, 12, 30, 45, 500); // 2024-06-15 12:30:45.500
303
+
304
+ it('应正确比较年份', () => {
305
+ expect(isSameDateInYear(date1, date2)).toBe(true);
306
+ expect(isSameDateInYear(date1, date5)).toBe(false);
307
+ });
308
+
309
+ it('应正确比较月份', () => {
310
+ expect(isSameDateInMonth(date1, date2)).toBe(true);
311
+ expect(isSameDateInMonth(date1, date4)).toBe(false);
312
+ });
313
+
314
+ it('应正确比较天数', () => {
315
+ expect(isSameDateInDay(date1, date2)).toBe(true);
316
+ expect(isSameDateInDay(date1, date3)).toBe(false);
317
+ });
318
+
319
+ it('应正确比较小时', () => {
320
+ expect(isSameDateInHour(date1, date2)).toBe(false);
321
+ expect(isSameDateInHour(date1, new Date(2023, 5, 15, 12, 0, 0, 0))).toBe(true);
322
+ });
323
+
324
+ it('应正确比较分钟', () => {
325
+ expect(isSameDateInMinute(date1, date2)).toBe(false);
326
+ expect(isSameDateInMinute(date1, new Date(2023, 5, 15, 12, 30, 0, 0))).toBe(true);
327
+ });
328
+
329
+ it('应正确比较秒', () => {
330
+ expect(isSameDateInSecond(date1, date2)).toBe(false);
331
+ expect(isSameDateInSecond(date1, '2023-06-15 12:30:45.0')).toBe(true);
332
+ });
333
+
334
+ it('应正确处理字符串和数值作为日期值', () => {
335
+ expect(isSameDateInDay('2023-06-15', '2023-06-15')).toBe(true);
336
+ expect(isSameDateInDay(1686814245500, 1686814245511)).toBe(true);
337
+ expect(isSameDateInDay('2023-06-15', '2023-06-16')).toBe(false);
338
+ });
339
+ });
340
+
341
+ describe('isLeapYear', () => {
342
+ it('应正确判断闰年', () => {
343
+ expect(isLeapYear(2020)).toBe(true);
344
+ expect(isLeapYear(2000)).toBe(true);
345
+ expect(isLeapYear(2024)).toBe(true);
346
+ });
347
+
348
+ it('应正确判断平年', () => {
349
+ expect(isLeapYear(2021)).toBe(false);
350
+ expect(isLeapYear(1900)).toBe(false);
351
+ expect(isLeapYear(1999)).toBe(false);
352
+ });
353
+ });
@@ -0,0 +1,33 @@
1
+ import { createEasingFn } from '@/easing';
2
+ import { describe, expect, it } from 'vitest';
3
+
4
+ describe('createEasingFn', () => {
5
+ it('应正确创建线性缓动函数', () => {
6
+ const linearEasing = createEasingFn(0, 0, 1, 1);
7
+ expect(linearEasing(0)).toBe(0);
8
+ expect(linearEasing(0.5)).toBe(0.5);
9
+ expect(linearEasing(1)).toBe(1);
10
+ });
11
+
12
+ it('应正确创建自定义缓动函数', () => {
13
+ const customEasing = createEasingFn(0.25, 0.1, 0.25, 1);
14
+ expect(customEasing(0)).toBe(0);
15
+ expect(customEasing(0.5)).toBeCloseTo(0.8, 1);
16
+ expect(customEasing(1)).toBe(1);
17
+ });
18
+
19
+ it('应抛出错误当 mX1 或 mX2 不在 [0, 1] 范围内', () => {
20
+ expect(() => createEasingFn(-0.1, 0.5, 0.5, 1)).toThrow('bezier x values must be in [0, 1] range');
21
+ expect(() => createEasingFn(1.1, 0.5, 0.5, 1)).toThrow('bezier x values must be in [0, 1] range');
22
+ expect(() => createEasingFn(0.5, 0.5, -0.1, 1)).toThrow('bezier x values must be in [0, 1] range');
23
+ expect(() => createEasingFn(0.5, 0.5, 1.1, 1)).toThrow('bezier x values must be in [0, 1] range');
24
+ });
25
+
26
+ it('应返回 LinearEasing 当 mX1=mY1 且 mX2=mY2', () => {
27
+ const linearEasing = createEasingFn(0, 0, 1, 1);
28
+ const customLinearEasing = createEasingFn(0.5, 0.5, 0.5, 0.5);
29
+ expect(customLinearEasing(0)).toBe(linearEasing(0));
30
+ expect(customLinearEasing(0.5)).toBe(linearEasing(0.5));
31
+ expect(customLinearEasing(1)).toBe(linearEasing(1));
32
+ });
33
+ });
@@ -0,0 +1,71 @@
1
+ import { Emitter } from '@/emitter';
2
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
3
+
4
+ type TestEvents = {
5
+ click: [x: number, y: number];
6
+ change: [value: string];
7
+ };
8
+
9
+ describe('Emitter', () => {
10
+ let emitter: Emitter<TestEvents>;
11
+
12
+ beforeEach(() => {
13
+ emitter = new Emitter<TestEvents>();
14
+ });
15
+
16
+ it('应正确注册和触发事件', () => {
17
+ const clickHandler = vi.fn();
18
+ emitter.on('click', clickHandler);
19
+ emitter.emit('click', 10, 20);
20
+ expect(clickHandler).toHaveBeenCalledWith(10, 20);
21
+ });
22
+
23
+ it('应按注册顺序触发监听器', () => {
24
+ const calls: number[] = [];
25
+ emitter.on('click', () => calls.push(1));
26
+ emitter.on('click', () => calls.push(2));
27
+ emitter.emit('click', 0, 0);
28
+ expect(calls).toEqual([1, 2]);
29
+ });
30
+
31
+ it('应在监听器返回 false 时中断触发', () => {
32
+ const handler1 = vi.fn().mockReturnValue(false);
33
+ const handler2 = vi.fn();
34
+ emitter.on('click', handler1);
35
+ emitter.on('click', handler2);
36
+ emitter.emit('click', 0, 0);
37
+ expect(handler1).toHaveBeenCalled();
38
+ expect(handler2).not.toHaveBeenCalled();
39
+ });
40
+
41
+ it('应移除特定事件的特定监听器', () => {
42
+ const handler = vi.fn();
43
+ emitter.on('click', handler);
44
+ emitter.off('click', handler);
45
+ emitter.emit('click', 0, 0);
46
+ expect(handler).not.toHaveBeenCalled();
47
+ });
48
+
49
+ it('应移除特定事件的所有监听器', () => {
50
+ const handler1 = vi.fn();
51
+ const handler2 = vi.fn();
52
+ emitter.on('click', handler1);
53
+ emitter.on('click', handler2);
54
+ emitter.off('click');
55
+ emitter.emit('click', 0, 0);
56
+ expect(handler1).not.toHaveBeenCalled();
57
+ expect(handler2).not.toHaveBeenCalled();
58
+ });
59
+
60
+ it('应移除所有事件的所有监听器', () => {
61
+ const clickHandler = vi.fn();
62
+ const changeHandler = vi.fn();
63
+ emitter.on('click', clickHandler);
64
+ emitter.on('change', changeHandler);
65
+ emitter.off();
66
+ emitter.emit('click', 0, 0);
67
+ emitter.emit('change', 'test');
68
+ expect(clickHandler).not.toHaveBeenCalled();
69
+ expect(changeHandler).not.toHaveBeenCalled();
70
+ });
71
+ });