@shopbb/helium 0.8.0 → 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 (51) hide show
  1. package/dist/page-schema/PageRenderer.d.ts +29 -0
  2. package/dist/page-schema/PageRenderer.d.ts.map +1 -0
  3. package/dist/page-schema/PageRenderer.js +73 -0
  4. package/dist/page-schema/PageRenderer.js.map +1 -0
  5. package/dist/page-schema/index.d.ts +20 -0
  6. package/dist/page-schema/index.d.ts.map +1 -0
  7. package/dist/page-schema/index.js +18 -0
  8. package/dist/page-schema/index.js.map +1 -0
  9. package/dist/page-schema/sections/Banner.d.ts +10 -0
  10. package/dist/page-schema/sections/Banner.d.ts.map +1 -0
  11. package/dist/page-schema/sections/Banner.js +35 -0
  12. package/dist/page-schema/sections/Banner.js.map +1 -0
  13. package/dist/page-schema/sections/Hero.d.ts +15 -0
  14. package/dist/page-schema/sections/Hero.d.ts.map +1 -0
  15. package/dist/page-schema/sections/Hero.js +44 -0
  16. package/dist/page-schema/sections/Hero.js.map +1 -0
  17. package/dist/page-schema/sections/Image.d.ts +10 -0
  18. package/dist/page-schema/sections/Image.d.ts.map +1 -0
  19. package/dist/page-schema/sections/Image.js +12 -0
  20. package/dist/page-schema/sections/Image.js.map +1 -0
  21. package/dist/page-schema/sections/ProductGrid.d.ts +29 -0
  22. package/dist/page-schema/sections/ProductGrid.d.ts.map +1 -0
  23. package/dist/page-schema/sections/ProductGrid.js +28 -0
  24. package/dist/page-schema/sections/ProductGrid.js.map +1 -0
  25. package/dist/page-schema/sections/RichText.d.ts +13 -0
  26. package/dist/page-schema/sections/RichText.d.ts.map +1 -0
  27. package/dist/page-schema/sections/RichText.js +16 -0
  28. package/dist/page-schema/sections/RichText.js.map +1 -0
  29. package/dist/page-schema/sections/Spacer.d.ts +10 -0
  30. package/dist/page-schema/sections/Spacer.d.ts.map +1 -0
  31. package/dist/page-schema/sections/Spacer.js +6 -0
  32. package/dist/page-schema/sections/Spacer.js.map +1 -0
  33. package/dist/page-schema/types.d.ts +138 -0
  34. package/dist/page-schema/types.d.ts.map +1 -0
  35. package/dist/page-schema/types.js +129 -0
  36. package/dist/page-schema/types.js.map +1 -0
  37. package/dist/react.d.ts +2 -0
  38. package/dist/react.d.ts.map +1 -1
  39. package/dist/react.js +5 -0
  40. package/dist/react.js.map +1 -1
  41. package/package.json +1 -1
  42. package/src/page-schema/PageRenderer.tsx +147 -0
  43. package/src/page-schema/index.ts +48 -0
  44. package/src/page-schema/sections/Banner.tsx +63 -0
  45. package/src/page-schema/sections/Hero.tsx +92 -0
  46. package/src/page-schema/sections/Image.tsx +42 -0
  47. package/src/page-schema/sections/ProductGrid.tsx +96 -0
  48. package/src/page-schema/sections/RichText.tsx +49 -0
  49. package/src/page-schema/sections/Spacer.tsx +15 -0
  50. package/src/page-schema/types.ts +286 -0
  51. package/src/react.tsx +33 -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":"AAAA;;;;;;;;;;;GAWG;AA2JH,+DAA+D;AAC/D,uBAAuB;AACvB,+DAA+D;AAE/D,MAAM,UAAU,YAAY,CAAC,IAAiB;IAC5C,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;IACpD,OAAO,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC;AACjD,CAAC;AAED,+DAA+D;AAC/D,yCAAyC;AACzC,+DAA+D;AAE/D,MAAM,CAAC,MAAM,YAAY,GAA+C;IACtE,IAAI,EAAE;QACJ,QAAQ,EAAE,WAAW;QACrB,WAAW,EAAE,WAAW;QACxB,MAAM,EAAE,QAAQ;QAChB,UAAU,EAAE,QAAQ;QACpB,gBAAgB,EAAE,SAAS;QAC3B,UAAU,EAAE,SAAS;QACrB,SAAS,EAAE,MAAM;QACjB,QAAQ,EAAE,WAAW;KACV;IACb,cAAc,EAAE;QACd,KAAK,EAAE,MAAM;QACb,cAAc,EAAE,QAAQ;QACxB,KAAK,EAAE,CAAC;QACR,OAAO,EAAE,CAAC;KACQ;IACpB,MAAM,EAAE;QACN,IAAI,EAAE,iBAAiB;QACvB,gBAAgB,EAAE,SAAS;QAC3B,UAAU,EAAE,SAAS;KACR;IACf,WAAW,EAAE;QACX,IAAI,EAAE,kBAAkB;QACxB,MAAM,EAAE,UAAU;QAClB,SAAS,EAAE,QAAQ;KACJ;IACjB,KAAK,EAAE;QACL,GAAG,EAAE,+BAA+B;QACpC,GAAG,EAAE,KAAK;QACV,KAAK,EAAE,MAAM;KACD;IACd,MAAM,EAAE;QACN,MAAM,EAAE,EAAE;KACG;CAChB,CAAC;AAEF,8BAA8B;AAC9B,MAAM,UAAU,aAAa,CAAwB,IAAO;IAC1D,OAAO;QACL,EAAE,EAAE,YAAY,CAAC,IAAI,CAAC;QACtB,IAAI;QACJ,IAAI,EAAE,EAAE,GAAI,YAAY,CAAC,IAAI,CAAY,EAAS;QAClD,OAAO,EAAE,IAAI;KACA,CAAC;AAClB,CAAC;AAED,kBAAkB;AAClB,MAAM,UAAU,eAAe,CAAC,IAAY;IAC1C,OAAO;QACL,IAAI;QACJ,QAAQ,EAAE,EAAE;QACZ,cAAc,EAAE,CAAC;KAClB,CAAC;AACJ,CAAC;AAED,+DAA+D;AAC/D,KAAK;AACL,+DAA+D;AAE/D,MAAM,eAAe,GAAqC;IACxD,IAAI,EAAE,CAAC,UAAU,CAAC;IAClB,cAAc,EAAE,CAAC,gBAAgB,CAAC;IAClC,MAAM,EAAE,CAAC,MAAM,CAAC;IAChB,WAAW,EAAE,CAAC,MAAM,CAAC;IACrB,KAAK,EAAE,CAAC,KAAK,CAAC;IACd,MAAM,EAAE,CAAC,QAAQ,CAAC;CACnB,CAAC;AAEF,wCAAwC;AACxC,MAAM,UAAU,eAAe,CAAC,OAAmB;IACjD,MAAM,IAAI,GAAa,EAAE,CAAC;IAC1B,IAAI,CAAC,OAAO,CAAC,EAAE;QAAE,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IAC1C,IAAI,CAAC,OAAO,CAAC,IAAI;QAAE,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IAC9C,MAAM,QAAQ,GAAG,eAAe,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/C,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,IAAI,CAAC,IAAI,CAAC,wBAAwB,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC;QAClD,OAAO,IAAI,CAAC;IACd,CAAC;IACD,MAAM,IAAI,GAAG,OAAO,CAAC,IAA0C,CAAC;IAChE,KAAK,MAAM,CAAC,IAAI,QAAQ,EAAE,CAAC;QACzB,IAAI,IAAI,CAAC,CAAC,CAAC,IAAI,IAAI,IAAI,IAAI,CAAC,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC;YACtC,IAAI,CAAC,IAAI,CAAC,iBAAiB,OAAO,CAAC,IAAI,IAAI,CAAC,EAAE,CAAC,CAAC;QAClD,CAAC;IACH,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,gBAAgB;AAChB,MAAM,UAAU,YAAY,CAAC,IAAU;IACrC,MAAM,IAAI,GAAa,EAAE,CAAC;IAC1B,IAAI,CAAC,IAAI,CAAC,IAAI;QAAE,IAAI,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;IAC3C,IAAI,IAAI,CAAC,cAAc,KAAK,CAAC;QAAE,IAAI,CAAC,IAAI,CAAC,0BAA0B,IAAI,CAAC,cAAc,EAAE,CAAC,CAAC;IAC1F,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;QAClC,IAAI,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;QACpC,OAAO,IAAI,CAAC;IACd,CAAC;IACD,MAAM,OAAO,GAAG,IAAI,GAAG,EAAU,CAAC;IAClC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QAC9C,MAAM,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;QAC3B,IAAI,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;YAAE,IAAI,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QACjE,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QAClB,MAAM,OAAO,GAAG,eAAe,CAAC,CAAC,CAAC,CAAC;QACnC,IAAI,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;IACzD,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC"}
