Skip to content

Optimize object detail page with Power Apps-inspired tabbed layout#1200

Merged
hotlong merged 4 commits intomainfrom
claude/optimize-record-detail-page
Apr 9, 2026
Merged

Optimize object detail page with Power Apps-inspired tabbed layout#1200
hotlong merged 4 commits intomainfrom
claude/optimize-record-detail-page

Conversation

@Claude
Copy link
Copy Markdown
Contributor

@Claude Claude AI commented Apr 9, 2026

The object management record detail page displayed all configuration sections in a single vertical stack of 6 cards, creating visual clutter and poor information hierarchy.

Changes

Tabbed Navigation

  • Consolidated all widgets into ObjectDetailTabsWidget with 4 logical tabs:
    • Details: Object properties and configuration
    • Fields: Field designer
    • Relationships: Relationships and unique keys
    • Data: Preview and experience settings

Visual Refinement

  • Removed nested card borders; widgets now render clean sections with uppercase labels
  • Form-style label/value pairs with improved spacing
  • Empty states with centered icons and messaging
  • Hover effects on interactive relationship/key cards

Architecture

  • buildObjectDetailPageSchema() now returns single object-detail-tabs widget
  • Individual widgets (properties, relationships, keys, etc.) remain registered for composition
  • ObjectDetailTabsWidget orchestrates tab rendering and delegates to existing widgets
// Before: 6 stacked widgets
const widgets = [
  { type: 'object-properties', ... },
  { type: 'object-relationships', ... },
  // ... 4 more
];

// After: Single tabbed container
const widgets = [
  { type: 'object-detail-tabs', objectName }
];

Updated MetadataDetailPage.test.tsx to register and validate the new tabs widget.

Claude AI and others added 3 commits April 9, 2026 08:16
Agent-Logs-Url: https://github.com/objectstack-ai/objectui/sessions/32d509ca-9783-48ab-a16a-0182b122bfc7

Co-authored-by: xuyushun441-sys <255036401+xuyushun441-sys@users.noreply.github.com>
Agent-Logs-Url: https://github.com/objectstack-ai/objectui/sessions/32d509ca-9783-48ab-a16a-0182b122bfc7

Co-authored-by: xuyushun441-sys <255036401+xuyushun441-sys@users.noreply.github.com>
Agent-Logs-Url: https://github.com/objectstack-ai/objectui/sessions/32d509ca-9783-48ab-a16a-0182b122bfc7

Co-authored-by: xuyushun441-sys <255036401+xuyushun441-sys@users.noreply.github.com>
@vercel
Copy link
Copy Markdown

vercel bot commented Apr 9, 2026

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

Project Deployment Actions Updated (UTC)
objectui-demo Ready Ready Preview, Comment Apr 9, 2026 8:37am
1 Skipped Deployment
Project Deployment Actions Updated (UTC)
objectui Ignored Ignored Apr 9, 2026 8:37am

Request Review

Copilot AI review requested due to automatic review settings April 9, 2026 08:22
@github-actions
Copy link
Copy Markdown
Contributor

github-actions bot commented Apr 9, 2026

❌ Console Performance Budget

Metric Value Budget
Main entry (gzip) ** KB** KB
Entry file ``
Status FAIL

📦 Bundle Size Report

