Edit this page
Learn how to add custom native code to your Expo project.
Edit this page
You can add custom native code by using one or both of the following approaches:
Expo and React Native developers typically spend the vast majority of their time writing JavaScript code and using native APIs and components that are made available through libraries like expo-camera
, react-native-safe-area-context
, and react-native
itself. These libraries allow developers to access and use device features (such as Camera) from their JavaScript code. They may also provide access to a third-party service SDK that is implemented in native code (such as @sentry/react-native
, which provides bindings to the Sentry native SDK for Android and iOS).
If you are using the sandbox app, Expo Go, you can only access native libraries that are included in the Expo SDK, or libraries that do not include any custom native code. In contrast, by creating a development build, you can control any part of your app. You can change the native code or configuration that is possible in any other native app. For more details on how to determine if a third-party library depends on custom code, see Using third party libraries.
When using development builds, using libraries with custom native code is straightforward:
npx expo install react-native-localize
You can now use the library in your application code.
The development overview provides details on key concepts for developing an app with Expo and the flow of the core development loop.
It's common to encounter situations where a library doesn't help you get your job done. For example, the library might not provide access to a specific platform feature, or a third-party library might not provide bindings for React Native. To solve this, you can write Kotlin (or Java) and/or Swift (or Objective-C) code to add any native functionality to your app directly, or to provide bindings to your JavaScript code. There are different tools you can use for this in React Native, and we typically recommend using the Expo Modules API. If you intend to write C++ code, you may want to explore the Turbo Modules API provided by React Native.
The Expo Modules API allows you to write Swift and Kotlin to add new capabilities to your app with native modules and views. We believe that using the Expo Modules API makes building and maintaining nearly all kinds of React Native modules about as easy as it can be. We think that the Expo Modules API is the best choice for most developers building native modules for their apps.
An overview of the APIs and utilities provided by Expo to develop native modules.
A tutorial on creating a native module that persists settings with the Expo Modules API.
A tutorial on creating a native view that renders a native WebView component with the Expo Modules API.
If you intend to use your native module in a single app (you can always change your mind later), we recommend using a "local" Expo module to write custom native code. Local Expo Modules function similarly to Expo Modules used by library developers and within the Expo SDK, like expo-camera
, but they are not published on npm. Instead, you create them directly inside your project.
Creating a local module scaffolds a Swift and Kotlin module inside the modules
directory in your project, and these modules are automatically linked to your app.
-
npx create-expo-module@latest --local
-
npx expo run
If you intend to use your native module with multiple apps, then use npx create-expo-module@latest,
leave out the --local
flag, and create a standalone module. You can publish your package to npm, or you can put it in a packages directory in your monorepo (if you have one) to use it in a similar way to local modules.
The following suggestions are most important when using CNG, but are good guidelines even if you don't use it.
By default, Expo projects created with create-expo-app
use CNG and do not contain android or ios native directories until you've run the npx expo prebuild
command in your project. When using CNG, developers typically do not commit the android and ios directories to source control and do not generate them locally, since EAS Build will do it automatically during the build process. That said, it is common to generate native directories and build locally with npx expo run
when writing custom native code, to have a fast feedback loop and full access to native debugging tools in Android Studio / Xcode.
If your native code requires that you make changes to your project configuration, such as modifying the project's AndroidManifest.xml or Info.plist, you should apply these changes through a config plugin rather than by modifying the files directly in the android and ios directories. Remember that changes made directly to native project directories will be lost the next time you run prebuild when you use CNG.
Additionally, if you need to hook into Android lifecycle events or AppDelegate
methods, use the APIs provided by Expo Modules for Android and iOS to accomplish this rather than modifying the source files in your native project directories directly or using a config plugin to add the code, which does not compose well with other plugins.