Back to Expo & Mobile

building-native-ui

ExpoReact NativeUI/UX DesignMobile DevelopmentNative ComponentsAnimationsNavigationApp Styling
2.1k📄 MIT🕒 2026-06-16Source ↗

Install this skill

npx skills add expo/skills

Works 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

  1. Initialize screen layout using a ScrollView with contentInsetAdjustmentBehavior set to automatic.
  2. Implement navigation headers and attach search functionality using headerSearchBarOptions.
  3. Add UI elements such as switches or custom sliders using the modern native control library.
  4. Style the view using flex gap and align items according to Human Interface Guidelines.
  5. 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

Should I always create a custom build for my native UI?
No. Always test in Expo Go first. Only create a custom native build if your project requires local native modules, specific Apple targets, or third-party libraries not included in the Expo Go environment.
What is the correct way to handle safe area insets?
Avoid SafeAreaView. Use ScrollView or FlatList with the contentInsetAdjustmentBehavior='automatic' prop to allow the system to handle safe area insets naturally.
Are there specific libraries I must avoid?
Yes. Avoid using removed modules like Picker, WebView, SafeAreaView, and AsyncStorage. Prefer their modern replacements like react-native-safe-area-context and specialized expo-prefixed modules.

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.

Source & trust

2.1k stars📄 MIT🕒 Updated 2026-06-16
📄 Full skill instructions — original source: expo/skills
# Expo UI Guidelines

## 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 Go
2. **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)

  1. Click "Download" above
  2. In your project, create the directory: .agent/skills/building-native-ui/
  3. Save the file as SKILL.md
  4. 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

Read the Master Guide: Mastering Agent Skills

Recommended Rules

View more rules

Recommended Workflows

View more workflows

Recommended MCP Servers

View more MCP servers

Take It Further

Maximize your productivity with these powerful resources

📋

Define Your Standards

Set up coding standards to ensure this workflow produces consistent, high-quality results.

Browse Rules Library
📖

Master Workflows

Learn how to create custom workflows, use Turbo Mode, and build your automation library.

Complete Guide

How to use this Skill in Claude Code & Cursor

For Claude Code (CLI)

To use this skill in Claude Code, copy the rule content into your project's custom instructions or follow our Add-Skill CLI guide. This ensures Claude follows your standards during every code generation.

For Cursor & Windsurf

For Cursor or Windsurf, individual skills are best used in the "Rules for AI" section. This specific unit helps the agent avoid expo & mobile issues, leading to cleaner, more efficient code.

Why the skill format matters: the standardized Agent Skills format lets your AI agent load detailed instructions only when they are relevant, keeping your prompt clean while improving results.

Source & attribution

This skill is categorized under Expo & Mobile and is published by Expo Team, maintained in expo/skills.

← Browse All Agent Skills
Sponsored AI assistant. Recommendations may be paid.