@shopbb/helium 0.7.7 → 0.9.0

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 (76) hide show
  1. package/dist/analytics/index.d.ts +18 -0
  2. package/dist/analytics/index.d.ts.map +1 -0
  3. package/dist/analytics/index.js +16 -0
  4. package/dist/analytics/index.js.map +1 -0
  5. package/dist/analytics/queue.d.ts +31 -0
  6. package/dist/analytics/queue.d.ts.map +1 -0
  7. package/dist/analytics/queue.js +203 -0
  8. package/dist/analytics/queue.js.map +1 -0
  9. package/dist/analytics/react.d.ts +44 -0
  10. package/dist/analytics/react.d.ts.map +1 -0
  11. package/dist/analytics/react.js +114 -0
  12. package/dist/analytics/react.js.map +1 -0
  13. package/dist/analytics/types.d.ts +68 -0
  14. package/dist/analytics/types.d.ts.map +1 -0
  15. package/dist/analytics/types.js +7 -0
  16. package/dist/analytics/types.js.map +1 -0
  17. package/dist/components/AddToCartButton.d.ts +7 -0
  18. package/dist/components/AddToCartButton.d.ts.map +1 -1
  19. package/dist/components/AddToCartButton.js +13 -2
  20. package/dist/components/AddToCartButton.js.map +1 -1
  21. package/dist/page-schema/PageRenderer.d.ts +29 -0
  22. package/dist/page-schema/PageRenderer.d.ts.map +1 -0
  23. package/dist/page-schema/PageRenderer.js +73 -0
  24. package/dist/page-schema/PageRenderer.js.map +1 -0
  25. package/dist/page-schema/index.d.ts +20 -0
  26. package/dist/page-schema/index.d.ts.map +1 -0
  27. package/dist/page-schema/index.js +18 -0
  28. package/dist/page-schema/index.js.map +1 -0
  29. package/dist/page-schema/sections/Banner.d.ts +10 -0
  30. package/dist/page-schema/sections/Banner.d.ts.map +1 -0
  31. package/dist/page-schema/sections/Banner.js +35 -0
  32. package/dist/page-schema/sections/Banner.js.map +1 -0
  33. package/dist/page-schema/sections/Hero.d.ts +15 -0
  34. package/dist/page-schema/sections/Hero.d.ts.map +1 -0
  35. package/dist/page-schema/sections/Hero.js +44 -0
  36. package/dist/page-schema/sections/Hero.js.map +1 -0
  37. package/dist/page-schema/sections/Image.d.ts +10 -0
  38. package/dist/page-schema/sections/Image.d.ts.map +1 -0
  39. package/dist/page-schema/sections/Image.js +12 -0
  40. package/dist/page-schema/sections/Image.js.map +1 -0
  41. package/dist/page-schema/sections/ProductGrid.d.ts +29 -0
  42. package/dist/page-schema/sections/ProductGrid.d.ts.map +1 -0
  43. package/dist/page-schema/sections/ProductGrid.js +28 -0
  44. package/dist/page-schema/sections/ProductGrid.js.map +1 -0
  45. package/dist/page-schema/sections/RichText.d.ts +13 -0
  46. package/dist/page-schema/sections/RichText.d.ts.map +1 -0
  47. package/dist/page-schema/sections/RichText.js +16 -0
  48. package/dist/page-schema/sections/RichText.js.map +1 -0
  49. package/dist/page-schema/sections/Spacer.d.ts +10 -0
  50. package/dist/page-schema/sections/Spacer.d.ts.map +1 -0
  51. package/dist/page-schema/sections/Spacer.js +6 -0
  52. package/dist/page-schema/sections/Spacer.js.map +1 -0
  53. package/dist/page-schema/types.d.ts +138 -0
  54. package/dist/page-schema/types.d.ts.map +1 -0
  55. package/dist/page-schema/types.js +129 -0
  56. package/dist/page-schema/types.js.map +1 -0
  57. package/dist/react.d.ts +7 -0
  58. package/dist/react.d.ts.map +1 -1
  59. package/dist/react.js +15 -0
  60. package/dist/react.js.map +1 -1
  61. package/package.json +1 -1
  62. package/src/analytics/index.ts +27 -0
  63. package/src/analytics/queue.ts +224 -0
  64. package/src/analytics/react.tsx +146 -0
  65. package/src/analytics/types.ts +81 -0
  66. package/src/components/AddToCartButton.tsx +18 -0
  67. package/src/page-schema/PageRenderer.tsx +147 -0
  68. package/src/page-schema/index.ts +48 -0
  69. package/src/page-schema/sections/Banner.tsx +63 -0
  70. package/src/page-schema/sections/Hero.tsx +92 -0
  71. package/src/page-schema/sections/Image.tsx +42 -0
  72. package/src/page-schema/sections/ProductGrid.tsx +96 -0
  73. package/src/page-schema/sections/RichText.tsx +49 -0
  74. package/src/page-schema/sections/Spacer.tsx +15 -0
  75. package/src/page-schema/types.ts +286 -0
  76. package/src/react.tsx +59 -0
