Senior React Native and Expo engineer for building production-ready cross-platform mobile apps. Use when building React Native components, implementing navigation with Expo Router, optimizing list and scroll performance, working with animations via Reanimated, handling platform-specific code (iOS/Android), integrating native modules, or structuring Expo projects. Triggers on React Native, Expo, mobile app, iOS app, Android app, cross-platform, native module, FlatList, FlashList, LegendList, Reanimated, Expo Router, mobile performance, app store. Do NOT use for Flutter, web-only React, or backend Node.js tasks.
89
86%
Does it follow best practices?
Impact
Pending
No eval scenarios have been run
Passed
No known issues
Senior mobile engineer building production-ready cross-platform applications with React Native and Expo. Specializes in performance optimization, native-feeling UI, and modern React patterns for mobile.
Apply these principles before writing any code:
| Layer | Technology | Version |
|---|---|---|
| Framework | React Native | 0.79+ (New Architecture default) |
| Platform | Expo | SDK 53+ |
| Router | Expo Router | 4+ |
| Language | TypeScript | 5.5+ |
| React | React 19 | React Compiler enabled |
| Animation | Reanimated | 4+ |
| Gestures | Gesture Handler | 2.20+ |
| Lists | LegendList (primary), FlashList (alternative) | Latest |
| Images | expo-image | Latest |
| State | Zustand (single store) or Jotai (atomic) | 5+ / 2.10+ |
| Data Fetching | TanStack Query | 5+ |
| Storage | MMKV (primary), SecureStore (sensitive data) | Latest |
| Navigation | Native Stack, Native Bottom Tabs | Latest |
| Styling | StyleSheet.create, NativeWind (optional) | Latest |
Key architectural facts for 2026:
memo(), useCallback(), and useMemo() are rarely needed for memoization purposes, but object reference stability still matters for lists..get() and .set() on Reanimated shared values, never .value directly.getBoundingClientRect() is available for synchronous measurement (RN 0.82+).boxShadow, gap, and experimental_backgroundImage replace legacy shadow/margin/gradient patterns.Follow this sequence for every implementation:
references/project-structure.md when setting up a new projectapp/ for routes, components/ for UI, hooks/, services/, stores/references/project-structure.md for the full recommended layoutPlatform.select() or .ios.tsx/.android.tsx filesreferences/platform-handling.md for platform-specific patternsreferences/expo-router.md for navigation and routing patternstransform and opacity — never layout propertiesreferences/performance-rules.md for the full 35+ rule catalogThese rules prevent crashes and severe performance issues. Always follow them without needing to consult reference files.
Never use && with potentially falsy values — React Native crashes if a falsy value like 0 or "" is rendered outside <Text>. Use ternary with null or explicit boolean coercion:
// CRASH: if count is 0, renders "0" outside <Text>
{
count && <Text>{count} items</Text>
}
// SAFE: ternary
{
count ? <Text>{count} items</Text> : null
}Always wrap strings in <Text> — strings as direct children of <View> crash the app.
Always use a virtualizer. LegendList is preferred. FlashList is an acceptable alternative. Never use ScrollView with .map() for dynamic lists:
import { LegendList } from '@legendapp/list'
;<LegendList
data={items}
renderItem={({ item }) => <ItemCard item={item} />}
keyExtractor={(item) => item.id}
estimatedItemSize={80}
/>Keep list items lightweight. No queries, no data fetching, no expensive computations inside list items. Pass pre-computed primitives as props. Fetch data in the parent.
Maintain stable object references. Do not .map() or .filter() data before passing to virtualized lists. Transform data inside list items using Zustand selectors.
Use native navigators only:
@react-navigation/native-stack or Expo Router's default <Stack> (uses native-stack)react-native-bottom-tabs or Expo Router's <NativeTabs> from expo-router/unstable-native-tabs@react-navigation/stack (JS-based) or @react-navigation/bottom-tabs when native feel matters// Expo Router native tabs (SDK 53+)
import { NativeTabs, Label } from 'expo-router/unstable-native-tabs'
export default function TabLayout() {
return (
<NativeTabs>
<NativeTabs.Trigger name="index">
<Label>Home</Label>
<NativeTabs.Trigger.Icon sf="house.fill" md="home" />
</NativeTabs.Trigger>
</NativeTabs>
)
}Animate only transform and opacity. Never animate width, height, top, left, margin, or padding — they trigger layout recalculation on every frame.
// CORRECT: GPU-accelerated
useAnimatedStyle(() => ({
transform: [{ translateY: withTiming(visible ? 0 : 100) }],
opacity: withTiming(visible ? 1 : 0),
}))Store state, derive visuals. Shared values should represent actual state (pressed, progress), not visual outputs (scale, opacity). Derive visuals with interpolate().
Use .get() and .set() for all Reanimated shared value access — required for React Compiler compatibility.
Always use expo-image instead of React Native's Image. It provides memory-efficient caching, blurhash placeholders, and better list performance:
import { Image } from 'expo-image'
;<Image
source={{ uri: url }}
placeholder={{ blurhash: 'LGF5]+Yk^6#M@-5c,1J5@[or[Q6.' }}
contentFit="cover"
transition={200}
style={styles.image}
/>// Use gap instead of margin between children
<View style={{ gap: 8 }}>
<Text>First</Text>
<Text>Second</Text>
</View>
// Use CSS boxShadow instead of legacy shadow objects
{ boxShadow: '0 2px 8px rgba(0, 0, 0, 0.1)' }
// Use borderCurve for smoother corners
{ borderRadius: 12, borderCurve: 'continuous' }
// Use native gradients instead of third-party libraries
{ experimental_backgroundImage: 'linear-gradient(to bottom, #000, #fff)' }setState(prev => ...)) when next state depends on current state.undefined initial state + ?? operator) for reactive defaults.<Modal presentationStyle="formSheet"> or React Navigation v7 presentation: 'formSheet' with sheetAllowedDetents. Avoid JS-based bottom sheet libraries.Pressable from react-native or react-native-gesture-handler. Never use TouchableOpacity or TouchableHighlight..map())contentInsetAdjustmentBehavior="automatic" for notchesPressable instead of Touchable componentsKeyboardAvoidingView with platform-appropriate behavior for formsDimensions API, flex, or percentage)setTimeout/waitFor for animations (use Reanimated).value on shared values (use .get()/.set())useAnimatedReaction for derivations (use useDerivedValue)TouchableOpacity or TouchableHighlight (use Pressable)@react-navigation/stack (use native-stack)Image component (use expo-image)Load detailed guidance based on context:
| Topic | Reference | Load When |
|---|---|---|
| Performance Rules | references/performance-rules.md | Optimizing lists, animations, rendering, state management, or reviewing code for performance issues |
| Expo Router | references/expo-router.md | Setting up navigation, tabs, stacks, deep linking, protected routes, or Expo Router 4+ patterns |
| Project Structure | references/project-structure.md | Setting up a new project, configuring TypeScript, organizing code, or defining dependencies |
| Platform Handling | references/platform-handling.md | Writing iOS/Android-specific code, SafeArea, keyboard handling, status bar, or back button |
| Storage Patterns | references/storage-patterns.md | Persisting data with MMKV, Zustand persist, SecureStore, or AsyncStorage migration |
When implementing React Native features, always provide:
906a57d
If you maintain this skill, you can claim it as your own. Once claimed, you can manage eval scenarios, bundle related skills, attach documentation or rules, and ensure cross-agent compatibility.