@simplysm/core-common 13.0.0-beta.2 → 13.0.0-beta.21

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 (84) hide show
  1. package/dist/common.types.js +4 -4
  2. package/dist/errors/argument-error.js +1 -1
  3. package/dist/errors/not-implemented-error.js +1 -1
  4. package/dist/errors/timeout-error.js +1 -1
  5. package/dist/extensions/arr-ext.helpers.js +4 -4
  6. package/dist/extensions/arr-ext.js +9 -9
  7. package/dist/features/debounce-queue.js +2 -2
  8. package/dist/features/serial-queue.js +3 -3
  9. package/dist/index.js +30 -30
  10. package/dist/types/date-only.js +2 -2
  11. package/dist/types/date-time.js +2 -2
  12. package/dist/types/time.js +2 -2
  13. package/dist/types/uuid.js +1 -1
  14. package/dist/utils/bytes.js +1 -1
  15. package/dist/utils/json.js +8 -8
  16. package/dist/utils/obj.js +5 -5
  17. package/dist/utils/primitive.js +5 -5
  18. package/dist/utils/transferable.js +4 -4
  19. package/dist/utils/wait.js +1 -1
  20. package/package.json +7 -4
  21. package/.cache/typecheck-browser.tsbuildinfo +0 -1
  22. package/.cache/typecheck-node.tsbuildinfo +0 -1
  23. package/.cache/typecheck-tests-browser.tsbuildinfo +0 -1
  24. package/.cache/typecheck-tests-node.tsbuildinfo +0 -1
  25. package/src/common.types.ts +0 -91
  26. package/src/env.ts +0 -11
  27. package/src/errors/argument-error.ts +0 -40
  28. package/src/errors/not-implemented-error.ts +0 -32
  29. package/src/errors/sd-error.ts +0 -53
  30. package/src/errors/timeout-error.ts +0 -36
  31. package/src/extensions/arr-ext.helpers.ts +0 -53
  32. package/src/extensions/arr-ext.ts +0 -777
  33. package/src/extensions/arr-ext.types.ts +0 -258
  34. package/src/extensions/map-ext.ts +0 -86
  35. package/src/extensions/set-ext.ts +0 -68
  36. package/src/features/debounce-queue.ts +0 -116
  37. package/src/features/event-emitter.ts +0 -112
  38. package/src/features/serial-queue.ts +0 -94
  39. package/src/globals.ts +0 -12
  40. package/src/index.ts +0 -55
  41. package/src/types/date-only.ts +0 -329
  42. package/src/types/date-time.ts +0 -294
  43. package/src/types/lazy-gc-map.ts +0 -244
  44. package/src/types/time.ts +0 -210
  45. package/src/types/uuid.ts +0 -113
  46. package/src/utils/bytes.ts +0 -160
  47. package/src/utils/date-format.ts +0 -239
  48. package/src/utils/json.ts +0 -230
  49. package/src/utils/num.ts +0 -97
  50. package/src/utils/obj.ts +0 -956
  51. package/src/utils/path.ts +0 -40
  52. package/src/utils/primitive.ts +0 -33
  53. package/src/utils/str.ts +0 -252
  54. package/src/utils/template-strings.ts +0 -132
  55. package/src/utils/transferable.ts +0 -269
  56. package/src/utils/wait.ts +0 -40
  57. package/src/utils/xml.ts +0 -105
  58. package/src/zip/sd-zip.ts +0 -218
  59. package/tests/errors/errors.spec.ts +0 -196
  60. package/tests/extensions/array-extension.spec.ts +0 -790
  61. package/tests/extensions/map-extension.spec.ts +0 -147
  62. package/tests/extensions/set-extension.spec.ts +0 -74
  63. package/tests/types/date-only.spec.ts +0 -636
  64. package/tests/types/date-time.spec.ts +0 -391
  65. package/tests/types/lazy-gc-map.spec.ts +0 -692
  66. package/tests/types/time.spec.ts +0 -559
  67. package/tests/types/types.spec.ts +0 -55
  68. package/tests/types/uuid.spec.ts +0 -91
  69. package/tests/utils/bytes-utils.spec.ts +0 -230
  70. package/tests/utils/date-format.spec.ts +0 -371
  71. package/tests/utils/debounce-queue.spec.ts +0 -272
  72. package/tests/utils/json.spec.ts +0 -475
  73. package/tests/utils/number.spec.ts +0 -184
  74. package/tests/utils/object.spec.ts +0 -827
  75. package/tests/utils/path.spec.ts +0 -78
  76. package/tests/utils/primitive.spec.ts +0 -55
  77. package/tests/utils/sd-event-emitter.spec.ts +0 -216
  78. package/tests/utils/serial-queue.spec.ts +0 -365
  79. package/tests/utils/string.spec.ts +0 -294
  80. package/tests/utils/template-strings.spec.ts +0 -96
  81. package/tests/utils/transferable.spec.ts +0 -698
  82. package/tests/utils/wait.spec.ts +0 -145
  83. package/tests/utils/xml.spec.ts +0 -146
  84. package/tests/zip/sd-zip.spec.ts +0 -234