@@ -0,0 +1,129 @@
1
+ /**
2
+ * Page Schema — 装修页面的结构化数据模型
3
+ *
4
+ * 一个 Page 是若干 Section 的有序数组。每个 Section 是 (type, data) 的结构化值,
5
+ * 不是任意 React 代码。这样 agent / 编辑器 / 渲染器都能精准操作。
6
+ *
7
+ * 设计原则:
8
+ * 1. 强 schema:每个 type 的 data 字段固定,agent 不能"乱发挥"
9
+ * 2. stable id:每个 section 有唯一 id,编辑器选中、agent 引用都靠它
10
+ * 3. 可演进:新加 section type 只需扩 SectionType 联合 + 实现对应 React 组件
11
+ * 4. 反规范化:data 是平铺字段而不是嵌套结构,方便 agent 修改单个属性
12
+ */
13
+ // ============================================================
14
+ // Helper:生成 section ID
15
+ // ============================================================
16
+ export function newSectionId(type) {
17
+ const rand = Math.random().toString(36).slice(2, 8);
18
+ return `sec_${type.replace('-', '_')}_${rand}`;
19
+ }
20
+ // ============================================================
21
+ // 默认 data —— 商家新增某 type 的 section 时用作初始值
22
+ // ============================================================
23
+ export const DEFAULT_DATA = {
24
+ hero: {
25
+ headline: '欢迎来到我们的店铺',
26
+ subheadline: '精选商品 全场包邮',
27
+ height: 'medium',
28
+ text_align: 'center',
29
+ background_color: '#0f172a',
30
+ text_color: '#ffffff',
31
+ cta_label: '立刻购买',
32
+ cta_link: '/products',
33
+ },
34
+ 'product-grid': {
35
+ title: '精选商品',
36
+ selection_mode: 'newest',
37
+ limit: 6,
38
+ columns: 3,
39
+ },
40
+ banner: {
41
+ text: '满 200 减 20,限时优惠',
42
+ background_color: '#f97316',
43
+ text_color: '#ffffff',
44
+ },
45
+ 'rich-text': {
46
+ body: '在这里写一段品牌故事或说明文字。',
47
+ format: 'markdown',
48
+ max_width: 'normal',
49
+ },
50
+ image: {
51
+ url: 'https://placehold.co/1200x400',
52
+ alt: '广告图',
53
+ width: 'full',
54
+ },
55
+ spacer: {
56
+ height: 40,
57
+ },
58
+ };
59
+ /** 创建一个新 section(带默认 data) */
60
+ export function createSection(type) {
61
+ return {
62
+ id: newSectionId(type),
63
+ type,
64
+ data: { ...DEFAULT_DATA[type] },
65
+ visible: true,
66
+ };
67
+ }
68
+ /** 创建一个空白 page */
69
+ export function createEmptyPage(slug) {
70
+ return {
71
+ slug,
72
+ sections: [],
73
+ schema_version: 1,
74
+ };
75
+ }
76
+ // ============================================================
77
+ // 校验
78
+ // ============================================================
79
+ const REQUIRED_FIELDS = {
80
+ hero: ['headline'],
81
+ 'product-grid': ['selection_mode'],
82
+ banner: ['text'],
83
+ 'rich-text': ['body'],
84
+ image: ['url'],
85
+ spacer: ['height'],
86
+ };
87
+ /** 校验一个 section 数据完整性。返回错误列表,空数组表示合法 */
88
+ export function validateSection(section) {
89
+ const errs = [];
90
+ if (!section.id)
91
+ errs.push('id_required');
92
+ if (!section.type)
93
+ errs.push('type_required');
94
+ const required = REQUIRED_FIELDS[section.type];
95
+ if (!required) {
96
+ errs.push(`unknown_section_type:${section.type}`);
97
+ return errs;
98
+ }
99
+ const data = section.data;
100
+ for (const f of required) {
101
+ if (data[f] == null || data[f] === '') {
102
+ errs.push(`missing_field:${section.type}.${f}`);
103
+ }
104
+ }
105
+ return errs;
106
+ }
107
+ /** 校验整个 page */
108
+ export function validatePage(page) {
109
+ const errs = [];
110
+ if (!page.slug)
111
+ errs.push('slug_required');
112
+ if (page.schema_version !== 1)
113
+ errs.push(`unknown_schema_version:${page.schema_version}`);
114
+ if (!Array.isArray(page.sections)) {
115
+ errs.push('sections_must_be_array');
116
+ return errs;
117
+ }
118
+ const seenIds = new Set();
119
+ for (let i = 0; i < page.sections.length; i++) {
120
+ const s = page.sections[i];
121
+ if (seenIds.has(s.id))
122
+ errs.push(`duplicate_section_id:${s.id}`);
123
+ seenIds.add(s.id);
124
+ const secErrs = validateSection(s);
125
+ errs.push(...secErrs.map((e) => `section[${i}]:${e}`));
126
+ }
127
+ return errs;
128
+ }
129
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/page-schema/types.ts"],"names":[],"mappings":"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"}
package/dist/react.d.ts CHANGED
@@ -95,4 +95,11 @@ export interface ReactRendererOptions {
95
95
  onError?: (err: unknown, request: Request) => Response | Promise<Response>;
96
96
  }