package/dist/react.d.ts CHANGED
@@ -98,6 +98,8 @@ export declare function createReactRenderer(opts: ReactRendererOptions): (reques
98
98
  /**
99
99
  * 把 head + react stream + tail 拼成一个 ReadableStream<Uint8Array>
100
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';
101
103
  export { TrackingProvider, useTracking, AnalyticsQueue, } from './analytics';
102
104
  export type { TrackingProviderProps, AnalyticsEvent, AnalyticsConfig, EventType as AnalyticsEventType, TrackFn, PageViewProps, ProductViewProps, AddToCartProps, CheckoutStartProps, } from './analytics';
103
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;AAYD;;GAEG;AAUH,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"}
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
@@ -149,6 +149,11 @@ function escapeHtml(s) {
149
149
  * 把 head + react stream + tail 拼成一个 ReadableStream<Uint8Array>
150
150
  */
151
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
+ // ============================================================
152
157
  // Tracking SDK re-export
153
158
  // ============================================================
154
159
  // 把 5 个核心事件(page_view / product_view / add_to_cart / checkout_start / checkout_paid)
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.8.0",
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,147 @@
1
+ /**
2
+ * <PageRenderer> — 按 Page Schema 渲染一整页
3
+ *
4
+ * 同一份组件在 production storefront 与 admin 编辑器画布共用。
5
+ *
6
+ * 商品网格的 selection_mode 自动模式(newest / top-selling / collection)
7
+ * 由父组件预先拉数据通过 productsBySection prop 注入,组件本身不发请求。
8
+ * 这让 renderer 保持纯组件、可在 SSR / CSR / 编辑器画布任何环境用。
9
+ */
10
+
11
+ import * as React from 'react';
12
+ import type { AnySection, Page } from './types';
13
+ import { HeroSection } from './sections/Hero';
14
+ import { ProductGridSection } from './sections/ProductGrid';
15
+ import { BannerSection } from './sections/Banner';
16
+ import { RichTextSection } from './sections/RichText';
17
+ import { ImageSection } from './sections/Image';
18
+ import { SpacerSection } from './sections/Spacer';
19
+
20
+ export interface PageRendererProps {
21
+ page: Page;
22
+ /**
23
+ * 按 section.id 提供该 section 用到的商品数据(仅 product-grid 用到)。
24
+ * Page renderer 不发请求,请父组件按 schema 计算后传入。
25
+ */
26
+ productsBySection?: Record<string, any[]>;
27
+ /**
28
+ * 编辑器模式下传入 onSectionClick / selectedSectionId,渲染时会给每个 section
29
+ * 加 hover / 选中外框,便于点选定位。
30
+ */
31
+ editorMode?: {
32
+ selectedSectionId?: string | null;
33
+ onSectionClick?: (sectionId: string) => void;
34
+ };
35
+ }
36
+
37
+ export function PageRenderer({
38
+ page,
39
+ productsBySection,
40
+ editorMode,
41
+ }: PageRendererProps): React.ReactElement {
42
+ const globalStyle: React.CSSProperties = {
43
+ backgroundColor: page.global_style?.background_color,
44
+ color: '#0f172a',
45
+ fontFamily:
46
+ page.global_style?.font_family === 'serif'
47
+ ? 'Georgia, "Times New Roman", serif'
48
+ : page.global_style?.font_family === 'sans-serif'
49
+ ? '-apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif'
50
+ : 'inherit',
51
+ };
52
+
53
+ return (
54
+ <div style={globalStyle} data-page-slug={page.slug}>
55
+ {page.sections
56
+ .filter((s) => s.visible !== false)
57
+ .map((section) => (
58
+ <SectionFrame
59
+ key={section.id}
60
+ section={section}
61
+ productsBySection={productsBySection}
62
+ editorMode={editorMode}
63
+ />
64
+ ))}
65
+ </div>
66
+ );
67
+ }
68
+
69
+ interface SectionFrameProps {
70
+ section: AnySection;
71
+ productsBySection?: Record<string, any[]>;
72
+ editorMode?: PageRendererProps['editorMode'];
73
+ }
74
+
75
+ function SectionFrame({ section, productsBySection, editorMode }: SectionFrameProps): React.ReactElement {
76
+ const isSelected = editorMode?.selectedSectionId === section.id;
77
+ const isEditor = !!editorMode;
78
+
79
+ const handleClick = (e: React.MouseEvent) => {
80
+ if (!isEditor) return;
81
+ // 阻止内部 a 链接跳转,让商家专注于选中
82
+ e.preventDefault();
83
+ e.stopPropagation();
84
+ editorMode?.onSectionClick?.(section.id);
85
+ };
86
+
87
+ const editorStyle: React.CSSProperties = isEditor
88
+ ? {
89
+ position: 'relative',
90
+ cursor: 'pointer',
91
+ outline: isSelected ? '2px solid #f97316' : '2px solid transparent',
92
+ outlineOffset: '-2px',
93
+ transition: 'outline-color 120ms ease',
94
+ }
95
+ : {};
96
+
97
+ const onMouseEnter = (e: React.MouseEvent<HTMLDivElement>) => {
98
+ if (!isEditor || isSelected) return;
99
+ (e.currentTarget as HTMLDivElement).style.outlineColor = '#fed7aa';
100
+ };
101
+ const onMouseLeave = (e: React.MouseEvent<HTMLDivElement>) => {
102
+ if (!isEditor || isSelected) return;
103
+ (e.currentTarget as HTMLDivElement).style.outlineColor = 'transparent';
104
+ };
105
+
106
+ return (
107
+ <div
108
+ data-section-id={section.id}
109
+ data-section-type={section.type}
110
+ style={editorStyle}
111
+ onClick={handleClick}
112
+ onMouseEnter={onMouseEnter}
113
+ onMouseLeave={onMouseLeave}
114
+ >
115
+ {renderSection(section, productsBySection)}
116
+ </div>
117
+ );
118
+ }
119
+
120
+ function renderSection(section: AnySection, productsBySection?: Record<string, any[]>) {
121
+ switch (section.type) {
122
+ case 'hero':
123
+ return <HeroSection data={section.data} />;
124
+ case 'product-grid':
125
+ return (
126
+ <ProductGridSection
127
+ data={section.data}
128
+ products={productsBySection?.[section.id]}
129
+ />
130
+ );
131
+ case 'banner':
132
+ return <BannerSection data={section.data} />;
133
+ case 'rich-text':
134
+ return <RichTextSection data={section.data} />;
135
+ case 'image':
136
+ return <ImageSection data={section.data} />;
137
+ case 'spacer':
138
+ return <SpacerSection data={section.data} />;
139
+ default:
140
+ // exhaustive check
141
+ return (
142
+ <div style={{ padding: 16, background: '#fef2f2', color: '#b91c1c' }}>
143
+ 未知 section 类型:{(section as any).type}
144
+ </div>
145
+ );
146
+ }
147
+ }
@@ -0,0 +1,48 @@
1
+ /**
2
+ * @shopbb/helium 的 Page Schema 模块
3
+ *
4
+ * 公开 API:
5
+ * - types:Page / Section / 各 section 的 data 类型
6
+ * - <PageRenderer>:按 schema 渲染一整页
7
+ * - 6 个 section 组件(也可以单独引用)
8
+ * - 工具:createSection / createEmptyPage / validatePage / DEFAULT_DATA
9
+ */
10
+
11
+ export type {
12
+ Page,
13
+ PageGlobalStyle,
14
+ Section,
15
+ AnySection,
16
+ SectionType,
17
+ HeroData,
18
+ HeroSection as HeroSectionType,
19
+ ProductGridData,
20
+ ProductGridSection as ProductGridSectionType,
21
+ BannerData,
22
+ BannerSection as BannerSectionType,
23
+ RichTextData,
24
+ RichTextSection as RichTextSectionType,
25
+ ImageData,
26
+ ImageSection as ImageSectionType,
27
+ SpacerData,
28
+ SpacerSection as SpacerSectionType,
29
+ } from './types';
30
+
31
+ export {
32
+ newSectionId,
33
+ createSection,
34
+ createEmptyPage,
35
+ validateSection,
36
+ validatePage,
37
+ DEFAULT_DATA,
38
+ } from './types';
39
+
40
+ export { PageRenderer } from './PageRenderer';
41
+ export type { PageRendererProps } from './PageRenderer';
42
+
43
+ export { HeroSection } from './sections/Hero';
44
+ export { ProductGridSection } from './sections/ProductGrid';
45
+ export { BannerSection } from './sections/Banner';
46
+ export { RichTextSection } from './sections/RichText';
47
+ export { ImageSection } from './sections/Image';
48
+ export { SpacerSection } from './sections/Spacer';
@@ -0,0 +1,63 @@
1
+ /**
2
+ * <BannerSection> — 顶部细条状横幅
3
+ */
4
+
5
+ import * as React from 'react';
6
+ import type { BannerData } from '../types';
7
+
8
+ export interface BannerSectionProps {
9
+ data: BannerData;
10
+ }
11
+
12
+ export function BannerSection({ data }: BannerSectionProps): React.ReactElement | null {
13
+ const [dismissed, setDismissed] = React.useState(false);
14
+ if (dismissed) return null;
15
+
16
+ const style: React.CSSProperties = {
17
+ padding: '12px 32px',
18
+ background: data.background_color ?? '#f97316',
19
+ color: data.text_color ?? '#ffffff',
20
+ display: 'flex',
21
+ alignItems: 'center',
22
+ justifyContent: 'center',
23
+ gap: 14,
24
+ fontSize: 14,
25
+ fontWeight: 500,
26
+ };
27
+
28
+ return (
29
+ <div style={style}>
30
+ <span>{data.text}</span>
31
+ {data.link && (
32
+ <a
33
+ href={data.link}
34
+ style={{
35
+ color: 'inherit',
36
+ textDecoration: 'underline',
37
+ fontWeight: 600,
38
+ }}
39
+ >
40
+ {data.link_label ?? '查看详情'}
41
+ </a>
42
+ )}
43
+ {data.dismissible && (
44
+ <button
45
+ type="button"
46
+ aria-label="关闭"
47
+ onClick={() => setDismissed(true)}
48
+ style={{
49
+ marginLeft: 'auto',
50
+ background: 'transparent',
51
+ border: 'none',
52
+ color: 'inherit',
53
+ cursor: 'pointer',
54
+ fontSize: 18,
55
+ opacity: 0.8,
56
+ }}
57
+ >
58
+ ×
59
+ </button>
60
+ )}
61
+ </div>
62
+ );
63
+ }
@@ -0,0 +1,92 @@
1
+ /**
2
+ * <HeroSection> — 渲染 hero 区块
3
+ *
4
+ * 这是按 Page Schema 设计的纯渲染组件:
5
+ * - props 直接接收 HeroData
6
+ * - 不读取 schema,不操作 schema,只渲染
7
+ * - 编辑器与 production storefront 共用同一份组件
8
+ */
9
+
10
+ import * as React from 'react';
11
+ import type { HeroData } from '../types';
12
+
13
+ export interface HeroSectionProps {
14
+ data: HeroData;
15
+ }
16
+
17
+ const HEIGHT_MAP: Record<NonNullable<HeroData['height']>, string> = {
18
+ small: '240px',
19
+ medium: '380px',
20
+ large: '540px',
21
+ full: '85vh',
22
+ };
23
+
24
+ export function HeroSection({ data }: HeroSectionProps): React.ReactElement {
25
+ const height = HEIGHT_MAP[data.height ?? 'medium'];
26
+ const align = data.text_align ?? 'center';
27
+
28
+ const style: React.CSSProperties = {
29
+ position: 'relative',
30
+ minHeight: height,
31
+ display: 'flex',
32
+ flexDirection: 'column',
33
+ justifyContent: 'center',
34
+ alignItems: align === 'center' ? 'center' : align === 'right' ? 'flex-end' : 'flex-start',
35
+ padding: '64px 32px',
36
+ background: data.background_image
37
+ ? `linear-gradient(rgba(0,0,0,0.35), rgba(0,0,0,0.35)), url(${data.background_image}) center/cover`
38
+ : data.background_color ?? '#0f172a',
39
+ color: data.text_color ?? '#ffffff',
40
+ textAlign: align,
41
+ };
42
+
43
+ return (
44
+ <section style={style}>
45
+ <h1 style={{ fontSize: 'clamp(2rem, 5vw, 3.5rem)', fontWeight: 800, margin: 0, maxWidth: '24ch' }}>
46
+ {data.headline}
47
+ </h1>
48
+ {data.subheadline && (
49
+ <p style={{ fontSize: 'clamp(1rem, 2vw, 1.25rem)', marginTop: 16, maxWidth: '40ch', opacity: 0.9 }}>
50
+ {data.subheadline}
51
+ </p>
52
+ )}
53
+ {(data.cta_label || data.secondary_cta_label) && (
54
+ <div style={{ marginTop: 28, display: 'flex', gap: 12, flexWrap: 'wrap', justifyContent: align }}>
55
+ {data.cta_label && (
56
+ <a
57
+ href={data.cta_link ?? '#'}
58
+ style={{
59
+ padding: '12px 28px',
60
+ background: data.text_color === '#ffffff' ? '#fff' : '#0f172a',
61
+ color: data.text_color === '#ffffff' ? '#0f172a' : '#ffffff',
62
+ borderRadius: 8,
63
+ fontWeight: 600,
64
+ textDecoration: 'none',
65
+ fontSize: 15,
66
+ }}
67
+ >
68
+ {data.cta_label}
69
+ </a>
70
+ )}
71
+ {data.secondary_cta_label && (
72
+ <a
73
+ href={data.secondary_cta_link ?? '#'}
74
+ style={{
75
+ padding: '12px 28px',
76
+ background: 'transparent',
77
+ color: 'inherit',
78
+ border: '1px solid currentColor',
79
+ borderRadius: 8,
80
+ fontWeight: 600,
81
+ textDecoration: 'none',
82
+ fontSize: 15,
83
+ }}
84
+ >
85
+ {data.secondary_cta_label}
86
+ </a>
87
+ )}
88
+ </div>
89
+ )}
90
+ </section>
91
+ );
92
+ }
@@ -0,0 +1,42 @@
1
+ /**
2
+ * <ImageSection> — 单张图片,可加链接
3
+ */
4
+
5
+ import * as React from 'react';
6
+ import type { ImageData } from '../types';
7
+
8
+ export interface ImageSectionProps {
9
+ data: ImageData;
10
+ }
11
+
12
+ const WIDTH_MAP: Record<NonNullable<ImageData['width']>, string> = {
13
+ full: '100%',
14
+ normal: '1200px',
15
+ narrow: '760px',
16
+ };
17
+
18
+ export function ImageSection({ data }: ImageSectionProps): React.ReactElement {
19
+ const maxWidth = WIDTH_MAP[data.width ?? 'normal'];
20
+
21
+ const img = (
22
+ <img
23
+ src={data.url}
24
+ alt={data.alt ?? ''}
25
+ style={{ display: 'block', width: '100%', height: 'auto' }}
26
+ />
27
+ );
28
+
29
+ return (
30
+ <section style={{ padding: data.width === 'full' ? 0 : '32px', textAlign: 'center' }}>
31
+ <div style={{ maxWidth, margin: '0 auto' }}>
32
+ {data.link ? (
33
+ <a href={data.link} style={{ display: 'block' }}>
34
+ {img}
35
+ </a>
36
+ ) : (
37
+ img
38
+ )}
39
+ </div>
40
+ </section>
41
+ );
42
+ }