@alibarbar/common 1.0.9 → 1.0.10

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 (176) hide show
  1. package/dist/{chunk-O3O67R4I.js → algorithm.cjs} +0 -2
  2. package/dist/algorithm.js +130 -44
  3. package/dist/{chunk-LOAZSPGQ.js → array.cjs} +0 -2
  4. package/dist/array.js +135 -84
  5. package/dist/{chunk-Y364QIQH.js → color.cjs} +0 -2
  6. package/dist/color.js +127 -40
  7. package/dist/{chunk-DYBSRI7V.js → crypto.cjs} +0 -2
  8. package/dist/crypto.js +172 -60
  9. package/dist/{chunk-BHCRFURU.js → data-structure.cjs} +0 -2
  10. package/dist/data-structure.js +481 -32
  11. package/dist/{chunk-I3L42475.js → date.cjs} +0 -2
  12. package/dist/date.js +125 -72
  13. package/dist/{chunk-JK2SE3I2.js → dom.cjs} +0 -2
  14. package/dist/dom.js +84 -56
  15. package/dist/{chunk-7V5UQXIO.js → file.cjs} +0 -2
  16. package/dist/file.js +79 -32
  17. package/dist/{chunk-OX5PLOWB.js → i18n.cjs} +0 -2
  18. package/dist/i18n.js +78 -40
  19. package/dist/index.cjs +4132 -0
  20. package/dist/index.d.mts +3 -159
  21. package/dist/index.d.ts +3 -159
  22. package/dist/index.js +3364 -17427
  23. package/dist/{chunk-4RGXV4SJ.js → network.cjs} +0 -2
  24. package/dist/network.js +97 -28
  25. package/dist/{chunk-3FRYJPHG.js → number.cjs} +0 -2
  26. package/dist/number.js +77 -56
  27. package/dist/{chunk-27UDDVLZ.js → object.cjs} +0 -2
  28. package/dist/object.js +237 -80
  29. package/dist/{chunk-JQZBPAPO.js → performance.cjs} +0 -2
  30. package/dist/performance.js +145 -40
  31. package/dist/services.cjs +291 -0
  32. package/dist/services.d.mts +161 -0
  33. package/dist/services.d.ts +161 -0
  34. package/dist/services.js +281 -0
  35. package/dist/{chunk-WZDOPUJW.js → storage.cjs} +216 -17
  36. package/dist/storage.js +550 -31
  37. package/dist/{chunk-D7CS5EKF.js → string.cjs} +0 -2
  38. package/dist/string.js +91 -68
  39. package/dist/{chunk-56W6YECK.js → tracking.cjs} +0 -2
  40. package/dist/tracking.js +359 -52
  41. package/dist/{chunk-ZDMFMUDR.js → transform.cjs} +0 -2
  42. package/dist/transform.js +299 -32
  43. package/dist/{chunk-NJARVI6X.mjs → upload.cjs} +41 -14
  44. package/dist/upload.js +402 -20
  45. package/dist/{chunk-KGFTD255.js → url.cjs} +0 -2
  46. package/dist/url.js +91 -44
  47. package/dist/{chunk-TQN37HIN.js → validation.cjs} +0 -2
  48. package/dist/validation.js +77 -60
  49. package/package.json +7 -2
  50. package/dist/algorithm.js.map +0 -1
  51. package/dist/algorithm.mjs +0 -4
  52. package/dist/algorithm.mjs.map +0 -1
  53. package/dist/array.js.map +0 -1
  54. package/dist/array.mjs +0 -4
  55. package/dist/array.mjs.map +0 -1
  56. package/dist/chunk-27UDDVLZ.js.map +0 -1
  57. package/dist/chunk-2R2QWFJC.mjs +0 -138
  58. package/dist/chunk-2R2QWFJC.mjs.map +0 -1
  59. package/dist/chunk-3FRYJPHG.js.map +0 -1
  60. package/dist/chunk-4RGXV4SJ.js.map +0 -1
  61. package/dist/chunk-56W6YECK.js.map +0 -1
  62. package/dist/chunk-5BGSUGTI.mjs +0 -128
  63. package/dist/chunk-5BGSUGTI.mjs.map +0 -1
  64. package/dist/chunk-7E6GELHJ.mjs +0 -302
  65. package/dist/chunk-7E6GELHJ.mjs.map +0 -1
  66. package/dist/chunk-7V5UQXIO.js.map +0 -1
  67. package/dist/chunk-A4SWQXX7.mjs +0 -484
  68. package/dist/chunk-A4SWQXX7.mjs.map +0 -1
  69. package/dist/chunk-BHCRFURU.js.map +0 -1
  70. package/dist/chunk-CDSGEAOK.mjs +0 -80
  71. package/dist/chunk-CDSGEAOK.mjs.map +0 -1
  72. package/dist/chunk-D7CS5EKF.js.map +0 -1
  73. package/dist/chunk-DYBSRI7V.js.map +0 -1
  74. package/dist/chunk-FEBKPX5A.js +0 -386
  75. package/dist/chunk-FEBKPX5A.js.map +0 -1
  76. package/dist/chunk-FJ6ZGZIA.mjs +0 -39
  77. package/dist/chunk-FJ6ZGZIA.mjs.map +0 -1
  78. package/dist/chunk-HLDFI7R2.mjs +0 -175
  79. package/dist/chunk-HLDFI7R2.mjs.map +0 -1
  80. package/dist/chunk-I3L42475.js.map +0 -1
  81. package/dist/chunk-JBLX27WD.mjs +0 -240
  82. package/dist/chunk-JBLX27WD.mjs.map +0 -1
  83. package/dist/chunk-JHZ7M2MR.mjs +0 -133
  84. package/dist/chunk-JHZ7M2MR.mjs.map +0 -1
  85. package/dist/chunk-JK2SE3I2.js.map +0 -1
  86. package/dist/chunk-JQZBPAPO.js.map +0 -1
  87. package/dist/chunk-JXYGC2C5.mjs +0 -100
  88. package/dist/chunk-JXYGC2C5.mjs.map +0 -1
  89. package/dist/chunk-KGFTD255.js.map +0 -1
  90. package/dist/chunk-LBHBNPNJ.mjs +0 -148
  91. package/dist/chunk-LBHBNPNJ.mjs.map +0 -1
  92. package/dist/chunk-LF4CILQS.mjs +0 -87
  93. package/dist/chunk-LF4CILQS.mjs.map +0 -1
  94. package/dist/chunk-LOAZSPGQ.js.map +0 -1
  95. package/dist/chunk-NJARVI6X.mjs.map +0 -1
  96. package/dist/chunk-NSSDYX2U.mjs +0 -80
  97. package/dist/chunk-NSSDYX2U.mjs.map +0 -1
  98. package/dist/chunk-O3O67R4I.js.map +0 -1
  99. package/dist/chunk-OIXQ3E6W.mjs +0 -354
  100. package/dist/chunk-OIXQ3E6W.mjs.map +0 -1
  101. package/dist/chunk-OX5PLOWB.js.map +0 -1
  102. package/dist/chunk-PJ7UCTX4.mjs +0 -362
  103. package/dist/chunk-PJ7UCTX4.mjs.map +0 -1
  104. package/dist/chunk-PR4QN5HX.js +0 -44
  105. package/dist/chunk-PR4QN5HX.js.map +0 -1
  106. package/dist/chunk-QIBE7GVN.mjs +0 -81
  107. package/dist/chunk-QIBE7GVN.mjs.map +0 -1
  108. package/dist/chunk-QIOC54LQ.mjs +0 -130
  109. package/dist/chunk-QIOC54LQ.mjs.map +0 -1
  110. package/dist/chunk-TQN37HIN.js.map +0 -1
  111. package/dist/chunk-WZDOPUJW.js.map +0 -1
  112. package/dist/chunk-XJTZDXSR.mjs +0 -94
  113. package/dist/chunk-XJTZDXSR.mjs.map +0 -1
  114. package/dist/chunk-Y364QIQH.js.map +0 -1
  115. package/dist/chunk-YXM6Q4JS.mjs +0 -94
  116. package/dist/chunk-YXM6Q4JS.mjs.map +0 -1
  117. package/dist/chunk-ZDMFMUDR.js.map +0 -1
  118. package/dist/chunk-ZVJ6NQUM.mjs +0 -82
  119. package/dist/chunk-ZVJ6NQUM.mjs.map +0 -1
  120. package/dist/color.js.map +0 -1
  121. package/dist/color.mjs +0 -4
  122. package/dist/color.mjs.map +0 -1
  123. package/dist/crypto.js.map +0 -1
  124. package/dist/crypto.mjs +0 -4
  125. package/dist/crypto.mjs.map +0 -1
  126. package/dist/data-structure.js.map +0 -1
  127. package/dist/data-structure.mjs +0 -4
  128. package/dist/data-structure.mjs.map +0 -1
  129. package/dist/date.js.map +0 -1
  130. package/dist/date.mjs +0 -4
  131. package/dist/date.mjs.map +0 -1
  132. package/dist/dom.js.map +0 -1
  133. package/dist/dom.mjs +0 -4
  134. package/dist/dom.mjs.map +0 -1
  135. package/dist/file.js.map +0 -1
  136. package/dist/file.mjs +0 -4
  137. package/dist/file.mjs.map +0 -1
  138. package/dist/i18n.js.map +0 -1
  139. package/dist/i18n.mjs +0 -4
  140. package/dist/i18n.mjs.map +0 -1
  141. package/dist/index.js.map +0 -1
  142. package/dist/index.mjs +0 -17155
  143. package/dist/index.mjs.map +0 -1
  144. package/dist/network.js.map +0 -1
  145. package/dist/network.mjs +0 -4
  146. package/dist/network.mjs.map +0 -1
  147. package/dist/number.js.map +0 -1
  148. package/dist/number.mjs +0 -4
  149. package/dist/number.mjs.map +0 -1
  150. package/dist/object.js.map +0 -1
  151. package/dist/object.mjs +0 -4
  152. package/dist/object.mjs.map +0 -1
  153. package/dist/performance.js.map +0 -1
  154. package/dist/performance.mjs +0 -4
  155. package/dist/performance.mjs.map +0 -1
  156. package/dist/storage.js.map +0 -1
  157. package/dist/storage.mjs +0 -5
  158. package/dist/storage.mjs.map +0 -1
  159. package/dist/string.js.map +0 -1
  160. package/dist/string.mjs +0 -4
  161. package/dist/string.mjs.map +0 -1
  162. package/dist/tracking.js.map +0 -1
  163. package/dist/tracking.mjs +0 -4
  164. package/dist/tracking.mjs.map +0 -1
  165. package/dist/transform.js.map +0 -1
  166. package/dist/transform.mjs +0 -4
  167. package/dist/transform.mjs.map +0 -1
  168. package/dist/upload.js.map +0 -1
  169. package/dist/upload.mjs +0 -5
  170. package/dist/upload.mjs.map +0 -1
  171. package/dist/url.js.map +0 -1
  172. package/dist/url.mjs +0 -4
  173. package/dist/url.mjs.map +0 -1
  174. package/dist/validation.js.map +0 -1
  175. package/dist/validation.mjs +0 -4
  176. package/dist/validation.mjs.map +0 -1