97
97
  export declare function createReactRenderer(opts: ReactRendererOptions): (request: Request, env: any, executionContext: ExecutionContext) => Promise<Response>;
98
+ /**
99
+ * 把 head + react stream + tail 拼成一个 ReadableStream<Uint8Array>
100
+ */
101
+ export { PageRenderer, HeroSection, ProductGridSection, BannerSection, RichTextSection, ImageSection, SpacerSection, createSection, createEmptyPage, validateSection, validatePage, DEFAULT_DATA, newSectionId, } from './page-schema';
102
+ export type { Page, PageGlobalStyle, AnySection, SectionType, HeroData, ProductGridData, BannerData, RichTextData, ImageData, SpacerData, PageRendererProps, } from './page-schema';
103
+ export { TrackingProvider, useTracking, AnalyticsQueue, } from './analytics';
104
+ export type { TrackingProviderProps, AnalyticsEvent, AnalyticsConfig, EventType as AnalyticsEventType, TrackFn, PageViewProps, ProductViewProps, AddToCartProps, CheckoutStartProps, } from './analytics';
98
105
  //# sourceMappingURL=react.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"react.d.ts","sourceRoot":"","sources":["../src/react.tsx"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AAEH,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAO/B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAQ7C,wBAAgB,qBAAqB,CAAC,EACpC,KAAK,EACL,QAAQ,GACT,EAAE;IACD,KAAK,EAAE,aAAa,CAAC;IACrB,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAC;CAC3B,6EAMA;AAED;;;GAGG;AACH,wBAAgB,gBAAgB,IAAI,aAAa,CAQhD;AAMD,MAAM,WAAW,oBAAoB;IACnC;;;;OAIG;IACH,GAAG,EAAE,CAAC,KAAK,EAAE;QACX,GAAG,EAAE,aAAa,CAAC;QACnB,GAAG,EAAE,GAAG,CAAC;QACT,OAAO,EAAE,OAAO,CAAC;KAClB,KAAK,KAAK,CAAC,YAAY,GAAG,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;IAEvD;;;;;OAKG;IACH,YAAY,CAAC,EAAE,MAAM,CAAC;IAEtB;;OAEG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB;;OAEG;IACH,KAAK,CAAC,EAAE,MAAM,CAAC;IAEf;;OAEG;IACH,QAAQ,CAAC,EAAE,MAAM,CAAC;IAElB;;;;OAIG;IACH,UAAU,CAAC,EAAE;QACX,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,iBAAiB,CAAC,EAAE,MAAM,CAAC;QAC3B,kBAAkB,CAAC,EAAE,MAAM,CAAC;QAC5B,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB,CAAC;IAEF;;OAEG;IACH,IAAI,CAAC,EAAE;QACL,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,CAAC;IAEF;;;OAGG;IACH,SAAS,CAAC,EAAE,OAAO,CAAC;IAEpB;;OAEG;IACH,OAAO,CAAC,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,KAAK,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;CAC5E;AAED,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,oBAAoB,IAE1D,SAAS,OAAO,EAChB,KAAK,GAAG,EACR,kBAAkB,gBAAgB,KACjC,OAAO,CAAC,QAAQ,CAAC,CAkHrB"}