Package Size Gzipped
auth (AuthContext.js) 0.31KB 0.24KB
auth (AuthGuard.js) 1.17KB 0.53KB
auth (AuthProvider.js) 7.37KB 1.79KB
auth (ForgotPasswordForm.js) 4.91KB 1.66KB
auth (LoginForm.js) 5.00KB 1.64KB
auth (PreviewBanner.js) 0.90KB 0.50KB
auth (RegisterForm.js) 6.85KB 1.86KB
auth (UserMenu.js) 3.40KB 1.22KB
auth (createAuthClient.js) 5.14KB 1.60KB
auth (createAuthenticatedFetch.js) 1.24KB 0.60KB
auth (index.js) 1.18KB 0.51KB
auth (types.js) 0.59KB 0.35KB
auth (useAuth.js) 1.57KB 0.57KB
collaboration (CommentThread.js) 18.38KB 4.49KB
collaboration (LiveCursors.js) 3.17KB 1.27KB
collaboration (PresenceAvatars.js) 3.65KB 1.42KB
collaboration (index.js) 1.16KB 0.50KB
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) 1978.91KB 460.53KB
core (index.js) 1.29KB 0.51KB
create-plugin (index.js) 10.13KB 3.17KB
data-objectstack (index.js) 45.20KB 11.14KB
fields (LookupField-BMf3hBS4.js) 30.76KB 8.34KB
fields (index.js) 74.80KB 15.84KB
fields (rolldown-runtime-CAFD8bLK.js) 0.24KB 0.24KB
fields (useFieldTranslation-Bv3oUVgb.js) 0.26KB 0.21KB
i18n (i18n.js) 2.03KB 0.77KB
i18n (index.js) 1.99KB 0.79KB
i18n (provider.js) 4.63KB 1.47KB
i18n (useObjectLabel.js) 4.51KB 1.61KB
i18n (useSafeTranslation.js) 1.63KB 0.57KB
layout (index.js) 88.09KB 24.91KB
mobile (MobileProvider.js) 0.92KB 0.49KB
mobile (ResponsiveContainer.js) 0.94KB 0.38KB
mobile (breakpoints.js) 1.51KB 0.70KB
mobile (index.js) 1.19KB 0.53KB
mobile (pwa.js) 0.97KB 0.49KB
mobile (serviceWorker.js) 1.48KB 0.62KB
mobile (useBreakpoint.js) 1.54KB 0.65KB
mobile (useGesture.js) 4.42KB 1.27KB
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 (PermissionContext.js) 0.31KB 0.25KB
permissions (PermissionGuard.js) 0.89KB 0.45KB
permissions (PermissionProvider.js) 3.11KB 0.87KB
permissions (evaluator.js) 4.00KB 1.23KB
permissions (index.js) 0.85KB 0.40KB
permissions (store.js) 0.91KB 0.42KB
permissions (useFieldPermissions.js) 1.28KB 0.52KB
permissions (usePermissions.js) 0.99KB 0.49KB
plugin-aggrid (AddressField-DKqaE9pD.js) 2.85KB 0.81KB
plugin-aggrid (AgGridImpl-Bj4QTgBN.js) 6.39KB 2.24KB
plugin-aggrid (AutoNumberField-0RU2dNKe.js) 0.43KB 0.34KB
plugin-aggrid (AvatarField-wqCg539O.js) 2.65KB 1.15KB
plugin-aggrid (BooleanField-DRzAZhSq.js) 1.23KB 0.58KB
plugin-aggrid (CodeField-RWhnDMyL.js) 0.86KB 0.53KB
plugin-aggrid (ColorField-B-YAFXdz.js) 1.26KB 0.59KB
plugin-aggrid (CurrencyField-Bu80a-sI.js) 1.52KB 0.80KB
plugin-aggrid (DateField-CV-NpLbM.js) 0.65KB 0.43KB
plugin-aggrid (DateTimeField--YTsgTjw.js) 0.82KB 0.48KB
plugin-aggrid (EmailField-B-3fWJsH.js) 0.88KB 0.53KB
plugin-aggrid (FileField-BPvzUIuL.js) 5.67KB 2.06KB
plugin-aggrid (FormulaField-MQXJZOep.js) 0.64KB 0.46KB
plugin-aggrid (GeolocationField-BHmdus6A.js) 4.10KB 1.48KB
plugin-aggrid (GridField-DHsGo9l2.js) 1.96KB 0.76KB
plugin-aggrid (ImageField-Do-iSxOD.js) 2.65KB 1.12KB
plugin-aggrid (LocationField-DSvhU9Dz.js) 1.00KB 0.59KB
plugin-aggrid (LookupField-C5avrPku.js) 33.79KB 8.95KB
plugin-aggrid (LookupField-Cf_ZHbXj.js) 0.08KB 0.11KB
plugin-aggrid (MasterDetailField-BVt_izEo.js) 3.40KB 1.14KB
plugin-aggrid (NumberField-Kz4_o5DE.js) 0.76KB 0.50KB
plugin-aggrid (ObjectAgGridImpl-Dk0eJeBi.js) 726.52KB 183.35KB
plugin-aggrid (ObjectField-C1qkl6s4.js) 1.63KB 0.82KB
plugin-aggrid (PasswordField-DB_RmpIU.js) 1.88KB 0.94KB
plugin-aggrid (PercentField-Dmipqv0I.js) 1.85KB 0.88KB
plugin-aggrid (PhoneField-Dn4h6V9H.js) 0.87KB 0.54KB
plugin-aggrid (QRCodeField-C59bI7oR.js) 3.27KB 1.25KB
plugin-aggrid (RatingField-DvTgKH0C.js) 2.05KB 0.93KB
plugin-aggrid (RichTextField-CN5BRd_7.js) 1.19KB 0.63KB
plugin-aggrid (SelectField-CWk7znNa.js) 0.08KB 0.11KB
plugin-aggrid (SelectField-DHaONWbh.js) 1.07KB 0.58KB
plugin-aggrid (SignatureField-DX0fBAS2.js) 3.14KB 1.29KB
plugin-aggrid (SliderField-C9IhmjbF.js) 1.11KB 0.55KB
plugin-aggrid (SummaryField-R9RENAZv.js) 0.60KB 0.45KB
plugin-aggrid (TextAreaField-BY63Nr6-.js) 1.12KB 0.64KB
plugin-aggrid (TextField-SIw8aMzf.js) 0.90KB 0.50KB
plugin-aggrid (TimeField-xxziHPjE.js) 0.61KB 0.40KB
plugin-aggrid (UrlField-Ihk3_ff5.js) 1.04KB 0.57KB
plugin-aggrid (UserField-BMaRAs-r.js) 2.62KB 0.98KB
plugin-aggrid (VectorField-Ci167cxr.js) 0.92KB 0.52KB
plugin-aggrid (createLucideIcon-73kh9fYs.js) 1.72KB 0.93KB
plugin-aggrid (image-DN4QRxZf.js) 0.36KB 0.26KB
plugin-aggrid (index.js) 10.33KB 2.43KB
plugin-aggrid (jsx-runtime-CGDkM_Jn.js) 7.83KB 2.89KB
plugin-aggrid (plus-DoGWIfjF.js) 0.19KB 0.19KB
plugin-aggrid (upload-DlNmJ3Bh.js) 0.29KB 0.23KB
plugin-aggrid (useFieldTranslation-Dg0L3DZd.js) 9.53KB 3.69KB
plugin-aggrid (x-B7f4dqYS.js) 0.19KB 0.19KB
plugin-ai (index.js) 24.21KB 6.44KB
plugin-calendar (index.js) 44.95KB 12.95KB
plugin-charts (AdvancedChartImpl-B65TSHrr.js) 118.19KB 24.21KB
plugin-charts (BarChart-bK3fN6vN.js) 523.67KB 126.89KB
plugin-charts (ChartImpl-C59n6NRW.js) 3.62KB 1.16KB
plugin-charts (index.js) 12.70KB 4.07KB
plugin-charts (jsx-runtime-C8d0IhUE.js) 8.40KB 3.09KB
plugin-chatbot (index.js) 1194.20KB 340.76KB
plugin-dashboard (index.js) 158.02KB 38.69KB
plugin-designer (index.js) 244.91KB 47.93KB
plugin-detail (AddressField-XkJhnK7U.js) 2.66KB 0.79KB
plugin-detail (AutoNumberField-DAXer96s.js) 0.44KB 0.34KB
plugin-detail (AvatarField-B4HWGjpy.js) 2.40KB 1.09KB
plugin-detail (BooleanField-BWbHFdPU.js) 1.14KB 0.55KB
plugin-detail (CodeField-CYK1R0K2.js) 0.80KB 0.50KB
plugin-detail (ColorField-Cf30zlLw.js) 1.19KB 0.57KB
plugin-detail (CurrencyField-CxHdjLtq.js) 1.47KB 0.78KB
plugin-detail (DateField-CzLE9dJa.js) 0.60KB 0.41KB
plugin-detail (DateTimeField-CkArgTZK.js) 0.78KB 0.46KB
plugin-detail (EmailField-D0XnlRh3.js) 0.83KB 0.51KB
plugin-detail (FileField-C_W9AHRC.js) 5.12KB 1.90KB
plugin-detail (FormulaField-7rJnN7AH.js) 0.66KB 0.45KB
plugin-detail (GeolocationField-C3UgdycN.js) 3.31KB 1.23KB
plugin-detail (GridField-BLGVUO_F.js) 1.83KB 0.74KB
plugin-detail (ImageField-CUgSmM7N.js) 2.45KB 1.09KB
plugin-detail (LocationField-xNNkiPPO.js) 0.97KB 0.58KB
plugin-detail (LookupField-B_Y3UoK0.js) 30.31KB 8.22KB
plugin-detail (LookupField-hPtjiFJ8.js) 0.08KB 0.11KB
plugin-detail (MasterDetailField-ScuAoycq.js) 2.90KB 0.99KB
plugin-detail (NumberField-PbGNg-Vr.js) 0.72KB 0.47KB
plugin-detail (ObjectField-DJXFQ8T4.js) 1.57KB 0.79KB
plugin-detail (PasswordField-BbME3sUs.js) 1.13KB 0.66KB
plugin-detail (PercentField-Doxloda9.js) 1.76KB 0.86KB
plugin-detail (PhoneField-BDsMzDQW.js) 0.82KB 0.51KB
plugin-detail (QRCodeField-DUCJZ-KK.js) 2.15KB 0.93KB
plugin-detail (RatingField-DfBtSh4U.js) 1.56KB 0.69KB
plugin-detail (RichTextField-CxRcgUYZ.js) 1.12KB 0.62KB
plugin-detail (SelectField-Bui_TXyx.js) 0.08KB 0.11KB
plugin-detail (SelectField-DuXgDqr9.js) 0.96KB 0.55KB
plugin-detail (SignatureField-B5MHguKv.js) 2.81KB 1.15KB
plugin-detail (SliderField-Cxz3qr_G.js) 1.04KB 0.53KB
plugin-detail (SummaryField-gQaKM67m.js) 0.62KB 0.45KB
plugin-detail (TextAreaField-C8LFCzwl.js) 1.06KB 0.62KB
plugin-detail (TextField-DJhHxw75.js) 0.84KB 0.47KB
plugin-detail (TimeField-fh1-EmU-.js) 0.57KB 0.38KB
plugin-detail (UrlField-DyFvBCLQ.js) 0.98KB 0.54KB
plugin-detail (UserField-h8L3E-j7.js) 2.43KB 0.96KB
plugin-detail (VectorField-CDJCUfGQ.js) 0.92KB 0.52KB
plugin-detail (index.js) 136.92KB 30.18KB
plugin-detail (src-BvG0sGzX.js) 1958.97KB 457.41KB
plugin-detail (useFieldTranslation-RqUbC-nK.js) 0.25KB 0.20KB
plugin-editor (MonacoImpl-BMgw4JKF.js) 17.97KB 5.53KB
plugin-editor (index.js) 1.61KB 0.74KB
plugin-editor (jsx-runtime-HjdTyiyb.js) 7.63KB 2.80KB
plugin-form (index.js) 60.57KB 13.62KB
plugin-gantt (index.js) 215.46KB 52.74KB
plugin-grid (index.js) 101.97KB 27.64KB
plugin-kanban (KanbanEnhanced-BYdG4dd1.js) 30.21KB 8.94KB
plugin-kanban (KanbanImpl-C0fDmIb0.js) 13.33KB 4.08KB
plugin-kanban (chevron-down-B6UH8BbF.js) 0.15KB 0.18KB
plugin-kanban (index.js) 18.55KB 5.80KB
plugin-kanban (plus-BTqoaaEC.js) 9.31KB 3.57KB
plugin-kanban (sortable.esm-DzUCoMzQ.js) 59.36KB 17.43KB
plugin-list (index.js) 2029.12KB 472.67KB
plugin-map (chunk-vKJrgz-R.js) 1.18KB 0.67KB
plugin-map (index.js) 123.47KB 30.25KB
plugin-map (maplibre-gl-dWSG1tSG.js) 1331.90KB 296.95KB
plugin-markdown (MarkdownImpl-zSR34qzi.js) 220.69KB 53.61KB
plugin-markdown (index.js) 1.04KB 0.58KB
plugin-markdown (jsx-runtime-B1W8iDPg.js) 8.39KB 3.09KB
plugin-report (index.js) 77.40KB 16.24KB
plugin-timeline (index.js) 99.93KB 24.18KB
plugin-view (index.js) 124.17KB 33.50KB
plugin-workflow (index.js) 81.18KB 17.06KB
react (LazyPluginLoader.js) 3.77KB 1.33KB
react (SchemaRenderer.js) 9.07KB 2.83KB
react (index.js) 0.76KB 0.42KB
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.74KB 0.39KB
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.25KB 0.58KB
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 (tenant.js) 0.20KB 0.18KB
types (theme.js) 0.20KB 0.18KB
types (ui-action.js) 0.20KB 0.18KB
types (views.js) 0.20KB 0.18KB
types (widget.js) 0.20KB 0.18KB
types (workflow.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

Copy link
Copy Markdown
Contributor

Copilot AI left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pull request overview

Updates the Console’s object detail page to reduce visual clutter by replacing the previous 6-section vertical stack with a single tabbed container, keeping the detail page schema PageSchema-driven while reorganizing content into four logical areas.

Changes:

  • Replaced the object detail page body schema with a single object-detail-tabs widget node.
  • Added and registered ObjectDetailTabsWidget to render 4 tabs (Details, Fields, Relationships, Data) and delegate to existing widgets.
  • Refined existing object detail widgets’ UI to a cleaner sectioned layout and adjusted MetadataDetailPage tests for the new entry widget.

Reviewed changes

Copilot reviewed 5 out of 5 changed files in this pull request and generated 4 comments.

Show a summary per file
File Description
apps/console/src/schemas/objectDetailPageSchema.ts Switches object detail schema body to a single tabbed widget node.
apps/console/src/components/schema/registerObjectDetailWidgets.ts Registers the new object-detail-tabs widget type in the ComponentRegistry.
apps/console/src/components/schema/objectDetailWidgets.tsx Refactors styling/structure of properties/relationships/keys/data widgets for the new layout.
apps/console/src/components/schema/ObjectDetailTabsWidget.tsx Introduces the tabbed container widget orchestrating the object detail sections.
apps/console/src/tests/MetadataDetailPage.test.tsx Updates tests to register/assert the new tabbed widget.

Comment on lines +20 to +28
import {
ObjectPropertiesWidget,
ObjectRelationshipsWidget,
ObjectKeysWidget,
ObjectDataExperienceWidget,
ObjectDataPreviewWidget,
} from './objectDetailWidgets';
import { ObjectFieldDesignerWidget } from './ObjectFieldDesignerWidget';

Copy link

Copilot AI Apr 9, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ObjectDetailTabsWidget composes object detail sections by importing and rendering the widget components directly. This bypasses SchemaRenderer features (expression evaluation, per-node error boundary, debug attributes/ARIA extraction) and prevents swapping/overriding implementations via ComponentRegistry. Consider rendering the per-tab child nodes via <SchemaRenderer schema={...} /> instead of direct component imports, and keep only SchemaNodes in this widget.

Suggested change
import {
ObjectPropertiesWidget,
ObjectRelationshipsWidget,
ObjectKeysWidget,
ObjectDataExperienceWidget,
ObjectDataPreviewWidget,
} from './objectDetailWidgets';
import { ObjectFieldDesignerWidget } from './ObjectFieldDesignerWidget';
import { SchemaRenderer } from '@object-ui/react';
const ObjectPropertiesWidget = ({ schema }: { schema: SchemaNode }) => (
<SchemaRenderer schema={schema} />
);
const ObjectRelationshipsWidget = ({ schema }: { schema: SchemaNode }) => (
<SchemaRenderer schema={schema} />
);
const ObjectKeysWidget = ({ schema }: { schema: SchemaNode }) => (
<SchemaRenderer schema={schema} />
);
const ObjectDataExperienceWidget = ({ schema }: { schema: SchemaNode }) => (
<SchemaRenderer schema={schema} />
);
const ObjectDataPreviewWidget = ({ schema }: { schema: SchemaNode }) => (
<SchemaRenderer schema={schema} />
);
const ObjectFieldDesignerWidget = ({ schema }: { schema: SchemaNode }) => (
<SchemaRenderer schema={schema} />
);

Copilot uses AI. Check for mistakes.
Comment on lines +84 to 91
<div className="space-y-1.5">
<label className="text-xs font-medium text-muted-foreground">Display Name</label>
<p className="text-sm font-medium">{object.label}</p>
</div>
)}
{object.group && (
<div>
<span className="text-muted-foreground">Group</span>
<p className="mt-0.5">{object.group}</p>
<div className="space-y-1.5">
<label className="text-xs font-medium text-muted-foreground">API Name</label>
<p className="text-sm font-mono bg-muted/50 rounded px-2 py-1">{object.name}</p>
</div>
Copy link

Copilot AI Apr 9, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The read-only label/value layout uses <label> elements without associated form controls (htmlFor). This is semantically incorrect and can confuse screen readers. Prefer non-form semantics like <div>/<span> or a <dl> with <dt>/<dd> for definition-style readouts.

Copilot uses AI. Check for mistakes.
Comment on lines +121 to +124
<div className="space-y-1.5">
<label className="text-xs font-medium text-muted-foreground">Field Count</label>
<p className="text-sm font-medium">{object.fieldCount ?? fields.length} fields</p>
</div>
Copy link

Copilot AI Apr 9, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Field Count always renders "{n} fields", which produces incorrect grammar for singular (e.g., "1 fields"). Consider pluralizing based on the count ("1 field" vs "N fields").

Copilot uses AI. Check for mistakes.
Comment on lines +263 to +278
<div className="grid grid-cols-1 sm:grid-cols-3 gap-4">
<div className="rounded-lg border p-6 text-center hover:bg-accent/50 transition-colors cursor-pointer" data-testid="data-experience-forms">
<PanelTop className="h-8 w-8 mx-auto mb-3 text-muted-foreground" />
<p className="text-sm font-semibold mb-1">Forms</p>
<p className="text-xs text-muted-foreground">Design forms for data entry</p>
</div>
<div className="rounded-lg border p-6 text-center hover:bg-accent/50 transition-colors cursor-pointer" data-testid="data-experience-views">
<LayoutList className="h-8 w-8 mx-auto mb-3 text-muted-foreground" />
<p className="text-sm font-semibold mb-1">Views</p>
<p className="text-xs text-muted-foreground">Configure list and detail views</p>
</div>
<div className="rounded-lg border p-6 text-center hover:bg-accent/50 transition-colors cursor-pointer" data-testid="data-experience-dashboards">
<BarChart3 className="h-8 w-8 mx-auto mb-3 text-muted-foreground" />
<p className="text-sm font-semibold mb-1">Dashboards</p>
<p className="text-xs text-muted-foreground">Build visual dashboards</p>
</div>
Copy link

Copilot AI Apr 9, 2026

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The Data Experience cards use cursor-pointer and hover styling but have no click handler or interactive semantics. Either remove the pointer/hover affordance for non-interactive placeholders, or render them as accessible interactive elements (e.g., buttons/links with keyboard support) when navigation is implemented.

Copilot uses AI. Check for mistakes.
@hotlong
Copy link
Copy Markdown
Contributor

hotlong commented Apr 9, 2026

@copilot fix all

…ML, pluralization, remove false affordance

Agent-Logs-Url: https://github.com/objectstack-ai/objectui/sessions/f45bbfad-453a-42a8-9f55-09107853caaa

Co-authored-by: hotlong <50353452+hotlong@users.noreply.github.com>
Copy link
Copy Markdown
Contributor

Copilot AI commented Apr 9, 2026

@copilot fix all

All 4 review comments addressed in 05d7e5c:

  1. SchemaRenderer delegationObjectDetailTabsWidget now renders child widgets via <SchemaRenderer schema={...} /> instead of direct component imports, preserving expression evaluation, error boundaries, and debug attributes.
  2. Semantic HTML — Replaced <label> elements with <dl>/<dt>/<dd> definition lists for read-only label/value pairs.
  3. Pluralization — Fixed "1 fields" → "1 field" with proper singular/plural handling.
  4. False affordance — Removed cursor-pointer and hover:bg-accent/50 from non-interactive Data Experience placeholder cards.

Copilot AI requested a review from hotlong April 9, 2026 08:40
@hotlong hotlong merged commit f4186c3 into main Apr 9, 2026
3 checks passed
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.

4 participants