@@ -1,362 +0,0 @@
1
- // src/helper/tracking.ts
2
- var Tracker = class {
3
- constructor(options) {
4
- this.eventQueue = [];
5
- this.batchTimer = null;
6
- this.userInfo = {};
7
- this.exposureObservers = /* @__PURE__ */ new Map();
8
- this.exposureTimers = /* @__PURE__ */ new Map();
9
- this.exposedElements = /* @__PURE__ */ new Set();
10
- if (!options.endpoint && !options.customSend) {
11
- throw new Error("Either endpoint or customSend must be provided");
12
- }
13
- this.options = {
14
- endpoint: options.endpoint || "",
15
- batchSize: options.batchSize ?? 10,
16
- batchDelay: options.batchDelay ?? 3e3,
17
- autoTrackPageView: options.autoTrackPageView ?? true,
18
- autoTrackClick: options.autoTrackClick ?? false,
19
- commonParams: options.commonParams || {},
20
- customSend: options.customSend,
21
- debug: options.debug ?? false
22
- };
23
- this.sessionId = this.generateSessionId();
24
- if (this.options.autoTrackPageView && typeof window !== "undefined") {
25
- this.trackPageView();
26
- this.setupPageViewListener();
27
- }
28
- if (this.options.autoTrackClick && typeof window !== "undefined") {
29
- this.setupClickListener();
30
- }
31
- }
32
- /**
33
- * 生成会话ID
34
- */
35
- generateSessionId() {
36
- return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;
37
- }
38
- /**
39
- * 获取公共数据
40
- */
41
- getCommonData() {
42
- const data = {
43
- timestamp: Date.now(),
44
- sessionId: this.sessionId,
45
- ...this.options.commonParams,
46
- ...this.userInfo
47
- };
48
- if (typeof window !== "undefined") {
49
- data.url = window.location.href;
50
- data.userAgent = navigator.userAgent;
51
- }
52
- return data;
53
- }
54
- /**
55
- * 添加事件到队列
56
- */
57
- enqueue(event) {
58
- this.eventQueue.push(event);
59
- if (this.options.debug) {
60
- console.log("[Tracker] Event enqueued:", event);
61
- }
62
- if (this.eventQueue.length >= this.options.batchSize) {
63
- this.flush();
64
- } else {
65
- this.scheduleBatch();
66
- }
67
- }
68
- /**
69
- * 安排批量上报
70
- */
71
- scheduleBatch() {
72
- if (this.batchTimer) {
73
- clearTimeout(this.batchTimer);
74
- }
75
- this.batchTimer = setTimeout(() => {
76
- this.flush();
77
- }, this.options.batchDelay);
78
- }
79
- /**
80
- * 上报事件
81
- */
82
- async sendEvents(events) {
83
- if (events.length === 0) return;
84
- try {
85
- if (this.options.customSend) {
86
- await this.options.customSend(events);
87
- } else if (this.options.endpoint) {
88
- await this.sendToEndpoint(events);
89
- }
90
- if (this.options.debug) {
91
- console.log("[Tracker] Events sent:", events);
92
- }
93
- } catch (error) {
94
- console.error("[Tracker] Failed to send events:", error);
95
- this.eventQueue.unshift(...events);
96
- }
97
- }
98
- /**
99
- * 发送到上报接口
100
- */
101
- async sendToEndpoint(events) {
102
- const response = await fetch(this.options.endpoint, {
103
- method: "POST",
104
- headers: {
105
- "Content-Type": "application/json"
106
- },
107
- body: JSON.stringify({ events })
108
- });
109
- if (!response.ok) {
110
- throw new Error(`HTTP error! status: ${response.status}`);
111
- }
112
- }
113
- /**
114
- * 立即上报所有待上报事件
115
- */
116
- async flush() {
117
- if (this.batchTimer) {
118
- clearTimeout(this.batchTimer);
119
- this.batchTimer = null;
120
- }
121
- if (this.eventQueue.length === 0) return;
122
- const events = [...this.eventQueue];
123
- this.eventQueue = [];
124
- await this.sendEvents(events);
125
- }
126
- /**
127
- * 追踪事件
128
- */
129
- trackEvent(name, params) {
130
- const event = {
131
- type: "event",
132
- name,
133
- params,
134
- ...this.getCommonData()
135
- };
136
- this.enqueue(event);
137
- }
138
- /**
139
- * 追踪页面浏览
140
- */
141
- trackPageView(params) {
142
- const event = {
143
- type: "pageview",
144
- name: "pageview",
145
- params: {
146
- path: typeof window !== "undefined" ? window.location.pathname : void 0,
147
- search: typeof window !== "undefined" ? window.location.search : void 0,
148
- hash: typeof window !== "undefined" ? window.location.hash : void 0,
149
- ...params
150
- },
151
- ...this.getCommonData()
152
- };
153
- this.enqueue(event);
154
- }
155
- /**
156
- * 追踪点击事件
157
- */
158
- trackClick(element, params) {
159
- const el = typeof element === "string" ? document.querySelector(element) : element;
160
- if (!el) {
161
- if (this.options.debug) {
162
- console.warn("[Tracker] Element not found:", element);
163
- }
164
- return;
165
- }
166
- const event = {
167
- type: "click",
168
- name: "click",
169
- params: {
170
- element: el.tagName.toLowerCase(),
171
- id: el.id || void 0,
172
- className: el.className || void 0,
173
- text: el.textContent?.trim().substring(0, 100) || void 0,
174
- ...params
175
- },
176
- ...this.getCommonData()
177
- };
178
- this.enqueue(event);
179
- }
180
- /**
181
- * 追踪曝光事件
182
- */
183
- trackExposure(element, options, params) {
184
- if (typeof window === "undefined" || !window.IntersectionObserver) {
185
- if (this.options.debug) {
186
- console.warn("[Tracker] IntersectionObserver is not supported");
187
- }
188
- return;
189
- }
190
- const el = typeof element === "string" ? document.querySelector(element) : element;
191
- if (!el) {
192
- if (this.options.debug) {
193
- console.warn("[Tracker] Element not found:", element);
194
- }
195
- return;
196
- }
197
- const { threshold = 0.5, duration = 1e3, once = true } = options || {};
198
- const elementId = `${Date.now()}-${Math.random()}`;
199
- if (once && this.exposedElements.has(el)) {
200
- return;
201
- }
202
- let exposureTimer = null;
203
- let startTime = null;
204
- const observer = new IntersectionObserver(
205
- (entries) => {
206
- entries.forEach((entry) => {
207
- if (entry.isIntersecting && entry.intersectionRatio >= threshold) {
208
- if (!startTime) {
209
- startTime = Date.now();
210
- const currentStartTime = startTime;
211
- exposureTimer = setTimeout(() => {
212
- const event = {
213
- type: "exposure",
214
- name: "exposure",
215
- params: {
216
- element: el.tagName.toLowerCase(),
217
- id: el.id || void 0,
218
- className: el.className || void 0,
219
- duration: currentStartTime ? Date.now() - currentStartTime : duration,
220
- ...params
221
- },
222
- ...this.getCommonData()
223
- };
224
- this.enqueue(event);
225
- this.exposedElements.add(el);
226
- observer.disconnect();
227
- this.exposureObservers.delete(elementId);
228
- if (exposureTimer) {
229
- clearTimeout(exposureTimer);
230
- this.exposureTimers.delete(elementId);
231
- }
232
- }, duration);
233
- this.exposureTimers.set(elementId, exposureTimer);
234
- }
235
- } else {
236
- if (exposureTimer) {
237
- clearTimeout(exposureTimer);
238
- this.exposureTimers.delete(elementId);
239
- }
240
- startTime = null;
241
- }
242
- });
243
- },
244
- {
245
- threshold: [threshold]
246
- }
247
- );
248
- observer.observe(el);
249
- this.exposureObservers.set(elementId, observer);
250
- }
251
- /**
252
- * 设置用户信息
253
- */
254
- setUserInfo(userInfo) {
255
- this.userInfo = { ...this.userInfo, ...userInfo };
256
- }
257
- /**
258
- * 设置公共参数
259
- */
260
- setCommonParams(params) {
261
- this.options.commonParams = { ...this.options.commonParams, ...params };
262
- }
263
- /**
264
- * 设置页面浏览监听器(SPA应用)
265
- */
266
- setupPageViewListener() {
267
- if (typeof window === "undefined") return;
268
- window.addEventListener("popstate", () => {
269
- this.trackPageView();
270
- });
271
- const originalPushState = history.pushState;
272
- const originalReplaceState = history.replaceState;
273
- history.pushState = function(...args) {
274
- originalPushState.apply(history, args);
275
- window.dispatchEvent(new Event("pushstate"));
276
- };
277
- history.replaceState = function(...args) {
278
- originalReplaceState.apply(history, args);
279
- window.dispatchEvent(new Event("replacestate"));
280
- };
281
- window.addEventListener("pushstate", () => {
282
- this.trackPageView();
283
- });
284
- window.addEventListener("replacestate", () => {
285
- this.trackPageView();
286
- });
287
- }
288
- /**
289
- * 设置点击事件监听器
290
- */
291
- setupClickListener() {
292
- if (typeof window === "undefined" || typeof document === "undefined") return;
293
- document.addEventListener(
294
- "click",
295
- (event) => {
296
- const target = event.target;
297
- if (target) {
298
- this.trackClick(target);
299
- }
300
- },
301
- true
302
- // 使用捕获阶段
303
- );
304
- }
305
- /**
306
- * 清理资源
307
- */
308
- destroy() {
309
- if (this.batchTimer) {
310
- clearTimeout(this.batchTimer);
311
- this.batchTimer = null;
312
- }
313
- this.exposureObservers.forEach((observer) => observer.disconnect());
314
- this.exposureObservers.clear();
315
- this.exposureTimers.forEach((timer) => clearTimeout(timer));
316
- this.exposureTimers.clear();
317
- this.flush().catch(console.error);
318
- }
319
- };
320
- function createTracker(options) {
321
- return new Tracker(options);
322
- }
323
- var defaultTracker = null;
324
- function initTracker(options) {
325
- if (defaultTracker) {
326
- console.warn("[Tracker] Default tracker already initialized");
327
- return defaultTracker;
328
- }
329
- defaultTracker = new Tracker(options);
330
- return defaultTracker;
331
- }
332
- function getTracker() {
333
- if (!defaultTracker) {
334
- throw new Error("Tracker not initialized. Call initTracker() first.");
335
- }
336
- return defaultTracker;
337
- }
338
- function trackEvent(name, params) {
339
- getTracker().trackEvent(name, params);
340
- }
341
- function trackPageView(params) {
342
- getTracker().trackPageView(params);
343
- }
344
- function trackClick(element, params) {
345
- getTracker().trackClick(element, params);
346
- }
347
- function trackExposure(element, options, params) {
348
- getTracker().trackExposure(element, options, params);
349
- }
350
- function setUserInfo(userInfo) {
351
- getTracker().setUserInfo(userInfo);
352
- }
353
- function setCommonParams(params) {
354
- getTracker().setCommonParams(params);
355
- }
356
- async function flush() {
357
- await getTracker().flush();
358
- }
359
-
360
- export { Tracker, createTracker, flush, getTracker, initTracker, setCommonParams, setUserInfo, trackClick, trackEvent, trackExposure, trackPageView };
361
- //# sourceMappingURL=chunk-PJ7UCTX4.mjs.map
362
- //# sourceMappingURL=chunk-PJ7UCTX4.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/helper/tracking.ts"],"names":[],"mappings":"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","file":"chunk-PJ7UCTX4.mjs","sourcesContent":["/**\n * 埋点追踪工具\n */\n\n/**\n * 埋点事件类型\n */\nexport type TrackingEventType = 'event' | 'pageview' | 'click' | 'exposure' | 'custom';\n\n/**\n * 埋点事件数据\n */\nexport interface TrackingEvent {\n /** 事件类型 */\n type: TrackingEventType;\n /** 事件名称 */\n name: string;\n /** 事件参数 */\n params?: Record<string, unknown>;\n /** 时间戳 */\n timestamp: number;\n /** 页面URL */\n url?: string;\n /** 用户代理 */\n userAgent?: string;\n /** 用户ID */\n userId?: string;\n /** 会话ID */\n sessionId?: string;\n /** 其他自定义字段 */\n [key: string]: unknown;\n}\n\n/**\n * 埋点配置选项\n */\nexport interface TrackerOptions {\n /** 上报接口URL */\n endpoint: string;\n /** 批量上报大小,默认10 */\n batchSize?: number;\n /** 批量上报延迟(毫秒),默认3000 */\n batchDelay?: number;\n /** 是否自动追踪页面浏览,默认true */\n autoTrackPageView?: boolean;\n /** 是否自动追踪点击事件,默认false */\n autoTrackClick?: boolean;\n /** 公共参数 */\n commonParams?: Record<string, unknown>;\n /** 自定义上报函数 */\n customSend?: (events: TrackingEvent[]) => Promise<void>;\n /** 是否启用调试模式,默认false */\n debug?: boolean;\n}\n\n/**\n * 曝光追踪选项\n */\nexport interface ExposureOptions {\n /** 曝光阈值(0-1),元素可见比例达到此值才触发,默认0.5 */\n threshold?: number;\n /** 曝光时长(毫秒),元素可见时长达到此值才触发,默认1000 */\n duration?: number;\n /** 是否只触发一次,默认true */\n once?: boolean;\n}\n\n/**\n * 埋点追踪器类\n */\nexport class Tracker {\n private options: Required<\n Pick<\n TrackerOptions,\n 'batchSize' | 'batchDelay' | 'autoTrackPageView' | 'autoTrackClick' | 'debug'\n >\n > &\n Pick<TrackerOptions, 'endpoint' | 'commonParams' | 'customSend'>;\n private eventQueue: TrackingEvent[] = [];\n private batchTimer: ReturnType<typeof setTimeout> | null = null;\n private userInfo: Record<string, unknown> = {};\n private sessionId: string;\n private exposureObservers: Map<string, IntersectionObserver> = new Map();\n private exposureTimers: Map<string, ReturnType<typeof setTimeout>> = new Map();\n private exposedElements: Set<Element> = new Set();\n\n constructor(options: TrackerOptions) {\n if (!options.endpoint && !options.customSend) {\n throw new Error('Either endpoint or customSend must be provided');\n }\n\n this.options = {\n endpoint: options.endpoint || '',\n batchSize: options.batchSize ?? 10,\n batchDelay: options.batchDelay ?? 3000,\n autoTrackPageView: options.autoTrackPageView ?? true,\n autoTrackClick: options.autoTrackClick ?? false,\n commonParams: options.commonParams || {},\n customSend: options.customSend,\n debug: options.debug ?? false,\n };\n\n // 生成会话ID\n this.sessionId = this.generateSessionId();\n\n // 自动追踪页面浏览\n if (this.options.autoTrackPageView && typeof window !== 'undefined') {\n this.trackPageView();\n // 监听路由变化(SPA应用)\n this.setupPageViewListener();\n }\n\n // 自动追踪点击事件\n if (this.options.autoTrackClick && typeof window !== 'undefined') {\n this.setupClickListener();\n }\n }\n\n /**\n * 生成会话ID\n */\n private generateSessionId(): string {\n return `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`;\n }\n\n /**\n * 获取公共数据\n */\n private getCommonData(): Pick<TrackingEvent, 'timestamp' | 'sessionId'> &\n Partial<Omit<TrackingEvent, 'timestamp' | 'sessionId'>> {\n const data: Pick<TrackingEvent, 'timestamp' | 'sessionId'> &\n Partial<Omit<TrackingEvent, 'timestamp' | 'sessionId'>> = {\n timestamp: Date.now(),\n sessionId: this.sessionId,\n ...this.options.commonParams,\n ...this.userInfo,\n };\n\n if (typeof window !== 'undefined') {\n data.url = window.location.href;\n data.userAgent = navigator.userAgent;\n }\n\n return data;\n }\n\n /**\n * 添加事件到队列\n */\n private enqueue(event: TrackingEvent): void {\n this.eventQueue.push(event);\n\n if (this.options.debug) {\n console.log('[Tracker] Event enqueued:', event);\n }\n\n // 如果队列达到批量大小,立即上报\n if (this.eventQueue.length >= this.options.batchSize) {\n this.flush();\n } else {\n // 否则设置延迟上报\n this.scheduleBatch();\n }\n }\n\n /**\n * 安排批量上报\n */\n private scheduleBatch(): void {\n if (this.batchTimer) {\n clearTimeout(this.batchTimer);\n }\n\n this.batchTimer = setTimeout(() => {\n this.flush();\n }, this.options.batchDelay);\n }\n\n /**\n * 上报事件\n */\n private async sendEvents(events: TrackingEvent[]): Promise<void> {\n if (events.length === 0) return;\n\n try {\n if (this.options.customSend) {\n await this.options.customSend(events);\n } else if (this.options.endpoint) {\n await this.sendToEndpoint(events);\n }\n\n if (this.options.debug) {\n console.log('[Tracker] Events sent:', events);\n }\n } catch (error) {\n console.error('[Tracker] Failed to send events:', error);\n // 失败后重新加入队列(可选:限制重试次数)\n this.eventQueue.unshift(...events);\n }\n }\n\n /**\n * 发送到上报接口\n */\n private async sendToEndpoint(events: TrackingEvent[]): Promise<void> {\n const response = await fetch(this.options.endpoint, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({ events }),\n });\n\n if (!response.ok) {\n throw new Error(`HTTP error! status: ${response.status}`);\n }\n }\n\n /**\n * 立即上报所有待上报事件\n */\n async flush(): Promise<void> {\n if (this.batchTimer) {\n clearTimeout(this.batchTimer);\n this.batchTimer = null;\n }\n\n if (this.eventQueue.length === 0) return;\n\n const events = [...this.eventQueue];\n this.eventQueue = [];\n\n await this.sendEvents(events);\n }\n\n /**\n * 追踪事件\n */\n trackEvent(name: string, params?: Record<string, unknown>): void {\n const event: TrackingEvent = {\n type: 'event',\n name,\n params,\n ...this.getCommonData(),\n };\n\n this.enqueue(event);\n }\n\n /**\n * 追踪页面浏览\n */\n trackPageView(params?: Record<string, unknown>): void {\n const event: TrackingEvent = {\n type: 'pageview',\n name: 'pageview',\n params: {\n path: typeof window !== 'undefined' ? window.location.pathname : undefined,\n search: typeof window !== 'undefined' ? window.location.search : undefined,\n hash: typeof window !== 'undefined' ? window.location.hash : undefined,\n ...params,\n },\n ...this.getCommonData(),\n };\n\n this.enqueue(event);\n }\n\n /**\n * 追踪点击事件\n */\n trackClick(element: Element | string, params?: Record<string, unknown>): void {\n const el = typeof element === 'string' ? document.querySelector(element) : element;\n if (!el) {\n if (this.options.debug) {\n console.warn('[Tracker] Element not found:', element);\n }\n return;\n }\n\n const event: TrackingEvent = {\n type: 'click',\n name: 'click',\n params: {\n element: el.tagName.toLowerCase(),\n id: el.id || undefined,\n className: el.className || undefined,\n text: el.textContent?.trim().substring(0, 100) || undefined,\n ...params,\n },\n ...this.getCommonData(),\n };\n\n this.enqueue(event);\n }\n\n /**\n * 追踪曝光事件\n */\n trackExposure(\n element: Element | string,\n options?: ExposureOptions,\n params?: Record<string, unknown>\n ): void {\n if (typeof window === 'undefined' || !window.IntersectionObserver) {\n if (this.options.debug) {\n console.warn('[Tracker] IntersectionObserver is not supported');\n }\n return;\n }\n\n const el = typeof element === 'string' ? document.querySelector(element) : element;\n if (!el) {\n if (this.options.debug) {\n console.warn('[Tracker] Element not found:', element);\n }\n return;\n }\n\n const { threshold = 0.5, duration = 1000, once = true } = options || {};\n\n const elementId = `${Date.now()}-${Math.random()}`;\n\n // 如果已经曝光过且只触发一次,直接返回\n if (once && this.exposedElements.has(el)) {\n return;\n }\n\n let exposureTimer: ReturnType<typeof setTimeout> | null = null;\n let startTime: number | null = null;\n\n const observer = new IntersectionObserver(\n entries => {\n entries.forEach(entry => {\n if (entry.isIntersecting && entry.intersectionRatio >= threshold) {\n // 开始计时\n if (!startTime) {\n startTime = Date.now();\n const currentStartTime = startTime;\n exposureTimer = setTimeout(() => {\n // 触发曝光事件\n const event: TrackingEvent = {\n type: 'exposure',\n name: 'exposure',\n params: {\n element: el.tagName.toLowerCase(),\n id: el.id || undefined,\n className: el.className || undefined,\n duration: currentStartTime ? Date.now() - currentStartTime : duration,\n ...params,\n },\n ...this.getCommonData(),\n };\n\n this.enqueue(event);\n this.exposedElements.add(el);\n\n // 清理\n observer.disconnect();\n this.exposureObservers.delete(elementId);\n if (exposureTimer) {\n clearTimeout(exposureTimer);\n this.exposureTimers.delete(elementId);\n }\n }, duration);\n this.exposureTimers.set(elementId, exposureTimer);\n }\n } else {\n // 不可见,重置计时\n if (exposureTimer) {\n clearTimeout(exposureTimer);\n this.exposureTimers.delete(elementId);\n }\n startTime = null;\n }\n });\n },\n {\n threshold: [threshold],\n }\n );\n\n observer.observe(el);\n this.exposureObservers.set(elementId, observer);\n }\n\n /**\n * 设置用户信息\n */\n setUserInfo(userInfo: Record<string, unknown>): void {\n this.userInfo = { ...this.userInfo, ...userInfo };\n }\n\n /**\n * 设置公共参数\n */\n setCommonParams(params: Record<string, unknown>): void {\n this.options.commonParams = { ...this.options.commonParams, ...params };\n }\n\n /**\n * 设置页面浏览监听器(SPA应用)\n */\n private setupPageViewListener(): void {\n if (typeof window === 'undefined') return;\n\n // 监听popstate事件(浏览器前进后退)\n window.addEventListener('popstate', () => {\n this.trackPageView();\n });\n\n // 监听pushState和replaceState(需要重写history方法)\n const originalPushState = history.pushState;\n const originalReplaceState = history.replaceState;\n\n history.pushState = function (...args) {\n originalPushState.apply(history, args);\n window.dispatchEvent(new Event('pushstate'));\n };\n\n history.replaceState = function (...args) {\n originalReplaceState.apply(history, args);\n window.dispatchEvent(new Event('replacestate'));\n };\n\n window.addEventListener('pushstate', () => {\n this.trackPageView();\n });\n\n window.addEventListener('replacestate', () => {\n this.trackPageView();\n });\n }\n\n /**\n * 设置点击事件监听器\n */\n private setupClickListener(): void {\n if (typeof window === 'undefined' || typeof document === 'undefined') return;\n\n document.addEventListener(\n 'click',\n event => {\n const target = event.target as Element;\n if (target) {\n this.trackClick(target);\n }\n },\n true // 使用捕获阶段\n );\n }\n\n /**\n * 清理资源\n */\n destroy(): void {\n // 清理定时器\n if (this.batchTimer) {\n clearTimeout(this.batchTimer);\n this.batchTimer = null;\n }\n\n // 清理曝光观察器\n this.exposureObservers.forEach(observer => observer.disconnect());\n this.exposureObservers.clear();\n\n // 清理曝光定时器\n this.exposureTimers.forEach(timer => clearTimeout(timer));\n this.exposureTimers.clear();\n\n // 上报剩余事件\n this.flush().catch(console.error);\n }\n}\n\n/**\n * 创建埋点追踪器实例\n */\nexport function createTracker(options: TrackerOptions): Tracker {\n return new Tracker(options);\n}\n\n/**\n * 默认追踪器实例(单例模式)\n */\nlet defaultTracker: Tracker | null = null;\n\n/**\n * 初始化默认追踪器\n */\nexport function initTracker(options: TrackerOptions): Tracker {\n if (defaultTracker) {\n console.warn('[Tracker] Default tracker already initialized');\n return defaultTracker;\n }\n defaultTracker = new Tracker(options);\n return defaultTracker;\n}\n\n/**\n * 获取默认追踪器\n */\nexport function getTracker(): Tracker {\n if (!defaultTracker) {\n throw new Error('Tracker not initialized. Call initTracker() first.');\n }\n return defaultTracker;\n}\n\n/**\n * 追踪事件(使用默认追踪器)\n */\nexport function trackEvent(name: string, params?: Record<string, unknown>): void {\n getTracker().trackEvent(name, params);\n}\n\n/**\n * 追踪页面浏览(使用默认追踪器)\n */\nexport function trackPageView(params?: Record<string, unknown>): void {\n getTracker().trackPageView(params);\n}\n\n/**\n * 追踪点击事件(使用默认追踪器)\n */\nexport function trackClick(element: Element | string, params?: Record<string, unknown>): void {\n getTracker().trackClick(element, params);\n}\n\n/**\n * 追踪曝光事件(使用默认追踪器)\n */\nexport function trackExposure(\n element: Element | string,\n options?: ExposureOptions,\n params?: Record<string, unknown>\n): void {\n getTracker().trackExposure(element, options, params);\n}\n\n/**\n * 设置用户信息(使用默认追踪器)\n */\nexport function setUserInfo(userInfo: Record<string, unknown>): void {\n getTracker().setUserInfo(userInfo);\n}\n\n/**\n * 设置公共参数(使用默认追踪器)\n */\nexport function setCommonParams(params: Record<string, unknown>): void {\n getTracker().setCommonParams(params);\n}\n\n/**\n * 立即上报(使用默认追踪器)\n */\nexport async function flush(): Promise<void> {\n await getTracker().flush();\n}\n"]}
@@ -1,44 +0,0 @@
1
- 'use strict';
2
-
3
- var __create = Object.create;
4
- var __defProp = Object.defineProperty;
5
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
6
- var __getOwnPropNames = Object.getOwnPropertyNames;
7
- var __getProtoOf = Object.getPrototypeOf;
8
- var __hasOwnProp = Object.prototype.hasOwnProperty;
9
- var __require = /* @__PURE__ */ ((x) => typeof require !== "undefined" ? require : typeof Proxy !== "undefined" ? new Proxy(x, {
10
- get: (a, b) => (typeof require !== "undefined" ? require : a)[b]
11
- }) : x)(function(x) {
12
- if (typeof require !== "undefined") return require.apply(this, arguments);
13
- throw Error('Dynamic require of "' + x + '" is not supported');
14
- });
15
- var __commonJS = (cb, mod) => function __require2() {
16
- return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
17
- };
18
- var __export = (target, all) => {
19
- for (var name in all)
20
- __defProp(target, name, { get: all[name], enumerable: true });
21
- };
22
- var __copyProps = (to, from, except, desc) => {
23
- if (from && typeof from === "object" || typeof from === "function") {
24
- for (let key of __getOwnPropNames(from))
25
- if (!__hasOwnProp.call(to, key) && key !== except)
26
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
27
- }
28
- return to;
29
- };
30
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
31
- // If the importer is in node compatibility mode or this is not an ESM
32
- // file that has been converted to a CommonJS file using a Babel-
33
- // compatible transform (i.e. "__esModule" has not been set), then set
34
- // "default" to the CommonJS "module.exports" for node compatibility.
35
- isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
36
- mod
37
- ));
38
-
39
- exports.__commonJS = __commonJS;
40
- exports.__export = __export;
41
- exports.__require = __require;
42
- exports.__toESM = __toESM;
43
- //# sourceMappingURL=chunk-PR4QN5HX.js.map
44
- //# sourceMappingURL=chunk-PR4QN5HX.js.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"chunk-PR4QN5HX.js"}
@@ -1,81 +0,0 @@
1
- // src/format/i18n.ts
2
- function getLocale() {
3
- if (typeof navigator !== "undefined" && navigator.language) {
4
- return navigator.language;
5
- }
6
- return "en-US";
7
- }
8
- function formatNumberI18n(value, locale, options) {
9
- const loc = locale || getLocale();
10
- return new Intl.NumberFormat(loc, options).format(value);
11
- }
12
- function formatDateI18n(date, locale, options) {
13
- const loc = locale || getLocale();
14
- const dateObj = typeof date === "number" ? new Date(date) : date;
15
- return new Intl.DateTimeFormat(loc, options).format(dateObj);
16
- }
17
- function formatCurrencyI18n(value, currency, locale, options) {
18
- const loc = locale || getLocale();
19
- return new Intl.NumberFormat(loc, {
20
- style: "currency",
21
- currency,
22
- ...options
23
- }).format(value);
24
- }
25
- function translate(key, dictionary, defaultValue) {
26
- const keys = key.split(".");
27
- let current = dictionary;
28
- for (const k of keys) {
29
- if (typeof current === "object" && current !== null && k in current) {
30
- current = current[k];
31
- } else {
32
- return defaultValue || key;
33
- }
34
- }
35
- if (typeof current === "string") {
36
- return current;
37
- }
38
- return defaultValue || key;
39
- }
40
- function createTranslator(dictionary) {
41
- return (key, defaultValue) => {
42
- return translate(key, dictionary, defaultValue);
43
- };
44
- }
45
- function formatRelativeTime(date, locale) {
46
- const loc = locale || getLocale();
47
- const dateObj = typeof date === "number" ? new Date(date) : date;
48
- const now = /* @__PURE__ */ new Date();
49
- const diffInSeconds = Math.floor((now.getTime() - dateObj.getTime()) / 1e3);
50
- const rtf = new Intl.RelativeTimeFormat(loc, { numeric: "auto" });
51
- const intervals = [
52
- { unit: "year", seconds: 31536e3 },
53
- { unit: "month", seconds: 2592e3 },
54
- { unit: "week", seconds: 604800 },
55
- { unit: "day", seconds: 86400 },
56
- { unit: "hour", seconds: 3600 },
57
- { unit: "minute", seconds: 60 },
58
- { unit: "second", seconds: 1 }
59
- ];
60
- for (const interval of intervals) {
61
- const count = Math.floor(Math.abs(diffInSeconds) / interval.seconds);
62
- if (count >= 1) {
63
- return rtf.format(diffInSeconds < 0 ? count : -count, interval.unit);
64
- }
65
- }
66
- return rtf.format(0, "second");
67
- }
68
- function pluralize(count, singular, plural, locale) {
69
- const loc = locale || getLocale();
70
- const pluralForm = plural || `${singular}s`;
71
- const pluralRules = new Intl.PluralRules(loc);
72
- const rule = pluralRules.select(count);
73
- if (rule === "one" || count === 1) {
74
- return `${count} ${singular}`;
75
- }
76
- return `${count} ${pluralForm}`;
77
- }
78
-
79
- export { createTranslator, formatCurrencyI18n, formatDateI18n, formatNumberI18n, formatRelativeTime, getLocale, pluralize, translate };
80
- //# sourceMappingURL=chunk-QIBE7GVN.mjs.map
81
- //# sourceMappingURL=chunk-QIBE7GVN.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/format/i18n.ts"],"names":[],"mappings":"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","file":"chunk-QIBE7GVN.mjs","sourcesContent":["/**\n * 国际化工具函数\n */\n\n/**\n * 语言环境类型\n */\nexport type Locale = string;\n\n/**\n * 货币代码类型(ISO 4217)\n */\nexport type CurrencyCode = string;\n\n/**\n * 获取当前语言环境\n * @returns 语言环境字符串(如 'zh-CN', 'en-US')\n */\nexport function getLocale(): Locale {\n if (typeof navigator !== 'undefined' && navigator.language) {\n return navigator.language;\n }\n return 'en-US';\n}\n\n/**\n * 数字本地化格式化\n * @param value - 数字值\n * @param locale - 语言环境,默认为当前环境\n * @param options - Intl.NumberFormat选项\n * @returns 格式化后的字符串\n */\nexport function formatNumberI18n(\n value: number,\n locale?: Locale,\n options?: Intl.NumberFormatOptions\n): string {\n const loc = locale || getLocale();\n return new Intl.NumberFormat(loc, options).format(value);\n}\n\n/**\n * 日期本地化格式化\n * @param date - 日期对象或时间戳\n * @param locale - 语言环境,默认为当前环境\n * @param options - Intl.DateTimeFormat选项\n * @returns 格式化后的字符串\n */\nexport function formatDateI18n(\n date: Date | number,\n locale?: Locale,\n options?: Intl.DateTimeFormatOptions\n): string {\n const loc = locale || getLocale();\n const dateObj = typeof date === 'number' ? new Date(date) : date;\n return new Intl.DateTimeFormat(loc, options).format(dateObj);\n}\n\n/**\n * 货币本地化格式化\n * @param value - 金额\n * @param currency - 货币代码(ISO 4217),如 'USD', 'CNY', 'EUR'\n * @param locale - 语言环境,默认为当前环境\n * @param options - Intl.NumberFormat选项\n * @returns 格式化后的货币字符串\n */\nexport function formatCurrencyI18n(\n value: number,\n currency: CurrencyCode,\n locale?: Locale,\n options?: Intl.NumberFormatOptions\n): string {\n const loc = locale || getLocale();\n return new Intl.NumberFormat(loc, {\n style: 'currency',\n currency,\n ...options,\n }).format(value);\n}\n\n/**\n * 翻译字典类型\n */\nexport type TranslationDictionary = Record<string, string | Record<string, unknown>>;\n\n/**\n * 翻译函数\n * @param key - 翻译键(支持点号分隔的嵌套键,如 'user.name')\n * @param dictionary - 翻译字典\n * @param defaultValue - 默认值(如果找不到翻译)\n * @returns 翻译后的字符串\n */\nexport function translate(\n key: string,\n dictionary: TranslationDictionary,\n defaultValue?: string\n): string {\n const keys = key.split('.');\n let current: unknown = dictionary;\n\n for (const k of keys) {\n if (typeof current === 'object' && current !== null && k in current) {\n current = (current as TranslationDictionary)[k];\n } else {\n return defaultValue || key;\n }\n }\n\n if (typeof current === 'string') {\n return current;\n }\n\n return defaultValue || key;\n}\n\n/**\n * 创建翻译函数(带预设字典)\n * @param dictionary - 翻译字典\n * @returns 翻译函数\n */\nexport function createTranslator(dictionary: TranslationDictionary) {\n return (key: string, defaultValue?: string): string => {\n return translate(key, dictionary, defaultValue);\n };\n}\n\n/**\n * 获取相对时间格式化(本地化)\n * @param date - 日期对象或时间戳\n * @param locale - 语言环境,默认为当前环境\n * @returns 格式化后的相对时间字符串\n */\nexport function formatRelativeTime(date: Date | number, locale?: Locale): string {\n const loc = locale || getLocale();\n const dateObj = typeof date === 'number' ? new Date(date) : date;\n const now = new Date();\n const diffInSeconds = Math.floor((now.getTime() - dateObj.getTime()) / 1000);\n\n const rtf = new Intl.RelativeTimeFormat(loc, { numeric: 'auto' });\n\n const intervals: Array<{ unit: Intl.RelativeTimeFormatUnit; seconds: number }> = [\n { unit: 'year', seconds: 31536000 },\n { unit: 'month', seconds: 2592000 },\n { unit: 'week', seconds: 604800 },\n { unit: 'day', seconds: 86400 },\n { unit: 'hour', seconds: 3600 },\n { unit: 'minute', seconds: 60 },\n { unit: 'second', seconds: 1 },\n ];\n\n for (const interval of intervals) {\n const count = Math.floor(Math.abs(diffInSeconds) / interval.seconds);\n if (count >= 1) {\n return rtf.format(diffInSeconds < 0 ? count : -count, interval.unit);\n }\n }\n\n return rtf.format(0, 'second');\n}\n\n/**\n * 获取复数形式(简单实现)\n * @param count - 数量\n * @param singular - 单数形式\n * @param plural - 复数形式(可选,默认为 singular + 's')\n * @param locale - 语言环境,默认为当前环境\n * @returns 格式化后的字符串\n */\nexport function pluralize(\n count: number,\n singular: string,\n plural?: string,\n locale?: Locale\n): string {\n const loc = locale || getLocale();\n const pluralForm = plural || `${singular}s`;\n\n // 使用 Intl.PluralRules 获取正确的复数形式\n const pluralRules = new Intl.PluralRules(loc);\n const rule = pluralRules.select(count);\n\n // 简单实现:大多数语言中,1 使用单数,其他使用复数\n if (rule === 'one' || count === 1) {\n return `${count} ${singular}`;\n }\n\n return `${count} ${pluralForm}`;\n}\n"]}