OSS-first docs
These docs teach the open system first: contracts, generated surfaces, runtimes, governance, and incremental adoption. Studio shows up as the operating layer on top, not as the source of truth.
Cross-platform UI
How ContractSpec keeps React and React Native components compatible by splitting responsibility across shared runtime models, platform primitives, resolver aliases, and the composed design-system layer.
What cross-platform means here
The shared rendering story is layered: the core package owns models and resolver helpers, the React packages own hook APIs, the UI kits own raw primitives, and the design-system owns the higher-level product surfaces that pair web and mobile implementations.
@contractspec/lib.presentation-runtime-core
Shared state, table models, workflow logic, visualization helpers, and the alias helpers consumed by web and native builds.
@contractspec/lib.presentation-runtime-react
React-facing hooks such as useContractTable, useDataViewTable, and useWorkflow. This is the shared hook surface most product code starts from.
@contractspec/lib.presentation-runtime-react-native
Native entrypoint for mobile apps. It re-exports the shared table and data-view hooks so the controller API stays aligned with the React package.
@contractspec/lib.ui-kit-web
Browser-first primitives and accessibility helpers. Reach for this layer when you want direct control over the web renderer.
@contractspec/lib.ui-kit
Native-first primitives for Expo and React Native. Reach for this layer when the render surface should stay mobile-native.
@contractspec/lib.design-system
Composed components, token helpers, and paired web/mobile implementations that sit on top of both UI kits.
useContractTable
Use this when your rows and column definitions already live in product code and you want one headless controller for sorting, pagination, selection, visibility, pinning, sizing, and expansion.
useDataViewTable
Use this when the table should stay driven by a DataViewSpec instead of hand-authored columns. It adapts the spec to the same generic controller model.
Native re-export boundary
On native apps, import the same hook names from @contractspec/lib.presentation-runtime-react-native when you want the import path itself to signal a mobile boundary.
withPlatformUI
Use this lightweight adapter when a design-system surface needs one object that carries the current platform, tokens, and breakpoints.
mapTokensForPlatform
Use this when resolved tokens need to be mapped into platform-specific token shapes before the final renderer consumes them.
import {
defaultTokens,
mapTokensForPlatform,
withPlatformUI,
} from '@contractspec/lib.design-system';
const nativeTokens = mapTokensForPlatform('native', defaultTokens);
const ui = withPlatformUI({
tokens: defaultTokens,
platform: 'web',
});
// ui is a lightweight config object for design-system consumers.
// nativeTokens is the mapped token shape for a native renderer.Resolver and alias setup
Teach the bundler what “web” and “native” mean before you try to share component code. These helpers are public from the root @contractspec/lib.presentation-runtime-core entrypoint.
Next.js / Turbopack
Use withPresentationTurbopackAliases as the default path in current Next.js apps. It patches nextConfig.turbopack instead of mutating a webpack config object.
Next.js / Webpack fallback
Use withPresentationWebpackAliases only when a Next.js app explicitly opts back into webpack via the CLI flags.
Expo / Metro
Use withPresentationMetroAliases on Metro when native platforms should resolve web ui-kit /ui imports and the shared React runtime root to native implementations.
import { withPresentationTurbopackAliases } from '@contractspec/lib.presentation-runtime-core';
/** @type {import('next').NextConfig} */
const nextConfig = withPresentationTurbopackAliases({
turbopack: {
resolveAlias: {
fs: { browser: 'browserify-fs' },
},
},
});
export default nextConfig;import { withPresentationWebpackAliases } from '@contractspec/lib.presentation-runtime-core';
/** @type {import('next').NextConfig} */
const nextConfig = {
webpack: (config) => withPresentationWebpackAliases(config),
};
export default nextConfig;const { getDefaultConfig } = require('expo/metro-config');
const {
withPresentationMetroAliases,
} = require('@contractspec/lib.presentation-runtime-core');
const projectRoot = __dirname;
const config = getDefaultConfig(projectRoot);
module.exports = withPresentationMetroAliases(config);How the remapping works
The helpers are intentionally asymmetric because Turbopack patches the Next config object, Webpack mutates a resolver config, and Metro maps modules at request time for native platforms.
What Webpack remaps
- @contractspec/lib.ui-kit -> @contractspec/lib.ui-kit-web
- @contractspec/lib.presentation-runtime-react-native -> @contractspec/lib.presentation-runtime-react
- Prepends .web.js, .web.jsx, .web.ts, and .web.tsx to webpack resolve.extensions
What Turbopack remaps
- @contractspec/lib.ui-kit -> @contractspec/lib.ui-kit-web
- @contractspec/lib.presentation-runtime-react-native -> @contractspec/lib.presentation-runtime-react
- Initializes or merges turbopack.resolveExtensions with a web-first extension list
What Metro remaps
- @contractspec/lib.ui-kit-web/ui/* -> @contractspec/lib.ui-kit/ui/* on ios/android/native/mobile
- Root @contractspec/lib.presentation-runtime-react -> @contractspec/lib.presentation-runtime-react-native
- Enables package exports and expands platform resolution ordering
Layout primitives
VStack, HStack, and Box are the closest thing to a shared layout vocabulary, but their defaults and a few props still differ across the web and native packages.
import { Box, HStack, VStack } from '@contractspec/lib.ui-kit-web/ui/stack';
export function AccountSummaryHeader() {
return (
<VStack gap="lg" align="stretch" className="w-full">
<HStack gap="sm" align="center" justify="between" wrap="wrap">
<Box gap="xs" align="center" justify="start" wrap="nowrap">
<StatusDot />
<Title />
</Box>
<Actions />
</HStack>
<Filters />
</VStack>
);
}
// In native-only files, swap the import to:
// @contractspec/lib.ui-kit/ui/stack
//
// On web-only pages, VStack / HStack / Box also support:
// <VStack as="section">...</VStack>Rendering patterns
Keep the controller stable, then decide whether the final surface should be a raw web primitive, a raw native primitive, or a composed design-system wrapper.
import { DataTable as DesignSystemTable } from '@contractspec/lib.design-system';
import { DataTable as NativeTable } from '@contractspec/lib.ui-kit/ui/data-table';
import { DataTable as WebTable } from '@contractspec/lib.ui-kit-web/ui/data-table';
import { useContractTable } from '@contractspec/lib.presentation-runtime-react';
export function useAccountsController(data, columns) {
return useContractTable({
data,
columns,
selectionMode: 'single',
initialState: {
sorting: [{ id: 'arr', desc: true }],
pagination: { pageIndex: 0, pageSize: 5 },
},
});
}
export function WebAccounts({ controller }) {
return <WebTable controller={controller} />;
}
export function NativeAccounts({ controller }) {
return <NativeTable controller={controller} />;
}
export function ProductAccounts({ controller }) {
return (
<DesignSystemTable
controller={controller}
title="Accounts"
description="Same controller, composed ContractSpec shell."
/>
);
}
// If the table is spec-driven instead of hand-authored,
// swap useContractTable for useDataViewTable.DataViewRenderer, ListTablePage, and DataTable. The web and mobile files stay separate inside the package while your app imports one design-system boundary.Gotchas and boundaries
- withPresentationNextAliases no longer exists. Use withPresentationTurbopackAliases for the default Next.js path or withPresentationWebpackAliases for explicit webpack fallback.
- Prefer root runtime imports when alias helpers matter. Metro remaps the root @contractspec/lib.presentation-runtime-react package, not arbitrary deep hook subpaths.
- Metro only rewrites @contractspec/lib.ui-kit-web/ui/* imports. Router-specific web packages and other web-only helpers still need platform-aware imports.
- presentation-runtime-core is headless. It owns models and config helpers, not rendered React components.
- design-system compatibility comes from paired .tsx / .mobile.tsx implementations and token helpers such as withPlatformUI and mapTokensForPlatform.
- Stack primitives are similar across platforms, but the prop surface is not identical. Stay inside the common subset for shared renderers.
- Alias helpers solve module resolution only. They do not replace app-level monorepo watchFolders, Expo Router setup, or other Next configuration.
Customer markdown kit
Copy these markdown snippets into your own AGENTS.md, LLM guide, README, or engineering playbook when you want to enforce the same cross-surface rules across customer projects.
# Cross-Surface Rendering Policy
- Import runtime bundler helpers from `@contractspec/lib.presentation-runtime-core` root only.
- Use `withPresentationTurbopackAliases` for default Next.js projects.
- Use `withPresentationWebpackAliases` only when the app explicitly opts into webpack.
- Use `withPresentationMetroAliases` for Expo and Metro builds.
- Prefer `@contractspec/lib.design-system` for shared product-facing surfaces.
- Use `@contractspec/lib.ui-kit-web` only for web-specific primitive lanes.
- Use `@contractspec/lib.ui-kit` only for native-specific primitive lanes.
- Keep shared layout code inside the common `VStack` / `HStack` / `Box` subset.
- Do not use removed `withPresentationNextAliases`.
- Treat `.tsx` / `.mobile.tsx` pairs as the standard design-system compatibility boundary.# Cross-Surface Rendering Checklist
1. Configure the bundler aliases before sharing any UI code.
2. Choose the controller layer:
- use `useContractTable` for app-owned rows and columns
- use `useDataViewTable` for DataViewSpec-driven tables
3. Choose the renderer lane:
- web primitive: `@contractspec/lib.ui-kit-web`
- native primitive: `@contractspec/lib.ui-kit`
- shared product surface: `@contractspec/lib.design-system`
4. Verify mirrored `.tsx` / `.mobile.tsx` implementations where the design-system owns the surface.
5. In shared layout code, set `gap`, `align`, `justify`, and `wrap` explicitly.
6. Check docs and examples for root imports and current helper names before copying them into product code.Why ContractSpec
Keep educational and comparison content reachable without letting it define the primary OSS learning path.