@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.
- package/dist/page-schema/PageRenderer.d.ts +29 -0
- package/dist/page-schema/PageRenderer.d.ts.map +1 -0
- package/dist/page-schema/PageRenderer.js +73 -0
- package/dist/page-schema/PageRenderer.js.map +1 -0
- package/dist/page-schema/index.d.ts +20 -0
- package/dist/page-schema/index.d.ts.map +1 -0
- package/dist/page-schema/index.js +18 -0
- package/dist/page-schema/index.js.map +1 -0
- package/dist/page-schema/sections/Banner.d.ts +10 -0
- package/dist/page-schema/sections/Banner.d.ts.map +1 -0
- package/dist/page-schema/sections/Banner.js +35 -0
- package/dist/page-schema/sections/Banner.js.map +1 -0
- package/dist/page-schema/sections/Hero.d.ts +15 -0
- package/dist/page-schema/sections/Hero.d.ts.map +1 -0
- package/dist/page-schema/sections/Hero.js +44 -0
- package/dist/page-schema/sections/Hero.js.map +1 -0
- package/dist/page-schema/sections/Image.d.ts +10 -0
- package/dist/page-schema/sections/Image.d.ts.map +1 -0
- package/dist/page-schema/sections/Image.js +12 -0
- package/dist/page-schema/sections/Image.js.map +1 -0
- package/dist/page-schema/sections/ProductGrid.d.ts +29 -0
- package/dist/page-schema/sections/ProductGrid.d.ts.map +1 -0
- package/dist/page-schema/sections/ProductGrid.js +28 -0
- package/dist/page-schema/sections/ProductGrid.js.map +1 -0
- package/dist/page-schema/sections/RichText.d.ts +13 -0
- package/dist/page-schema/sections/RichText.d.ts.map +1 -0
- package/dist/page-schema/sections/RichText.js +16 -0
- package/dist/page-schema/sections/RichText.js.map +1 -0
- package/dist/page-schema/sections/Spacer.d.ts +10 -0
- package/dist/page-schema/sections/Spacer.d.ts.map +1 -0
- package/dist/page-schema/sections/Spacer.js +6 -0
- package/dist/page-schema/sections/Spacer.js.map +1 -0
- package/dist/page-schema/types.d.ts +138 -0
- package/dist/page-schema/types.d.ts.map +1 -0
- package/dist/page-schema/types.js +129 -0
- package/dist/page-schema/types.js.map +1 -0
- package/dist/react.d.ts +2 -0
- package/dist/react.d.ts.map +1 -1
- package/dist/react.js +5 -0
- package/dist/react.js.map +1 -1
- package/package.json +1 -1
- package/src/page-schema/PageRenderer.tsx +147 -0
- package/src/page-schema/index.ts +48 -0
- package/src/page-schema/sections/Banner.tsx +63 -0
- package/src/page-schema/sections/Hero.tsx +92 -0
- package/src/page-schema/sections/Image.tsx +42 -0
- package/src/page-schema/sections/ProductGrid.tsx +96 -0
- package/src/page-schema/sections/RichText.tsx +49 -0
- package/src/page-schema/sections/Spacer.tsx +15 -0
- package/src/page-schema/types.ts +286 -0
- 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
|
package/dist/react.d.ts.map
CHANGED
|
@@ -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;
|
|
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
|
@@ -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
|
+
}
|