1
+ {"version":3,"file":"react.d.ts","sourceRoot":"","sources":["../src/react.tsx"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AAEH,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAO/B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAQ7C,wBAAgB,qBAAqB,CAAC,EACpC,KAAK,EACL,QAAQ,GACT,EAAE;IACD,KAAK,EAAE,aAAa,CAAC;IACrB,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAC;CAC3B,6EAMA;AAED;;;GAGG;AACH,wBAAgB,gBAAgB,IAAI,aAAa,CAQhD;AAMD,MAAM,WAAW,oBAAoB;IACnC;;;;OAIG;IACH,GAAG,EAAE,CAAC,KAAK,EAAE;QACX,GAAG,EAAE,aAAa,CAAC;QACnB,GAAG,EAAE,GAAG,CAAC;QACT,OAAO,EAAE,OAAO,CAAC;KAClB,KAAK,KAAK,CAAC,YAAY,GAAG,OAAO,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;IAEvD;;;;;OAKG;IACH,YAAY,CAAC,EAAE,MAAM,CAAC;IAEtB;;OAEG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB;;OAEG;IACH,KAAK,CAAC,EAAE,MAAM,CAAC;IAEf;;OAEG;IACH,QAAQ,CAAC,EAAE,MAAM,CAAC;IAElB;;;;OAIG;IACH,UAAU,CAAC,EAAE;QACX,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,iBAAiB,CAAC,EAAE,MAAM,CAAC;QAC3B,kBAAkB,CAAC,EAAE,MAAM,CAAC;QAC5B,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB,CAAC;IAEF;;OAEG;IACH,IAAI,CAAC,EAAE;QACL,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,CAAC;IAEF;;;OAGG;IACH,SAAS,CAAC,EAAE,OAAO,CAAC;IAEpB;;OAEG;IACH,OAAO,CAAC,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,OAAO,EAAE,OAAO,KAAK,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;CAC5E;AAED,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,oBAAoB,IAE1D,SAAS,OAAO,EAChB,KAAK,GAAG,EACR,kBAAkB,gBAAgB,KACjC,OAAO,CAAC,QAAQ,CAAC,CAkHrB;AAYD;;GAEG;AAKH,OAAO,EACL,YAAY,EACZ,WAAW,EACX,kBAAkB,EAClB,aAAa,EACb,eAAe,EACf,YAAY,EACZ,aAAa,EACb,aAAa,EACb,eAAe,EACf,eAAe,EACf,YAAY,EACZ,YAAY,EACZ,YAAY,GACb,MAAM,eAAe,CAAC;AACvB,YAAY,EACV,IAAI,EACJ,eAAe,EACf,UAAU,EACV,WAAW,EACX,QAAQ,EACR,eAAe,EACf,UAAU,EACV,YAAY,EACZ,SAAS,EACT,UAAU,EACV,iBAAiB,GAClB,MAAM,eAAe,CAAC;AAWvB,OAAO,EACL,gBAAgB,EAChB,WAAW,EACX,cAAc,GACf,MAAM,aAAa,CAAC;AACrB,YAAY,EACV,qBAAqB,EACrB,cAAc,EACd,eAAe,EACf,SAAS,IAAI,kBAAkB,EAC/B,OAAO,EACP,aAAa,EACb,gBAAgB,EAChB,cAAc,EACd,kBAAkB,GACnB,MAAM,aAAa,CAAC"}
package/dist/react.js CHANGED
@@ -148,6 +148,21 @@ function escapeHtml(s) {
148
148
  /**
149
149
  * 把 head + react stream + tail 拼成一个 ReadableStream<Uint8Array>
150
150
  */
151
+ // ============================================================
152
+ // Page Schema re-export
153
+ // ============================================================
154
+ // 结构化页面渲染。Admin 编辑器与 storefront 共用同一份组件。
155
+ export { PageRenderer, HeroSection, ProductGridSection, BannerSection, RichTextSection, ImageSection, SpacerSection, createSection, createEmptyPage, validateSection, validatePage, DEFAULT_DATA, newSectionId, } from './page-schema';
156
+ // ============================================================
157
+ // Tracking SDK re-export
158
+ // ============================================================
159
+ // 把 5 个核心事件(page_view / product_view / add_to_cart / checkout_start / checkout_paid)
160
+ // 批量发到 platform-api 的 /api/events。
161
+ //
162
+ // 与 components/AnalyticsProvider 解耦:
163
+ // - AnalyticsProvider 是商家用的事件总线(自定义 reporter)
164
+ // - TrackingProvider 是平台的标准埋点(发到 platform-api)
165
+ export { TrackingProvider, useTracking, AnalyticsQueue, } from './analytics';
151
166
  function composeStream(head, body, tail) {
152
167
  const encoder = new TextEncoder();
153
168
  return new ReadableStream({
package/dist/react.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"react.js","sourceRoot":"","sources":["../src/react.tsx"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"react.js","sourceRoot":"","sources":["../src/react.tsx"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@shopbb/helium",
3
- "version": "0.7.7",
3
+ "version": "0.9.0",
4
4
  "description": "shopbb storefront framework — components, React SSR, GraphQL client, cart handler, cache for Cloudflare Workers",
5
5
  "license": "MIT",
6
6
  "type": "module",
@@ -0,0 +1,27 @@
1
+ /**
2
+ * @shopbb/helium/analytics — 埋点 SDK 公开 API
3
+ *
4
+ * 浏览器侧用法:
5
+ * - <TrackingProvider>:包在 root,初始化 session、起 flush 定时器、自动 page_view
6
+ * - useTracking():在组件内拿到 track() 函数
7
+ *
8
+ * 服务端不要用本模块——服务端事件(checkout_paid)由 platform-api 直接 INSERT。
9
+ *
10
+ * 注意:本模块与 components/AnalyticsProvider 是两个独立产品:
11
+ * - AnalyticsProvider(components):客户端事件总线,挂 GA / 自定义 reporter
12
+ * - TrackingProvider(本模块):把 5 个核心事件批量发到 /api/events,给 agent 用
13
+ */
14
+
15
+ export { TrackingProvider, useTracking } from './react';
16
+ export type { TrackingProviderProps } from './react';
17
+ export { AnalyticsQueue } from './queue';
18
+ export type {
19
+ AnalyticsEvent,
20
+ AnalyticsConfig,
21
+ EventType,
22
+ PageViewProps,
23
+ ProductViewProps,
24
+ AddToCartProps,
25
+ CheckoutStartProps,
26
+ TrackFn,
27
+ } from './types';
@@ -0,0 +1,224 @@
1
+ /**
2
+ * AnalyticsQueue — 浏览器侧事件缓冲队列
3
+ *
4
+ * 职责:
5
+ * 1. 把 track() 调用收集到内存队列
6
+ * 2. 定时(3s)或满批(20)触发 flush
7
+ * 3. flush 调 POST /api/events,失败重试 1 次
8
+ * 4. beforeunload 时用 navigator.sendBeacon 兜底刷出
9
+ *
10
+ * 注意:这是浏览器侧逻辑。SSR 阶段调 track() 会被丢弃(typeof window === 'undefined' 检查)。
11
+ */
12
+
13
+ import type { AnalyticsConfig, AnalyticsEvent, QueuedEvent } from './types';
14
+
15
+ const SESSION_COOKIE = 'sbb_sid';
16
+ const SESSION_TTL_DAYS = 365;
17
+
18
+ /**
19
+ * 读 sbb_sid cookie,没有则生成一个新的并 set。
20
+ * 仅浏览器环境调用。
21
+ */
22
+ function getOrCreateSessionId(): string {
23
+ if (typeof document === 'undefined') return '';
24
+
25
+ const match = document.cookie.match(/(?:^|;\s*)sbb_sid=([^;]+)/);
26
+ if (match) return match[1];
27
+
28
+ // 生成 UUID(不依赖 crypto.randomUUID 以兼容老浏览器)
29
+ const sid = crypto?.randomUUID?.() ?? generateFallbackUuid();
30
+
31
+ const maxAge = SESSION_TTL_DAYS * 24 * 60 * 60;
32
+ document.cookie = `${SESSION_COOKIE}=${sid}; max-age=${maxAge}; path=/; samesite=lax`;
33
+ return sid;
34
+ }
35
+
36
+ function generateFallbackUuid(): string {
37
+ // RFC4122 v4 简化版
38
+ return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
39
+ const r = (Math.random() * 16) | 0;
40
+ const v = c === 'x' ? r : (r & 0x3) | 0x8;
41
+ return v.toString(16);
42
+ });
43
+ }
44
+
45
+ export class AnalyticsQueue {
46
+ private readonly config: Required<Omit<AnalyticsConfig, 'buyerId' | 'debug'>> & {
47
+ buyerId: string | null;
48
+ debug: boolean;
49
+ };
50
+ private queue: QueuedEvent[] = [];
51
+ private timer: ReturnType<typeof setInterval> | null = null;
52
+ private sessionId = '';
53
+ private flushing = false;
54
+
55
+ constructor(config: AnalyticsConfig) {
56
+ this.config = {
57
+ apiBase: config.apiBase.replace(/\/+$/, ''),
58
+ publicAccessToken: config.publicAccessToken,
59
+ buyerId: config.buyerId ?? null,
60
+ debug: !!config.debug,
61
+ flushIntervalMs: config.flushIntervalMs ?? 3000,
62
+ flushBatchSize: config.flushBatchSize ?? 20,
63
+ };
64
+ }
65
+
66
+ start(): void {
67
+ if (typeof window === 'undefined') return;
68
+ this.sessionId = getOrCreateSessionId();
69
+
70
+ if (this.timer) return;
71
+ this.timer = setInterval(() => {
72
+ void this.flush();
73
+ }, this.config.flushIntervalMs);
74
+
75
+ // 页面卸载兜底
76
+ window.addEventListener('pagehide', () => this.flushBeacon());
77
+ window.addEventListener('beforeunload', () => this.flushBeacon());
78
+ if (typeof document !== 'undefined') {
79
+ document.addEventListener('visibilitychange', () => {
80
+ if (document.visibilityState === 'hidden') {
81
+ this.flushBeacon();
82
+ }
83
+ });
84
+ }
85
+ }
86
+
87
+ stop(): void {
88
+ if (this.timer) {
89
+ clearInterval(this.timer);
90
+ this.timer = null;
91
+ }
92
+ }
93
+
94
+ setBuyerId(id: string | null): void {
95
+ this.config.buyerId = id;
96
+ }
97
+
98
+ track(event: AnalyticsEvent): void {
99
+ if (typeof window === 'undefined') {
100
+ this.log('skip-ssr', event.type);
101
+ return;
102
+ }
103
+
104
+ if (!this.sessionId) {
105
+ this.sessionId = getOrCreateSessionId();
106
+ }
107
+
108
+ const queued: QueuedEvent = {
109
+ type: event.type,
110
+ occurredAt: event.occurredAt ?? Math.floor(Date.now() / 1000),
111
+ path:
112
+ event.path !== undefined
113
+ ? event.path
114
+ : typeof location !== 'undefined'
115
+ ? location.pathname + location.search
116
+ : null,
117
+ referrer:
118
+ event.referrer !== undefined
119
+ ? event.referrer
120
+ : typeof document !== 'undefined'
121
+ ? document.referrer || null
122
+ : null,
123
+ props: event.props ?? {},
124
+ };
125
+
126
+ this.queue.push(queued);
127
+ this.log('queued', queued.type, this.queue.length);
128
+
129
+ if (this.queue.length >= this.config.flushBatchSize) {
130
+ void this.flush();
131
+ }
132
+ }
133
+
134
+ /** 主动 flush。返回是否实际发送了请求 */
135
+ async flush(): Promise<boolean> {
136
+ if (this.flushing) return false;
137
+ if (this.queue.length === 0) return false;
138
+ if (!this.config.publicAccessToken) {
139
+ this.log('skip-no-token');
140
+ return false;
141
+ }
142
+
143
+ this.flushing = true;
144
+ const batch = this.queue.splice(0, this.config.flushBatchSize);
145
+ const body = JSON.stringify({
146
+ events: batch.map((e) => ({
147
+ type: e.type,
148
+ occurredAt: e.occurredAt,
149
+ path: e.path ?? undefined,
150
+ referrer: e.referrer ?? undefined,
151
+ props: e.props,
152
+ })),
153
+ });
154
+
155
+ try {
156
+ const res = await fetch(`${this.config.apiBase}/api/events`, {
157
+ method: 'POST',
158
+ headers: this.headers(),
159
+ body,
160
+ keepalive: true,
161
+ });
162
+ if (!res.ok) {
163
+ this.log('flush-failed-status', res.status);
164
+ // 失败不重试入队(避免堆积),demo 阶段简化
165
+ return false;
166
+ }
167
+ this.log('flushed', batch.length);
168
+ return true;
169
+ } catch (err) {
170
+ this.log('flush-error', err);
171
+ return false;
172
+ } finally {
173
+ this.flushing = false;
174
+ }
175
+ }
176
+
177
+ /** 用 sendBeacon 兜底,页面卸载场景 */
178
+ private flushBeacon(): void {
179
+ if (this.queue.length === 0) return;
180
+ if (typeof navigator === 'undefined' || !navigator.sendBeacon) return;
181
+ if (!this.config.publicAccessToken) return;
182
+
183
+ const batch = this.queue.splice(0, this.config.flushBatchSize);
184
+ const body = JSON.stringify({
185
+ events: batch.map((e) => ({
186
+ type: e.type,
187
+ occurredAt: e.occurredAt,
188
+ path: e.path ?? undefined,
189
+ referrer: e.referrer ?? undefined,
190
+ props: e.props,
191
+ })),
192
+ // sendBeacon 无法设 header,所以把 token / session 塞进 body 作为兜底
193
+ // platform-api handler 优先认 header;body 字段用于 beacon 场景
194
+ _auth: {
195
+ token: this.config.publicAccessToken,
196
+ sessionId: this.sessionId,
197
+ buyerId: this.config.buyerId,
198
+ },
199
+ });
200
+
201
+ const blob = new Blob([body], { type: 'application/json' });
202
+ const ok = navigator.sendBeacon(`${this.config.apiBase}/api/events?beacon=1`, blob);
203
+ this.log('beacon', ok, batch.length);
204
+ }
205
+
206
+ private headers(): HeadersInit {
207
+ const h: Record<string, string> = {
208
+ 'Content-Type': 'application/json',
209
+ 'X-Storefront-Access-Token': this.config.publicAccessToken,
210
+ 'X-Session-Id': this.sessionId,
211
+ };
212
+ if (this.config.buyerId) {
213
+ h['X-Buyer-Id'] = this.config.buyerId;
214
+ }
215
+ return h;
216
+ }
217
+
218
+ private log(...args: unknown[]): void {
219
+ if (this.config.debug) {
220
+ // eslint-disable-next-line no-console
221
+ console.log('[helium-analytics]', ...args);
222
+ }
223
+ }
224
+ }
@@ -0,0 +1,146 @@
1
+ /**
2
+ * TrackingProvider + useTracking — React 绑定(后端事件埋点)
3
+ *
4
+ * 与 components/AnalyticsProvider 不同:
5
+ * - AnalyticsProvider(旧):客户端事件总线,商家挂 GA / Plausible 等 reporter
6
+ * - TrackingProvider(新):把 5 个核心事件发到 platform-api /api/events
7
+ *
8
+ * 用法:
9
+ * <TrackingProvider
10
+ * apiBase={store.apiBase}
11
+ * publicAccessToken={store.publicAccessToken}
12
+ * buyerId={buyer?.id}
13
+ * debug
14
+ * >
15
+ * <App />
16
+ * </TrackingProvider>
17
+ *
18
+ * const { track } = useTracking();
19
+ * track({ type: 'add_to_cart', props: { productId, quantity: 1 } });
20
+ *
21
+ * 自动事件:
22
+ * - 首次挂载发一次 page_view
23
+ * - 监听 history pushState/replaceState/popstate,每次 URL 变化补发 page_view
24
+ */
25
+
26
+ import * as React from 'react';
27
+ import { AnalyticsQueue } from './queue';
28
+ import type { AnalyticsEvent, AnalyticsConfig, TrackFn } from './types';
29
+
30
+ interface ContextValue {
31
+ track: TrackFn;
32
+ setBuyerId: (id: string | null) => void;
33
+ }
34
+
35
+ const TrackingCtx = React.createContext<ContextValue | null>(null);
36
+
37
+ export interface TrackingProviderProps extends AnalyticsConfig {
38
+ children: React.ReactNode;
39
+ /** 是否自动发 page_view(默认 true) */
40
+ autoPageview?: boolean;
41
+ }
42
+
43
+ export function TrackingProvider(props: TrackingProviderProps): React.ReactElement {
44
+ const {
45
+ children,
46
+ autoPageview = true,
47
+ apiBase,
48
+ publicAccessToken,
49
+ buyerId,
50
+ debug,
51
+ flushIntervalMs,
52
+ flushBatchSize,
53
+ } = props;
54
+
55
+ // 用 ref 持有 queue,避免每次 re-render 重建
56
+ const queueRef = React.useRef<AnalyticsQueue | null>(null);
57
+
58
+ if (queueRef.current === null) {
59
+ queueRef.current = new AnalyticsQueue({
60
+ apiBase,
61
+ publicAccessToken,
62
+ buyerId,
63
+ debug,
64
+ flushIntervalMs,
65
+ flushBatchSize,
66
+ });
67
+ }
68
+
69
+ // buyerId 变化时同步到 queue
70
+ React.useEffect(() => {
71
+ queueRef.current?.setBuyerId(buyerId ?? null);
72
+ }, [buyerId]);
73
+
74
+ // 启动 queue(只在浏览器侧)
75
+ React.useEffect(() => {
76
+ if (typeof window === 'undefined') return;
77
+ const q = queueRef.current;
78
+ if (!q) return;
79
+ q.start();
80
+ return () => q.stop();
81
+ }, []);
82
+
83
+ // 自动 page_view
84
+ React.useEffect(() => {
85
+ if (!autoPageview) return;
86
+ if (typeof window === 'undefined') return;
87
+ const q = queueRef.current;
88
+ if (!q) return;
89
+
90
+ const send = () => {
91
+ q.track({
92
+ type: 'page_view',
93
+ props: { title: typeof document !== 'undefined' ? document.title : undefined },
94
+ });
95
+ };
96
+
97
+ // 首次
98
+ send();
99
+
100
+ // patch history methods 以捕获 SPA 导航
101
+ const origPush = history.pushState;
102
+ const origReplace = history.replaceState;
103
+ history.pushState = function (...args) {
104
+ origPush.apply(this, args as any);
105
+ send();
106
+ };
107
+ history.replaceState = function (...args) {
108
+ origReplace.apply(this, args as any);
109
+ send();
110
+ };
111
+ const onPop = () => send();
112
+ window.addEventListener('popstate', onPop);
113
+
114
+ return () => {
115
+ history.pushState = origPush;
116
+ history.replaceState = origReplace;
117
+ window.removeEventListener('popstate', onPop);
118
+ };
119
+ }, [autoPageview]);
120
+
121
+ const value = React.useMemo<ContextValue>(
122
+ () => ({
123
+ track: (e: AnalyticsEvent) => queueRef.current?.track(e),
124
+ setBuyerId: (id) => queueRef.current?.setBuyerId(id),
125
+ }),
126
+ [],
127
+ );
128
+
129
+ return <TrackingCtx.Provider value={value}>{children}</TrackingCtx.Provider>;
130
+ }
131
+
132
+ /**
133
+ * 在组件里拿到 track 函数。
134
+ *
135
+ * 在 TrackingProvider 之外调用会得到一个 no-op,不报错(方便在测试/SSR 中使用)。
136
+ */
137
+ export function useTracking(): ContextValue {
138
+ const ctx = React.useContext(TrackingCtx);
139
+ if (ctx) return ctx;
140
+ return NOOP;
141
+ }
142
+
143
+ const NOOP: ContextValue = {
144
+ track: () => {},
145
+ setBuyerId: () => {},
146
+ };