@@ -1,258 +0,0 @@
1
- /**
2
- * Array 확장 타입 정의
3
- */
4
-
5
- import type { PrimitiveTypeMap, PrimitiveTypeStr, Type } from "../common.types";
6
- import type { DateTime } from "../types/date-time";
7
- import type { DateOnly } from "../types/date-only";
8
- import type { Time } from "../types/time";
9
-
10
- //#region 인터페이스
11
-
12
- export interface ReadonlyArrayExt<T> {
13
- /**
14
- * 조건에 맞는 단일 요소 반환
15
- * @param predicate 필터 조건 (생략 시 배열 전체 대상)
16
- * @returns 요소가 없으면 undefined
17
- * @throws ArgumentError 조건에 맞는 요소가 2개 이상이면 발생
18
- */
19
- single(predicate?: (item: T, index: number) => boolean): T | undefined;
20
-
21
- /**
22
- * 첫 번째 요소 반환
23
- * @param predicate 필터 조건 (생략 시 첫 번째 요소 반환)
24
- * @returns 요소가 없으면 undefined
25
- */
26
- first(predicate?: (item: T, index: number) => boolean): T | undefined;
27
-
28
- /** 비동기 필터 (순차 실행) */
29
- filterAsync(predicate: (item: T, index: number) => Promise<boolean>): Promise<T[]>;
30
-
31
- /**
32
- * 마지막 요소 반환
33
- * @param predicate 필터 조건 (생략 시 마지막 요소 반환)
34
- * @returns 요소가 없으면 undefined
35
- */
36
- last(predicate?: (item: T, index: number) => boolean): T | undefined;
37
-
38
- /** null/undefined 제거 */
39
- filterExists(): NonNullable<T>[];
40
-
41
- /** 특정 타입의 요소만 필터링 (PrimitiveTypeStr 또는 생성자 타입) */
42
- ofType<K extends PrimitiveTypeStr>(type: K): Extract<T, PrimitiveTypeMap[K]>[];
43
- ofType<N extends T>(type: Type<N>): N[];
44
-
45
- /** 비동기 매핑 (순차 실행) */
46
- mapAsync<R>(selector: (item: T, index: number) => Promise<R>): Promise<R[]>;
47
-
48
- /** 중첩 배열 평탄화 */
49
- mapMany(): T extends readonly (infer U)[] ? U[] : T;
50
-
51
- /** 매핑 후 평탄화 */
52
- mapMany<R>(selector: (item: T, index: number) => R[]): R[];
53
-
54
- /** 비동기 매핑 후 평탄화 (순차 실행) */
55
- mapManyAsync<R>(selector: (item: T, index: number) => Promise<R[]>): Promise<R[]>;
56
-
57
- /**
58
- * 비동기 병렬 처리 (Promise.all 사용)
59
- * @note 하나라도 reject되면 전체가 fail-fast로 reject됨 (Promise.all 동작)
60
- */
61
- parallelAsync<R>(fn: (item: T, index: number) => Promise<R>): Promise<R[]>;
62
-
63
- /**
64
- * 키 기준 그룹화
65
- * @param keySelector 그룹 키 선택 함수
66
- * @note O(n²) 복잡도 (객체 키 지원을 위해 깊은 비교 사용). primitive 키만 필요하면 toArrayMap()이 O(n)으로 더 효율적
67
- */
68
- groupBy<K>(keySelector: (item: T, index: number) => K): { key: K; values: T[] }[];
69
-
70
- /**
71
- * 키 기준 그룹화 (값 변환 포함)
72
- * @param keySelector 그룹 키 선택 함수
73
- * @param valueSelector 값 변환 함수
74
- * @note O(n²) 복잡도 (객체 키 지원을 위해 깊은 비교 사용). primitive 키만 필요하면 toArrayMap()이 O(n)으로 더 효율적
75
- */
76
- groupBy<K, V>(
77
- keySelector: (item: T, index: number) => K,
78
- valueSelector: (item: T, index: number) => V,
79
- ): {
80
- key: K;
81
- values: V[];
82
- }[];
83
-
84
- toMap<K>(keySelector: (item: T, index: number) => K): Map<K, T>;
85
-
86
- toMap<K, V>(keySelector: (item: T, index: number) => K, valueSelector: (item: T, index: number) => V): Map<K, V>;
87
-
88
- toMapAsync<K>(keySelector: (item: T, index: number) => Promise<K>): Promise<Map<K, T>>;
89
-
90
- toMapAsync<K, V>(
91
- keySelector: (item: T, index: number) => Promise<K> | K,
92
- valueSelector: (item: T, index: number) => Promise<V> | V,
93
- ): Promise<Map<K, V>>;
94
-
95
- toArrayMap<K>(keySelector: (item: T, index: number) => K): Map<K, T[]>;
96
-
97
- toArrayMap<K, V>(
98
- keySelector: (item: T, index: number) => K,
99
- valueSelector: (item: T, index: number) => V,
100
- ): Map<K, V[]>;
101
-
102
- toSetMap<K>(keySelector: (item: T, index: number) => K): Map<K, Set<T>>;
103
- toSetMap<K, V>(
104
- keySelector: (item: T, index: number) => K,
105
- valueSelector: (item: T, index: number) => V,
106
- ): Map<K, Set<V>>;
107
-
108
- toMapValues<K, V>(keySelector: (item: T, index: number) => K, valueSelector: (items: T[]) => V): Map<K, V>;
109
-
110
- toObject(keySelector: (item: T, index: number) => string): Record<string, T>;
111
-
112
- toObject<V>(
113
- keySelector: (item: T, index: number) => string,
114
- valueSelector: (item: T, index: number) => V,
115
- ): Record<string, V>;
116
-
117
- /**
118
- * 평탄한 배열을 트리 구조로 변환한다
119
- *
120
- * @param keyProp 각 항목의 고유 키 속성명
121
- * @param parentKey 부모 항목의 키를 참조하는 속성명
122
- * @returns 루트 항목들의 배열 (각 항목에 children 속성 추가)
123
- *
124
- * @remarks
125
- * - parentKey 값이 null/undefined인 항목이 루트가 된다
126
- * - 내부적으로 toArrayMap을 사용하여 O(n) 복잡도로 처리한다
127
- * - 원본 항목은 복사되어 children 속성이 추가된다
128
- *
129
- * @example
130
- * ```typescript
131
- * interface Item {
132
- * id: number;
133
- * parentId?: number;
134
- * name: string;
135
- * }
136
- *
137
- * const items: Item[] = [
138
- * { id: 1, name: "root" },
139
- * { id: 2, parentId: 1, name: "child1" },
140
- * { id: 3, parentId: 1, name: "child2" },
141
- * { id: 4, parentId: 2, name: "grandchild" },
142
- * ];
143
- *
144
- * const tree = items.toTree("id", "parentId");
145
- * // [{ id: 1, name: "root", children: [
146
- * // { id: 2, name: "child1", children: [
147
- * // { id: 4, name: "grandchild", children: [] }
148
- * // ]},
149
- * // { id: 3, name: "child2", children: [] }
150
- * // ]}]
151
- * ```
152
- */
153
- toTree<K extends keyof T, P extends keyof T>(keyProp: K, parentKey: P): TreeArray<T>[];
154
-
155
- /**
156
- * 중복 제거
157
- * @param options matchAddress: 주소 비교 (true면 Set 사용), keyFn: 커스텀 키 함수 (O(n) 성능)
158
- * @note 객체 배열에서 keyFn 없이 사용 시 O(n²) 복잡도. 대량 데이터는 keyFn 사용 권장
159
- */
160
- distinct(options?: boolean | { matchAddress?: boolean; keyFn?: (item: T) => string | number }): T[];
161
-
162
- orderBy(selector?: (item: T) => string | number | DateOnly | DateTime | Time | undefined): T[];
163
-
164
- orderByDesc(selector?: (item: T) => string | number | DateOnly | DateTime | Time | undefined): T[];
165
-
166
- /**
167
- * 두 배열 비교 (INSERT/DELETE/UPDATE)
168
- * @param target 비교 대상 배열
169
- * @param options keys: 키 비교용, excludes: 비교 제외 속성
170
- * @note target에 중복 키가 있으면 첫 번째 매칭만 사용됨
171
- */
172
- diffs<P>(target: P[], options?: { keys?: string[]; excludes?: string[] }): ArrayDiffsResult<T, P>[];
173
-
174
- oneWayDiffs<K extends keyof T>(
175
- orgItems: T[] | Map<T[K], T>,
176
- keyPropNameOrFn: K | ((item: T) => K),
177
- options?: {
178
- includeSame?: boolean;
179
- excludes?: string[];
180
- includes?: string[];
181
- },
182
- ): ArrayDiffs2Result<T>[];
183
-
184
- merge<P>(target: P[], options?: { keys?: string[]; excludes?: string[] }): (T | P | (T & P))[];
185
-
186
- /**
187
- * 요소의 합계 반환
188
- * @param selector 값 선택 함수 (생략 시 요소 자체를 number로 사용)
189
- * @returns 빈 배열인 경우 0 반환
190
- */
191
- sum(selector?: (item: T, index: number) => number): number;
192
-
193
- min(): T extends number | string ? T | undefined : never;
194
-
195
- min<P extends number | string>(selector?: (item: T, index: number) => P): P | undefined;
196
-
197
- max(): T extends number | string ? T | undefined : never;
198
-
199
- max<P extends number | string>(selector?: (item: T, index: number) => P): P | undefined;
200
-
201
- shuffle(): T[];
202
- }
203
-
204
- /**
205
- * 원본 배열을 변경하는 확장 메서드
206
- * @mutates 모든 메서드가 원본 배열을 직접 변경합니다
207
- */
208
- export interface MutableArrayExt<T> {
209
- /**
210
- * 원본 배열에서 중복 제거
211
- * @param options matchAddress: 주소 비교 (true면 Set 사용), keyFn: 커스텀 키 함수 (O(n) 성능)
212
- * @note 객체 배열에서 keyFn 없이 사용 시 O(n²) 복잡도. 대량 데이터는 keyFn 사용 권장
213
- * @mutates
214
- */
215
- distinctThis(options?: boolean | { matchAddress?: boolean; keyFn?: (item: T) => string | number }): T[];
216
-
217
- /** 원본 배열 오름차순 정렬 @mutates */
218
- orderByThis(selector?: (item: T) => string | number | DateOnly | DateTime | Time | undefined): T[];
219
-
220
- /** 원본 배열 내림차순 정렬 @mutates */
221
- orderByDescThis(selector?: (item: T) => string | number | DateOnly | DateTime | Time | undefined): T[];
222
-
223
- /** 원본 배열에 항목 삽입 @mutates */
224
- insert(index: number, ...items: T[]): this;
225
-
226
- /** 원본 배열에서 항목 제거 @mutates */
227
- remove(item: T): this;
228
-
229
- /** 원본 배열에서 조건에 맞는 항목 제거 @mutates */
230
- remove(selector: (item: T, index: number) => boolean): this;
231
-
232
- /** 원본 배열에서 항목 토글 (있으면 제거, 없으면 추가) @mutates */
233
- toggle(item: T): this;
234
-
235
- /** 원본 배열 비우기 @mutates */
236
- clear(): this;
237
- }
238
-
239
- //#endregion
240
-
241
- //#region 내보내기 타입
242
-
243
- export type ArrayDiffsResult<T, P> =
244
- | { source: undefined; target: P } // INSERT
245
- | { source: T; target: undefined } // DELETE
246
- | { source: T; target: P }; // UPDATE
247
-
248
- export type ArrayDiffs2Result<T> =
249
- | { type: "create"; item: T; orgItem: undefined }
250
- | { type: "update"; item: T; orgItem: T }
251
- | { type: "same"; item: T; orgItem: T };
252
-
253
- export type TreeArray<T> = T & { children: TreeArray<T>[] };
254
-
255
- /** 정렬/비교 가능한 타입 */
256
- export type ComparableType = string | number | boolean | DateTime | DateOnly | Time | undefined;
257
-
258
- //#endregion
@@ -1,86 +0,0 @@
1
- /**
2
- * Map 확장 메서드
3
- */
4
-
5
- declare global {
6
- interface Map<K, V> {
7
- /**
8
- * 키에 해당하는 값이 없으면 새 값을 설정하고 반환
9
- *
10
- * @remarks
11
- * **주의**: V 타입이 함수인 경우(예: `Map<string, () => void>`),
12
- * 두 번째 인자로 함수를 직접 전달하면 팩토리로 인식되어 호출됩니다.
13
- * 함수 자체를 값으로 저장하려면 팩토리로 감싸세요.
14
- *
15
- * @example
16
- * ```typescript
17
- * // 일반 값
18
- * map.getOrCreate("key", 0);
19
- * map.getOrCreate("key", []);
20
- *
21
- * // 팩토리 함수 (계산 비용이 큰 경우)
22
- * map.getOrCreate("key", () => expensiveComputation());
23
- *
24
- * // 함수를 값으로 저장하는 경우
25
- * const fnMap = new Map<string, () => void>();
26
- * const myFn = () => console.log("hello");
27
- * fnMap.getOrCreate("key", () => myFn); // 팩토리로 감싸기
28
- * ```
29
- */
30
- getOrCreate(key: K, newValue: V): V;
31
- getOrCreate(key: K, newValueFn: () => V): V;
32
-
33
- /**
34
- * 키에 해당하는 값을 함수로 업데이트한다
35
- *
36
- * @param key 업데이트할 키
37
- * @param updateFn 현재 값을 받아 새 값을 반환하는 함수 (키가 없으면 undefined 전달)
38
- *
39
- * @remarks
40
- * 키가 존재하지 않아도 updateFn이 호출되어 새 값이 설정된다.
41
- * 기존 값 기반 계산이 필요한 경우 (카운터 증가, 배열에 추가 등) 유용하다.
42
- *
43
- * @example
44
- * ```typescript
45
- * const countMap = new Map<string, number>();
46
- *
47
- * // 카운터 증가
48
- * countMap.update("key", (v) => (v ?? 0) + 1);
49
- *
50
- * // 배열에 항목 추가
51
- * const arrayMap = new Map<string, string[]>();
52
- * arrayMap.update("key", (v) => [...(v ?? []), "item"]);
53
- * ```
54
- */
55
- update(key: K, updateFn: (v: V | undefined) => V): void;
56
- }
57
- }
58
-
59
- Object.defineProperty(Map.prototype, "getOrCreate", {
60
- value: function <K, V>(this: Map<K, V>, key: K, newValue: V | (() => V)): V {
61
- if (!this.has(key)) {
62
- if (typeof newValue === "function") {
63
- this.set(key, (newValue as () => V)());
64
- } else {
65
- this.set(key, newValue);
66
- }
67
- }
68
- return this.get(key)!;
69
- },
70
- enumerable: false,
71
- writable: true,
72
- configurable: true,
73
- });
74
-
75
- Object.defineProperty(Map.prototype, "update", {
76
- value: function <K, V>(this: Map<K, V>, key: K, updateFn: (v: V | undefined) => V): void {
77
- const val = this.get(key);
78
- const res = updateFn(val);
79
- this.set(key, res);
80
- },
81
- enumerable: false,
82
- writable: true,
83
- configurable: true,
84
- });
85
-
86
- export {};
@@ -1,68 +0,0 @@
1
- /**
2
- * Set 확장 메서드
3
- */
4
-
5
- declare global {
6
- interface Set<T> {
7
- /**
8
- * 여러 값을 한 번에 추가
9
- */
10
- adds(...values: T[]): this;
11
-
12
- /**
13
- * 값을 토글한다 (있으면 제거, 없으면 추가)
14
- *
15
- * @param value 토글할 값
16
- * @param addOrDel 강제로 추가("add") 또는 제거("del") 지정 (생략 시 자동 토글)
17
- * @returns this (메서드 체이닝 가능)
18
- *
19
- * @remarks
20
- * addOrDel 파라미터로 조건부 추가/제거를 간결하게 표현할 수 있다.
21
- *
22
- * @example
23
- * ```typescript
24
- * const set = new Set<number>([1, 2, 3]);
25
- *
26
- * set.toggle(2); // 2가 있으므로 제거 → {1, 3}
27
- * set.toggle(4); // 4가 없으므로 추가 → {1, 3, 4}
28
- *
29
- * // 조건부 토글
30
- * const isAdmin = true;
31
- * set.toggle(5, isAdmin ? "add" : "del"); // 강제 추가
32
- * ```
33
- */
34
- toggle(value: T, addOrDel?: "add" | "del"): this;
35
- }
36
- }
37
-
38
- Object.defineProperty(Set.prototype, "adds", {
39
- value: function <T>(this: Set<T>, ...values: T[]): Set<T> {
40
- for (const val of values) {
41
- this.add(val);
42
- }
43
- return this;
44
- },
45
- enumerable: false,
46
- writable: true,
47
- configurable: true,
48
- });
49
-
50
- Object.defineProperty(Set.prototype, "toggle", {
51
- value: function <T>(this: Set<T>, value: T, addOrDel?: "add" | "del"): Set<T> {
52
- if (addOrDel === "add") {
53
- this.add(value);
54
- } else if (addOrDel === "del") {
55
- this.delete(value);
56
- } else if (this.has(value)) {
57
- this.delete(value);
58
- } else {
59
- this.add(value);
60
- }
61
- return this;
62
- },
63
- enumerable: false,
64
- writable: true,
65
- configurable: true,
66
- });
67
-
68
- export {};
@@ -1,116 +0,0 @@
1
- /**
2
- * 비동기 함수 디바운스 큐
3
- *
4
- * 짧은 시간 내에 여러 번 호출되면 마지막 요청만 실행하고 이전 요청은 무시합니다.
5
- * 입력 필드 자동완성, 연속적인 상태 변경 배치 처리 등에 유용합니다.
6
- *
7
- * @remarks
8
- * 실행 중에 추가된 요청은 디바운스 지연 없이 현재 실행이 완료된 직후 처리됩니다.
9
- * 이는 실행 완료 전에 들어온 요청이 누락되지 않도록 하기 위한 의도적 설계입니다.
10
- *
11
- * @example
12
- * const queue = new DebounceQueue(300); // 300ms 딜레이
13
- * queue.run(() => console.log("1")); // 무시됨
14
- * queue.run(() => console.log("2")); // 무시됨
15
- * queue.run(() => console.log("3")); // 300ms 후 실행됨
16
- *
17
- * @example
18
- * // 에러 처리
19
- * queue.on("error", (err) => console.error(err));
20
- */
21
- import { SdError } from "../errors/sd-error";
22
- import { EventEmitter } from "./event-emitter";
23
- import { createConsola } from "consola";
24
-
25
- interface DebounceQueueEvents {
26
- error: SdError;
27
- }
28
-
29
- export class DebounceQueue extends EventEmitter<DebounceQueueEvents> {
30
- private static readonly _logger = createConsola().withTag("DebounceQueue");
31
-
32
- private _pendingFn: (() => void | Promise<void>) | undefined;
33
- private _isRunning = false;
34
- private _isDisposed = false;
35
- private _timer: ReturnType<typeof setTimeout> | undefined;
36
-
37
- /**
38
- * @param _delay 디바운스 지연 시간 (밀리초). 생략 시 즉시 실행 (다음 이벤트 루프)
39
- */
40
- constructor(private readonly _delay?: number) {
41
- super();
42
- }
43
-
44
- /**
45
- * 대기 중인 작업과 타이머 정리
46
- */
47
- override dispose(): void {
48
- this._isDisposed = true;
49
- if (this._timer) {
50
- clearTimeout(this._timer);
51
- this._timer = undefined;
52
- }
53
- this._pendingFn = undefined;
54
- super.dispose();
55
- }
56
-
57
- /**
58
- * using 문 지원
59
- */
60
- override [Symbol.dispose](): void {
61
- this.dispose();
62
- }
63
-
64
- /**
65
- * 함수를 큐에 추가
66
- * 이전에 추가된 함수가 있으면 대체됨
67
- */
68
- run(fn: () => void | Promise<void>): void {
69
- if (this._isDisposed) return;
70
-
71
- this._pendingFn = fn;
72
-
73
- if (this._timer) {
74
- clearTimeout(this._timer);
75
- this._timer = undefined;
76
- }
77
-
78
- if (!this._isRunning) {
79
- this._timer = setTimeout(() => {
80
- void this._processLast();
81
- }, this._delay);
82
- }
83
- }
84
-
85
- private async _processLast(): Promise<void> {
86
- if (this._isDisposed || this._isRunning || !this._pendingFn) return;
87
-
88
- this._isRunning = true;
89
- this._timer = undefined;
90
-
91
- try {
92
- // 실행 중에 새 요청이 들어오면 디바운스 지연 없이 즉시 처리
93
- // 이는 "실행 완료 전에 들어온 요청은 실행 직후 바로 처리"하는 의도적 설계
94
- while (this._pendingFn) {
95
- const currentFn = this._pendingFn;
96
- this._pendingFn = undefined;
97
-
98
- try {
99
- await currentFn();
100
- } catch (err) {
101
- const error = err instanceof Error ? err : new Error(String(err));
102
- const sdError = new SdError(error, "작업 실행 중 오류 발생");
103
-
104
- // 리스너가 있으면 이벤트로 전달, 없으면 로깅
105
- if (this.listenerCount("error") > 0) {
106
- this.emit("error", sdError);
107
- } else {
108
- DebounceQueue._logger.error(sdError);
109
- }
110
- }
111
- }
112
- } finally {
113
- this._isRunning = false;
114
- }
115
- }
116
- }
@@ -1,112 +0,0 @@
1
- /**
2
- * EventTarget 래퍼 - EventEmitter와 유사한 API 제공
3
- *
4
- * 브라우저와 Node.js 모두에서 사용 가능한 타입 안전한 이벤트 에미터이다.
5
- * 내부적으로 EventTarget을 사용하여 구현되어 있다.
6
- *
7
- * @typeParam T 이벤트 타입 맵. 키는 이벤트 이름, 값은 이벤트 데이터 타입
8
- *
9
- * @example
10
- * interface MyEvents {
11
- * data: string;
12
- * error: Error;
13
- * done: void;
14
- * }
15
- *
16
- * class MyEmitter extends EventEmitter<MyEvents> {}
17
- *
18
- * const emitter = new MyEmitter();
19
- * emitter.on("data", (data) => console.log(data)); // data: string
20
- * emitter.emit("data", "hello");
21
- * emitter.emit("done"); // void 타입은 인자 없이 호출
22
- */
23
- export class EventEmitter<T extends { [K in keyof T]: unknown } = Record<string, unknown>> {
24
- private readonly _target = new EventTarget();
25
- // 이벤트 타입별로 리스너 맵 관리 (같은 리스너를 다른 이벤트에 등록 가능)
26
- // 다형적 리스너 관리를 위해 Function 타입 사용
27
- private readonly _listenerMap = new Map<string, Map<Function, (e: Event) => void>>();
28
-
29
- /**
30
- * 이벤트 리스너 등록
31
- *
32
- * @param type 이벤트 타입
33
- * @param listener 이벤트 핸들러
34
- * @note 같은 리스너를 같은 이벤트에 중복 등록하면 무시됨
35
- */
36
- on<K extends keyof T & string>(type: K, listener: (data: T[K]) => void): void {
37
- // 이벤트 타입별 맵 가져오거나 생성
38
- let typeMap = this._listenerMap.get(type);
39
- if (!typeMap) {
40
- typeMap = new Map();
41
- this._listenerMap.set(type, typeMap);
42
- }
43
-
44
- // 이미 해당 이벤트에 등록된 리스너면 무시 (중복 등록 방지)
45
- if (typeMap.has(listener)) return;
46
-
47
- const wrappedListener = (e: Event) => listener((e as CustomEvent).detail);
48
- typeMap.set(listener, wrappedListener);
49
- this._target.addEventListener(type, wrappedListener);
50
- }
51
-
52
- /**
53
- * 이벤트 리스너 제거
54
- *
55
- * @param type 이벤트 타입
56
- * @param listener 제거할 이벤트 핸들러
57
- */
58
- off<K extends keyof T & string>(type: K, listener: (data: T[K]) => void): void {
59
- const typeMap = this._listenerMap.get(type);
60
- if (!typeMap) return;
61
-
62
- const wrappedListener = typeMap.get(listener);
63
- if (wrappedListener) {
64
- this._target.removeEventListener(type, wrappedListener);
65
- typeMap.delete(listener);
66
-
67
- // 빈 맵 정리
68
- if (typeMap.size === 0) {
69
- this._listenerMap.delete(type);
70
- }
71
- }
72
- }
73
-
74
- /**
75
- * 이벤트 발생
76
- *
77
- * @param type 이벤트 타입
78
- * @param args 이벤트 데이터 (void 타입이면 생략)
79
- */
80
- emit<K extends keyof T & string>(type: K, ...args: T[K] extends void ? [] : [data: T[K]]): void {
81
- this._target.dispatchEvent(new CustomEvent(type, { detail: args[0] }));
82
- }
83
-
84
- /**
85
- * 특정 이벤트의 리스너 수 반환
86
- *
87
- * @param type 이벤트 타입
88
- * @returns 등록된 리스너 수
89
- */
90
- listenerCount<K extends keyof T & string>(type: K): number {
91
- return this._listenerMap.get(type)?.size ?? 0;
92
- }
93
-
94
- /**
95
- * 모든 이벤트 리스너를 제거한다.
96
- */
97
- dispose(): void {
98
- for (const [type, typeMap] of this._listenerMap) {
99
- for (const wrappedListener of typeMap.values()) {
100
- this._target.removeEventListener(type, wrappedListener);
101
- }
102
- }
103
- this._listenerMap.clear();
104
- }
105
-
106
- /**
107
- * using 문 지원
108
- */
109
- [Symbol.dispose](): void {
110
- this.dispose();
111
- }
112
- }