Skip to content

feat(detail): 详情页支持对象 fieldGroups 分组 + 详情提示迁移到 spec 可写的 detail.* 块 (#2148)#2149

Merged
baozhoutao merged 1 commit into
mainfrom
claude/blissful-mendeleev-601119
Jul 2, 2026
Merged

feat(detail): 详情页支持对象 fieldGroups 分组 + 详情提示迁移到 spec 可写的 detail.* 块 (#2148)#2149
baozhoutao merged 1 commit into
mainfrom
claude/blissful-mendeleev-601119

Conversation

@baozhoutao

Copy link
Copy Markdown
Contributor

关联 #2148,同时收编 #2065(detail.stageField: false 关闭阶段条)。

问题一:字段分组在详情页不生效

对象上定义的 fieldGroups[] + fields[].group(设计器里的"字段分组")此前只有表单在用(plugin-form 的 deriveFieldGroupSections),默认详情页和 synth 详情页完全忽略,全部字段被自动拆成"主要字段 + 更多详情"两段。

改动:

  • plugin-detail synth 新增导出 deriveFieldGroupDetailSections(def):按 fieldGroups 声明顺序分桶 fields[].group,语义对齐 plugin-form——空的已声明分组丢弃、collapsible/collapsed 透传(collapsed 映射为 DetailSection 的 defaultCollapsed)、未分组字段进末尾无标题分区。未分组桶跳过审计/系统字段(created_atupdated_bytenant_id 等,与 app-shell 的 AUDIT+HIDDEN 集合完全一致);显式归入某分组的审计字段保留(显式声明优先,与手写 sections 语义一致)。
  • 新增导出 resolveDetailSections(def, sections?) 统一优先级,buildDefaultDetails 接入,所以所有 synth 消费方(默认详情页、Studio 预览、metadata-admin)自动获得分组分区。
  • RecordDetailView 复用同一推导:命名分组标题走 sectionLabel i18n 约定({ns}.objects.{objectName}._sections.{key}.label),未分组剩余字段仍保留 primary/"更多详情" 折叠拆分,不破坏现有"更多详情"按钮体验。

分区来源优先级: 显式 sections(detail.sections ?? views.form.sections)> fieldGroups 推导 > 自动两段式兜底。

问题二:详情提示写在 spec 不认的键上

console 此前从 views.detail.highlightFieldsviews.form.sections、顶层 stageField 读取提示,但 @objectstack/specObjectSchema.create() 对这些键直接抛错、safeParse 静默剥离——即用户按文档写了也到不了运行时。spec 里顶层 detail 块是 passthrough(可写),所以:

  • 所有详情提示优先读 detail.*:detail.stageField(string 指定 / false 关闭,详情页:支持 detail.stageField: false 显式关闭自动状态进度条 #2065)、detail.highlightFields(支持 string{name} 条目,畸形条目丢弃)、detail.sectionsdetail.sectionGroupsdetail.useFieldGroups: false(关闭 fieldGroups 推导的逃生门)。
  • 旧键(views.*、顶层 stageField/highlightFields)全部保留为回退,零破坏。

⚠️ 行为变化(release note)

已声明 fieldGroups 的对象,详情页会从自动两段式变为按分组渲染。逃生门:detail: { useFieldGroups: false }

测试

  • plugin-detail synth 新增 19 个测试(detail.* 提示优先级/回退、deriveFieldGroupDetailSections 各分支、resolveDetailSections 优先级、buildDefaultPageSchema 集成含 detail.stageField: false 去掉 record:path),全部通过;synth 定向套件 73/73。
  • app-shell 全量:123 文件 / 955 用例全绿。
  • turbo type-check(plugin-detail + app-shell)通过;turbo build(两条依赖链 29 个任务)通过。
  • 本 worktree 有 7 个 plugin-detail 测试套件因 components/dist 解析失败——已用 git stash 在干净树上复现同样 7 个套件失败,属预置环境问题,与本次改动无关。

后续(不在本 PR)

Closes #2065

@vercel

vercel Bot commented Jul 2, 2026

Copy link
Copy Markdown

The latest updates on your projects. Learn more about Vercel for GitHub.

1 Skipped Deployment
Project Deployment Actions Updated (UTC)
objectui Ignored Ignored Jul 2, 2026 9:01am

Request Review

@github-actions

github-actions Bot commented Jul 2, 2026

Copy link
Copy Markdown
Contributor

✅ Console Performance Budget

Metric Value Budget
Main entry (gzip) 59.1 KB 350 KB
Entry file index-DkrqPIK5.js
Status PASS

📦 Bundle Size Report

Package Size Gzipped
app-shell (index.js) 7.28KB 2.63KB
app-shell (runtime-config.js) 4.72KB 1.69KB
app-shell (types.js) 0.01KB 0.04KB
auth (AuthContext.js) 0.31KB 0.24KB
auth (AuthGuard.js) 1.17KB 0.53KB
auth (AuthProvider.js) 17.90KB 3.67KB
auth (AuthShell.js) 3.49KB 1.40KB
auth (ForgotPasswordForm.js) 4.79KB 1.88KB
auth (LoginForm.js) 9.55KB 3.36KB
auth (PreviewBanner.js) 0.90KB 0.50KB
auth (RegisterForm.js) 6.63KB 2.15KB
auth (SocialSignInButtons.js) 8.89KB 3.61KB
auth (UserMenu.js) 3.40KB 1.22KB
auth (auth-gate-events.js) 1.29KB 0.66KB
auth (authStyles.js) 5.04KB 1.72KB
auth (createAuthClient.js) 26.07KB 6.30KB
auth (createAuthenticatedFetch.js) 3.93KB 1.55KB
auth (index.js) 1.75KB 0.76KB
auth (types.js) 0.59KB 0.35KB
auth (useAuth.js) 4.29KB 0.82KB
auth (useIsWorkspaceAdmin.js) 1.61KB 0.85KB
collaboration (CommentThread.js) 18.38KB 4.49KB
collaboration (LiveCursors.js) 3.17KB 1.27KB
collaboration (PresenceAvatars.js) 3.65KB 1.42KB
collaboration (PresenceProvider.js) 2.42KB 0.96KB
collaboration (index.js) 1.25KB 0.53KB
collaboration (useCommentSearch.js) 1.98KB 0.88KB
collaboration (useConflictResolution.js) 7.75KB 1.86KB
collaboration (useMentionNotifications.js) 1.81KB 0.68KB
collaboration (usePresence.js) 6.33KB 1.84KB
collaboration (useRealtimeSubscription.js) 7.91KB 2.01KB
components (index.js) 434.78KB 93.36KB
core (index.js) 1.65KB 0.59KB
create-plugin (index.js) 9.28KB 2.98KB
data-objectstack (index.js) 107.48KB 26.47KB
fields (index.js) 167.32KB 40.69KB
i18n (LocalizationContext.js) 1.76KB 0.96KB
i18n (currency.js) 1.22KB 0.64KB
i18n (i18n.js) 4.32KB 1.77KB
i18n (index.js) 2.46KB 0.96KB
i18n (pickLocalized.js) 1.31KB 0.67KB
i18n (provider.js) 5.37KB 1.72KB
i18n (useObjectLabel.js) 21.15KB 4.68KB
i18n (useSafeTranslation.js) 2.68KB 0.98KB
layout (index.js) 36.30KB 10.04KB
mobile (MobileProvider.js) 0.92KB 0.49KB
mobile (ResponsiveContainer.js) 0.94KB 0.38KB
mobile (breakpoints.js) 1.51KB 0.70KB
mobile (createOfflineDataSource.js) 5.61KB 1.74KB
mobile (index.js) 1.50KB 0.62KB
mobile (offlineQueue.js) 3.91KB 1.35KB
mobile (pwa.js) 0.97KB 0.49KB
mobile (serviceWorker.js) 1.48KB 0.62KB
mobile (serviceWorkerSource.js) 3.41KB 1.48KB
mobile (useBreakpoint.js) 1.54KB 0.65KB
mobile (useGesture.js) 4.42KB 1.27KB
mobile (useOfflineSync.js) 1.99KB 0.72KB
mobile (usePullToRefresh.js) 2.53KB 0.85KB
mobile (useResponsive.js) 0.71KB 0.42KB
mobile (useResponsiveConfig.js) 1.36KB 0.63KB
mobile (useSpecGesture.js) 1.77KB 0.77KB
mobile (useTouchTarget.js) 1.01KB 0.54KB
permissions (MePermissionsProvider.js) 5.09KB 1.84KB
permissions (PermissionContext.js) 0.31KB 0.25KB
permissions (PermissionGuard.js) 0.89KB 0.45KB
permissions (PermissionProvider.js) 3.46KB 1.03KB
permissions (evaluator.js) 4.00KB 1.23KB
permissions (index.js) 0.91KB 0.41KB
permissions (store.js) 0.91KB 0.42KB
permissions (useFieldPermissions.js) 1.28KB 0.52KB
permissions (usePermissions.js) 1.50KB 0.70KB
plugin-ai (index.js) 15.71KB 3.79KB
plugin-calendar (index.js) 45.10KB 12.33KB
plugin-charts (index.js) 46.11KB 12.99KB
plugin-chatbot (index.js) 172.75KB 41.13KB
plugin-dashboard (index.js) 108.28KB 26.86KB
plugin-designer (index.js) 213.48KB 42.95KB
plugin-detail (index.js) 202.63KB 48.77KB
plugin-editor (index.js) 2.46KB 1.10KB
plugin-form (index.js) 97.03KB 23.51KB
plugin-gantt (index.js) 136.67KB 33.88KB
plugin-grid (index.js) 156.94KB 41.43KB
plugin-kanban (index.js) 48.15KB 12.94KB
plugin-list (index.js) 97.93KB 23.09KB
plugin-map (index.js) 16.02KB 4.98KB
plugin-markdown (index.js) 13.65KB 4.67KB
plugin-report (index.js) 37.83KB 9.97KB
plugin-timeline (index.js) 25.37KB 7.20KB
plugin-tree (index.js) 8.21KB 2.76KB
plugin-view (index.js) 84.50KB 20.37KB
providers (DataSourceProvider.js) 0.75KB 0.39KB
providers (MetadataProvider.js) 1.37KB 0.59KB
providers (ThemeProvider.js) 1.55KB 0.67KB
providers (UploadProvider.js) 11.71KB 3.53KB
providers (index.js) 0.44KB 0.22KB
providers (types.js) 0.01KB 0.04KB
react-runtime (index.js) 3.19KB 1.38KB
react (LazyPluginLoader.js) 3.77KB 1.33KB
react (SchemaRenderer.js) 18.23KB 5.97KB
react (index.js) 0.76KB 0.42KB
sdui-parser (codegen.js) 4.09KB 1.74KB
sdui-parser (index.js) 2.16KB 0.94KB
sdui-parser (parse.js) 10.04KB 2.82KB
sdui-parser (types.js) 0.29KB 0.24KB
sdui-parser (validate.js) 4.69KB 1.48KB
tenant (TenantContext.js) 0.31KB 0.25KB
tenant (TenantGuard.js) 1.04KB 0.43KB
tenant (TenantProvider.js) 2.76KB 0.98KB
tenant (TenantScopedQuery.js) 0.77KB 0.44KB
tenant (index.js) 0.75KB 0.38KB
tenant (resolver.js) 2.64KB 0.76KB
tenant (useTenant.js) 0.50KB 0.32KB
tenant (useTenantBranding.js) 0.62KB 0.39KB
types (ai.js) 0.20KB 0.17KB
types (api-types.js) 0.20KB 0.18KB
types (app.js) 2.87KB 0.99KB
types (base.js) 0.20KB 0.18KB
types (blocks.js) 0.20KB 0.18KB
types (complex.js) 0.20KB 0.18KB
types (crud.js) 0.20KB 0.18KB
types (data-display.js) 0.20KB 0.18KB
types (data-protocol.js) 0.20KB 0.19KB
types (data.js) 0.20KB 0.18KB
types (designer.js) 0.77KB 0.41KB
types (disclosure.js) 0.20KB 0.18KB
types (feedback.js) 0.20KB 0.18KB
types (field-types.js) 0.20KB 0.18KB
types (form.js) 0.20KB 0.18KB
types (index.js) 1.54KB 0.68KB
types (layout.js) 0.20KB 0.18KB
types (mobile.js) 0.20KB 0.18KB
types (navigation.js) 0.20KB 0.18KB
types (objectql.js) 0.20KB 0.18KB
types (overlay.js) 0.20KB 0.18KB
types (permissions.js) 0.20KB 0.18KB
types (plugin-scope.js) 0.20KB 0.18KB
types (record-components.js) 0.20KB 0.19KB
types (registry.js) 0.20KB 0.18KB
types (reports.js) 0.20KB 0.18KB
types (spec-report.js) 5.26KB 1.96KB
types (tenant.js) 0.20KB 0.18KB
types (theme.js) 0.20KB 0.18KB
types (ui-action.js) 0.75KB 0.46KB
types (views.js) 0.20KB 0.18KB
types (widget.js) 0.20KB 0.18KB

Size Limits

  • ✅ Core packages should be < 50KB gzipped
  • ✅ Component packages should be < 100KB gzipped
  • ⚠️ Plugin packages should be < 150KB gzipped

@os-zhuang

Copy link
Copy Markdown
Contributor

感谢这个 PR——问题一/问题二的诊断和 fieldGroups 推导的实现质量都很高,尤其是"先修 spec 可写通道、再接线"的施工顺序。评审过程中我们把整个 hints 面向上追了一层,最终落成了协议级决定 ADR-0085(已合并,framework#2520;spec 执行 PR framework#2521 已开):对象的展示意图只以跨 surface 语义角色声明(nameField / highlightFields(由 compactLayout 改名) / stageField: string|false / fieldGroups),不再有 detail 提示块。对本 PR 的影响整体是做减法:

保留(核心价值,约 80% 不动)

调整

  1. hints 改读顶层语义角色,删掉 detail.* 分支:detectStatusField 只读顶层 def.stageField(spec 已类型化为 string | false,见 framework#2521);deriveHighlightFields 的声明源改读顶层 highlightFields ?? compactLayout(改名别名过渡),删 detail.highlightFields;
  2. 删除 detail.sections / detail.sectionGroups / detail.useFieldGroups 及对应测试——v1 分组只有 fieldGroups 一种声明、全 surface 生效,无逃生门(理由见 ADR "Alternatives considered":防名单漂移/静默丢字段/静默遮蔽三类陷阱);
  3. RecordDetailView 顺手删 views.* 三处读取(1229/1389/1393;双仓已验证零作者,ADR 决定不留弃用窗口)——显式 sections 分支整体消失,坍缩为"fieldGroups 推导 → 自动两段式"两级;
  4. 补 changeset(PR 描述里那段行为变化 release note 直接搬进去)。

两个不阻塞本 PR 的后续(已有跟踪):spec 侧新增了单源推导 deriveFieldGroupLayout(framework#2521),spec 版本 bump 后会有一个 consumer-switch PR 把本 PR 的推导实现迁移过去;fieldGroups 折叠语义收敛为 collapse 枚举(collapsible/collapsed 在 spec 解析期做了别名映射,UI 侧读取可在 consumer-switch PR 里一并对齐)。

合并前建议起 showcase 对一个声明了 fieldGroups 的对象做一轮浏览器验证(分组卡片/折叠/stageField: false 去进度条)。有任何疑问欢迎直接评论,我们可以配合调整。

@baozhoutao baozhoutao merged commit 1d5e6e9 into main Jul 2, 2026
10 checks passed
@baozhoutao baozhoutao deleted the claude/blissful-mendeleev-601119 branch July 2, 2026 15:06
…s from spec-writable detail.* block (#2148)

问题一: 对象上定义的 fieldGroups(字段分组)此前只在表单里生效,
详情页(默认详情 + synth 页面)完全忽略。现在:

- plugin-detail synth 新增 deriveFieldGroupDetailSections():按声明顺序
  把 fields[].group 归入 fieldGroups[] 定义的分组,透传 collapsible/
  collapsed(映射为 defaultCollapsed),空分组丢弃,未分组字段进末尾
  无标题分区(跳过审计/系统字段,显式分组的审计字段保留)。
- 新增 resolveDetailSections():options.sections > detail.sections >
  fieldGroups 推导;buildDefaultDetails 全部 synth 路径接入。
- RecordDetailView 复用该推导:分组标题走 sectionLabel i18n,未分组
  剩余字段保留 primary/"更多详情" 折叠拆分。

问题二: console 此前从 spec ObjectSchema 拒绝/剥离的键读取详情提示
(views.detail.highlightFields / views.form.sections / 顶层 stageField)。
现在优先读 spec 可写的 detail.* 透传块,旧键保留为向后兼容回退:

- detail.stageField: string 指定 / false 关闭(收编 #2065)
- detail.highlightFields: 支持 string 或 {name} 条目
- detail.sections / detail.sectionGroups
- detail.useFieldGroups: false 可关闭 fieldGroups 推导

行为变化: 已声明 fieldGroups 的对象,详情页从自动两段式变为按分组
渲染;可用 detail.useFieldGroups: false 退回旧行为。

抽屉(RecordDetailDrawer)分组渲染留待后续 PR,见 #2148
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

详情页:支持 detail.stageField: false 显式关闭自动状态进度条

2 participants