building-native-ui
Install this skill
npx skills add expo/skillsWorks across Claude Code, Cursor, Codex, Copilot & Antigravity
Building native UI in Expo relies on modern primitives that align with Apple's Human Interface Guidelines. This skill focuses on delivering high-performance mobile interfaces by favoring specific expo-prefixed libraries over legacy React Native alternatives. It emphasizes the use of fluid layout techniques, such as flex gap and ScrollView with automatic content inset adjustments, to ensure responsive behavior across device screen sizes. Developers integrate native-feeling components—like SF Symbols for iconography and platform-specific controls—while maintaining a clean project architecture by separating components from routing logic. By defaulting to Expo Go for rapid iteration and applying precise configuration for custom native modules when strictly necessary, this approach ensures code remains maintainable, performant, and consistent with the current Expo ecosystem standards for professional-grade mobile application development.
When to Use This Skill
- •Developing high-fidelity iOS and Android screens with native design patterns
- •Modernizing existing applications by replacing deprecated modules like WebView or AsyncStorage
- •Building complex navigation stacks with dynamic headers and integrated search bars
- •Incorporating hardware-accelerated animations using Reanimated and gesture handlers
How to Invoke This Skill
Example prompts that trigger this skill in Claude Code, Cursor, or Antigravity:
- “How do I implement a native search bar in my Expo Router stack?
- “Show me how to use SF Symbols in an Expo project.
- “What is the best way to handle safe area insets on iOS using Expo?
- “How can I add a blur effect to my modal background?
- “Replace my legacy Picker component with the modern native version.
Pro Tips
- 💡Always prioritize native modules (like NativeTabs, `expo-symbols`) over JavaScript alternatives for better performance and a truly native feel.
- 💡Master `react-native-reanimated` early for creating fluid, gesture-driven animations that enhance user engagement without sacrificing performance.
- 💡Leverage Expo Router's dynamic routes and groups effectively to organize your application's navigation structure for scalability and maintainability.
What this skill does
- •Implementation of Apple SF Symbols via expo-symbols
- •Responsive layout management using useWindowDimensions and flexbox gap
- •High-performance media integration with expo-video and expo-audio
- •Native-style interface construction using specialized controls like SegmentedControl and DateTimePicker
- •Visual enhancement via expo-blur and liquid glass effects
When not to use it
- ✕When building a standard responsive website that does not require native mobile hardware capabilities
- ✕When the project requires legacy modules explicitly removed from the modern React Native environment
Example workflow
- Initialize screen layout using a ScrollView with contentInsetAdjustmentBehavior set to automatic.
- Implement navigation headers and attach search functionality using headerSearchBarOptions.
- Add UI elements such as switches or custom sliders using the modern native control library.
- Style the view using flex gap and align items according to Human Interface Guidelines.
- Test the implementation directly in Expo Go to verify touch responsiveness and visual output.
Prerequisites
- –Expo Router installed
- –Basic familiarity with React Native flexbox
- –Project setup following modern naming conventions
Pitfalls & limitations
- !Attempting to co-locate components within the app directory instead of keeping them in a separate directory
- !Using legacy components like SafeAreaView instead of modern context-based solutions
- !Forgetting to escape backticks or quotes in string-heavy UI definitions
FAQ
How it compares
This skill mandates a strict, modern ecosystem-first approach that enforces architectural consistency, whereas generic prompting often results in outdated, non-performant code using deprecated legacy React Native modules.
📄 Full skill instructions — original source: expo/skills
## References
Consult these resources as needed:
- ./references/route-structure.md -- Route file conventions, dynamic routes, query parameters, groups, and folder organization
- ./references/tabs.md -- Native tab bar with NativeTabs, migration from JS tabs, iOS 26 features
- ./references/icons.md -- SF Symbols with expo-symbols, common icon names, animations, and weights
- ./references/controls.md -- Native iOS controls: Switch, Slider, SegmentedControl, DateTimePicker, Picker
- ./references/visual-effects.md -- Blur effects with expo-blur and liquid glass with expo-glass-effect
- ./references/animations.md -- Reanimated animations: entering, exiting, layout, scroll-driven, and gestures
- ./references/search.md -- Search bar integration with headers, useSearch hook, and filtering patterns
- ./references/gradients.md -- CSS gradients using experimental_backgroundImage (New Architecture only)
- ./references/media.md -- Media handling for Expo Router including camera, audio, video, and file saving
- ./references/storage.md -- Data storage patterns including SQLite, AsyncStorage, and SecureStore
- ./references/webgpu-three.md -- 3D graphics, games, and GPU-powered visualizations with WebGPU and Three.js
## Running the App
**CRITICAL: Always try Expo Go first before creating custom builds.**
Most Expo apps work in Expo Go without any custom native code. Before running
npx expo run:ios or npx expo run:android:1. **Start with Expo Go**: Run
npx expo start and scan the QR code with Expo Go2. **Check if features work**: Test your app thoroughly in Expo Go
3. **Only create custom builds when required** - see below
### When Custom Builds Are Required
You need
npx expo run:ios/android or eas build ONLY when using:- **Local Expo modules** (custom native code in
modules/)- **Apple targets** (widgets, app clips, extensions via
@bacons/apple-targets)- **Third-party native modules** not included in Expo Go
- **Custom native configuration** that can't be expressed in
app.json### When Expo Go Works
Expo Go supports a huge range of features out of the box:
- All
expo-* packages (camera, location, notifications, etc.)- Expo Router navigation
- Most UI libraries (reanimated, gesture handler, etc.)
- Push notifications, deep links, and more
**If you're unsure, try Expo Go first.** Creating custom builds adds complexity, slower iteration, and requires Xcode/Android Studio setup.
## Code Style
- Be cautious of unterminated strings. Ensure nested backticks are escaped; never forget to escape quotes correctly.
- Always use import statements at the top of the file.
- Always use kebab-case for file names, e.g.
comment-card.tsx- Always remove old route files when moving or restructuring navigation
- Never use special characters in file names
- Configure tsconfig.json with path aliases, and prefer aliases over relative imports for refactors.
## Routes
See
./references/route-structure.md for detailed route conventions.- Routes belong in the
app directory.- Never co-locate components, types, or utilities in the app directory. This is an anti-pattern.
- Ensure the app always has a route that matches "/", it may be inside a group route.
## Library Preferences
- Never use modules removed from React Native such as Picker, WebView, SafeAreaView, or AsyncStorage
- Never use legacy expo-permissions
-
expo-audio not expo-av-
expo-video not expo-av-
expo-symbols not @expo/vector-icons-
react-native-safe-area-context not react-native SafeAreaView-
process.env.EXPO_OS not Platform.OS-
React.use not React.useContext-
expo-image Image component instead of intrinsic element img-
expo-glass-effect for liquid glass backdrops## Responsiveness
- Always wrap root component in a scroll view for responsiveness
- Use
<ScrollView contentInsetAdjustmentBehavior="automatic" /> instead of <SafeAreaView> for smarter safe area insets-
contentInsetAdjustmentBehavior="automatic" should be applied to FlatList and SectionList as well- Use flexbox instead of Dimensions API
- ALWAYS prefer
useWindowDimensions over Dimensions.get() to measure screen size## Behavior
- Use expo-haptics conditionally on iOS to make more delightful experiences
- Use views with built-in haptics like
<Switch /> from React Native and @react-native-community/datetimepicker- When a route belongs to a Stack, its first child should almost always be a ScrollView with
contentInsetAdjustmentBehavior="automatic" set- Prefer
headerSearchBarOptions in Stack.Screen options to add a search bar- Use the
<Text selectable /> prop on text containing data that could be copied- Consider formatting large numbers like 1.4M or 38k
- Never use intrinsic elements like 'img' or 'div' unless in a webview or Expo DOM component
# Styling
Follow Apple Human Interface Guidelines.
## General Styling Rules
- Prefer flex gap over margin and padding styles
- Prefer padding over margin where possible
- Always account for safe area, either with stack headers, tabs, or ScrollView/FlatList
contentInsetAdjustmentBehavior="automatic"- Ensure both top and bottom safe area insets are accounted for
- Inline styles not StyleSheet.create unless reusing styles is faster
- Add entering and exiting animations for state changes
- Use
{ borderCurve: 'continuous' } for rounded corners unless creating a capsule shape- ALWAYS use a navigation stack title instead of a custom text element on the page
- When padding a ScrollView, use
contentContainerStyle padding and gap instead of padding on the ScrollView itself (reduces clipping)- CSS and Tailwind are not supported - use inline styles
## Text Styling
- Add the
selectable prop to every <Text/> element displaying important data or error messages- Counters should use
{ fontVariant: 'tabular-nums' } for alignment## Shadows
Use CSS
boxShadow style prop. NEVER use legacy React Native shadow or elevation styles.<View style={{ boxShadow: "0 1px 2px rgba(0, 0, 0, 0.05)" }} />'inset' shadows are supported.
# Navigation
## Link
Use
<Link href="/path" /> from 'expo-router' for navigation between routes.import { Link } from 'expo-router';
// Basic link
<Link href="/path" />
// Wrapping custom components
<Link href="/path" asChild>
<Pressable>...</Pressable>
</Link>Whenever possible, include a
<Link.Preview> to follow iOS conventions. Add context menus and previews frequently to enhance navigation.## Stack
- ALWAYS use
_layout.tsx files to define stacks- Use Stack from 'expo-router/stack' for native navigation stacks
### Page Title
Set the page title in Stack.Screen options:
<Stack.Screen options={{ title: "Home" }} />## Context Menus
Add long press context menus to Link components:
import { Link } from "expo-router";
<Link href="/settings" asChild>
<Link.Trigger>
<Pressable>
<Card />
</Pressable>
</Link.Trigger>
<Link.Menu>
<Link.MenuAction
title="Share"
icon="square.and.arrow.up"
onPress={handleSharePress}
/>
<Link.MenuAction
title="Block"
icon="nosign"
destructive
onPress={handleBlockPress}
/>
<Link.Menu title="More" icon="ellipsis">
<Link.MenuAction title="Copy" icon="doc.on.doc" onPress={() => {}} />
<Link.MenuAction
title="Delete"
icon="trash"
destructive
onPress={() => {}}
/>
</Link.Menu>
</Link.Menu>
</Link>;## Link Previews
Use link previews frequently to enhance navigation:
<Link href="/settings">
<Link.Trigger>
<Pressable>
<Card />
</Pressable>
</Link.Trigger>
<Link.Preview />
</Link>Link preview can be used with context menus.
## Modal
Present a screen as a modal:
<Stack.Screen name="modal" options={{ presentation: "modal" }} />Prefer this to building a custom modal component.
## Sheet
Present a screen as a dynamic form sheet:
<Stack.Screen
name="sheet"
options={{
presentation: "formSheet",
sheetGrabberVisible: true,
sheetAllowedDetents: [0.5, 1.0],
contentStyle: { backgroundColor: "transparent" },
}}
/>- Using
contentStyle: { backgroundColor: "transparent" } makes the background liquid glass on iOS 26+.## Common route structure
A standard app layout with tabs and stacks inside each tab:
app/
_layout.tsx — <NativeTabs />
(index,search)/
_layout.tsx — <Stack />
index.tsx — Main list
search.tsx — Search view// app/_layout.tsx
import { NativeTabs, Icon, Label } from "expo-router/unstable-native-tabs";
import { Theme } from "../components/theme";
export default function Layout() {
return (
<Theme>
<NativeTabs>
<NativeTabs.Trigger name="(index)">
<Icon sf="list.dash" />
<Label>Items</Label>
</NativeTabs.Trigger>
<NativeTabs.Trigger name="(search)" role="search" />
</NativeTabs>
</Theme>
);
}Create a shared group route so both tabs can push common screens:
// app/(index,search)/_layout.tsx
import { Stack } from "expo-router/stack";
import { PlatformColor } from "react-native";
export default function Layout({ segment }) {
const screen = segment.match(/\((.*)\)/)?.[1]!;
const titles: Record<string, string> = { index: "Items", search: "Search" };
return (
<Stack
screenOptions={{
headerTransparent: true,
headerShadowVisible: false,
headerLargeTitleShadowVisible: false,
headerLargeStyle: { backgroundColor: "transparent" },
headerTitleStyle: { color: PlatformColor("label") },
headerLargeTitle: true,
headerBlurEffect: "none",
headerBackButtonDisplayMode: "minimal",
}}
>
<Stack.Screen name={screen} options={{ title: titles[screen] }} />
<Stack.Screen name="i/[id]" options={{ headerLargeTitle: false }} />
</Stack>
);
}How to Use This Skill Unit
Option A: Project-Specific (Recommended)
- Click "Download" above
- In your project, create the directory:
.agent/skills/building-native-ui/ - Save the file as
SKILL.md - The agent will automatically discover the skill based on its description.
Option B: Global Installation (All Agents)
Save the file to these locations to make it available across all projects:
- Claude Code:
~/.claude/skills/expo/skills/building-native-ui/SKILL.md - Cursor:
~/.cursor/skills/expo/skills/building-native-ui/SKILL.md - Antigravity:
~/.gemini/antigravity/skills/expo/skills/building-native-ui/SKILL.md
🚀 Install with CLI:npx skills add expo/skills
