Skip to content

feat: adaptive record surface + semantic span + responsive columns; fix #2515 (framework#2578)#2237

Merged
os-zhuang merged 4 commits into
mainfrom
feat/adaptive-layout-2578
Jul 4, 2026
Merged

feat: adaptive record surface + semantic span + responsive columns; fix #2515 (framework#2578)#2237
os-zhuang merged 4 commits into
mainfrom
feat/adaptive-layout-2578

Conversation

@os-zhuang

Copy link
Copy Markdown
Contributor

Renderer side of framework#2578. Companion to objectstack-ai/framework#2595 (spec/lint). All metadata is AI-authored → the renderer derives presentation (surface / width / columns) at runtime; AI writes zero presentation.

What

  • Adaptive surface — a record's create/edit/detail opens as a full page when the object is field-heavy, or a drawer when it's light. Derived from field count (deriveRecordSurface), not authored; mobile always pages. Wired into the app-shell ObjectView detail navigation (an authored view/object navigation still wins).
  • Semantic field spanFormField.span (auto/full); legacy colSpan is clamped so it never overflows the derived grid. ObjectForm honours per-section columns and carries span/colSpan from section defs — fixes #2515 (type:'simple' ignored section.columns, grouped fields rendered single-column).
  • Responsive columnsinferColumns scales the column cap with field count (≤3→1, ≤8→2, ≤15→3, 16+→4); the actual count follows the form's real width via CSS container queries, so the same form goes 1→2→3→4 columns as a drawer widens or becomes a page.
  • Runtime overlay widthNavigationConfig.size bucket resolved to a viewport-clamped width (overlayWidthFor); a pixel width is never authored.

Verified in the browser (live showcase server)

Detail surface — field_zoo (57 fields) row-click → full page /…/record/<id>; product (6 fields) → drawer overlay.

Responsive columns, same field_zoo 57-field create form (measured gridTemplateColumns):

viewport form container actual columns
1680px 1498px 4
760px 624px 2
430px 398px 1

Tests

  • autoLayout 43 + plugin-form 173 tests — green
  • turbo type-check 29/29 — green

Refs framework#2578 · companion of objectstack-ai/framework#2595

🤖 Generated with Claude Code

os-zhuang and others added 4 commits July 4, 2026 22:45
…ntic span; fix #2515 (#2578)

Step 2 (objectui renderer) of #2578:
- autoLayout: resolveColSpan — span-aware (`span:'full'` → whole row), clamps a
  legacy colSpan to the current grid (never overflows), and lays each section
  out at its own column density. applyAutoColSpan gains a `sectionColumns` arg.
- ObjectForm (simple/grouped path): honour per-section `columns` within the
  shared form grid, and carry `span`/`colSpan` from section field defs — fixes
  #2515 (section.columns was ignored; grouped fields rendered single-column).
- FormField.span ('auto'|'full') added to @object-ui/types; carried through
  sectionFields.
- ObjectView: default the record surface from field count via a local
  deriveRecordSurface mirror (field-heavy → full page; mobile → page); explicit
  schema.layout / navigation config still wins. Consolidate to the
  @objectstack/spec export when objectui adopts spec >= 11.10.

autoLayout 43 tests (16 new); plugin-form 173 tests green; changed files type-clean.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
…s in the console (#2578)

The console renders app-shell's ObjectView, which owns the record-detail
surface (defaulted to 'drawer'). Wire deriveRecordSurface so a FIELD-HEAVY
object's record peek defaults to a full page — automating the "a heavy detail
object can set navigation.mode = 'page'" note the code already carried — while
light objects keep the drawer peek. An authored view/object `navigation` still
wins. Also export deriveRecordSurface from @object-ui/plugin-view.

Browser-verified against a live showcase server: field_zoo (57 fields) row-click
navigates to a full /record page; product (6 fields) opens a drawer overlay.
turbo type-check green (29/29).

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
…ime overlay width (#2578)

- inferColumns: scale the column CAP with field count (<=3->1, <=8->2, <=15->3,
  16+->4). The cap is an upper bound; CSS container queries pick the ACTUAL
  column count from the form's real width, so the same form goes 1->2->3->4
  columns as a drawer widens or becomes a page.
- recordSurface: deriveOverlaySize + overlayWidthFor — map a size bucket (or
  'auto' from field count) to a viewport-clamped width min(cap, 92vw). Consumes
  spec NavigationConfig.size (no inert key).
- app-shell ObjectView: detailNavigation resolves size->width at runtime; the
  auto drawer is sized to the content. Authored navigation still wins.
- types: ViewNavigationConfig.size.

autoLayout 43 tests; turbo type-check 29/29.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
…(framework#2578)

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
@vercel

vercel Bot commented Jul 4, 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 4, 2026 4:55pm

Request Review

@os-zhuang os-zhuang merged commit 8bf6295 into main Jul 4, 2026
10 checks passed
@os-zhuang os-zhuang deleted the feat/adaptive-layout-2578 branch July 4, 2026 16:56
@github-actions

github-actions Bot commented Jul 4, 2026

Copy link
Copy Markdown
Contributor

✅ Console Performance Budget

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

📦 Bundle Size Report

Package Size Gzipped
app-shell (index.js) 7.51KB 2.72KB
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) 436.69KB 93.78KB
core (index.js) 1.65KB 0.59KB
create-plugin (index.js) 9.28KB 2.98KB
data-objectstack (index.js) 109.66KB 27.05KB
fields (index.js) 185.28KB 45.20KB
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.09KB 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.99KB 48.90KB
plugin-editor (index.js) 2.46KB 1.10KB
plugin-form (index.js) 101.47KB 24.65KB
plugin-gantt (index.js) 136.67KB 33.88KB
plugin-grid (index.js) 165.48KB 43.77KB
plugin-kanban (index.js) 48.15KB 12.94KB
plugin-list (index.js) 97.89KB 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) 85.46KB 20.80KB
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

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.

1 participant