Edit this page
Learn how to perform link redirection and utilize third-party deep links with +native-intent when using Expo Router.
Expo Router uses an extended version of web standards to navigate through an app. However, native apps do not always conform to server-based routing. This can lead to misalignment when integrating any third-party service. For example, apps can be launched with arbitrary strings or intent objects instead of URLs. There are two common scenarios where you may need to customize a link:
App Closed: When the app is not open, incoming deep-link URLs may require rewriting to ensure seamless navigation.
App Open: When the app is already open, URL customization may be necessary based on specific business logic or user interactions. This logic can be global for the entire app, or localized to a set of routes.
See the Linking into your app guide for instructions on how to set up and test linking in your app.
Expo Router will always evaluate a URL with the assumption that the URL targets a specific page within the app. However, in practice, these URLs can vary in nature:
<my-scheme>:://<provider-hostname>/<uuid>
, and are generated by external sources to navigate users to designated content within the app.In such scenarios, the URL needs to be rewritten to correctly target a route.
To facilitate this, create a special file called +native-intent.tsx at the top level of your project's app directory. This file exports a special redirectSystemPath
method designed to handle URL/path processing. When invoked, it receives an options
object with two attributes: path
and initial
.
app
+native-intent.tsx
Here's an example the applies practices on how redirectSystemPath
is used inside +native-intent.tsx file. Following this example, you can ensure the stability and reliability of your app's URL processing functionality and mitigate the risk of unexpected errors and crashes.
import ThirdPartyService from 'third-party-sdk';
export function redirectSystemPath({ path, initial }) {
try {
if (initial) {
// While the parameter is called `path` there is no guarantee that this is a path or a valid URL
const url = new URL(path, 'myapp://app.home');
// Detection of third-party URLs will change based on the provider
if (url.hostname === '<third-party-provider-hostname>') {
return ThirdPartyService.processReferringUrl(url).catch(() => {
// Something went wrong
return '/unexpected-error';
});
}
return path;
}
return path;
} catch {
// Do not crash inside this function! Instead you should redirect users
// to a custom route to handle unexpected errors, where they are able to report the incident
return '/unexpected-error';
}
}
Handling deep links on the web differs from native platforms, as the initial routing process occurs differently. Expo Router cannot provide a direct counterpart to +native-intent
for web, as web routing is resolved before the website's JavaScript is executed and will differ based upon deployment output and your chosen provider.
As a result, you should implement one of the following patterns that best suits your requirements:
_layout
. This approach is ideal for projects with a single output format targeting client-side rendering.Choose the pattern that aligns with your deployment strategy and technical requirements to ensure seamless handling of incoming deep links on the web platform.
While your app is open, you can react to URL changes within your _layout
files using the usePathname()
hook. The location of the _layout
dictates the scope of the subscription.
_layout
file_layout
file to an existing directory (or create a new group directory)import { Slot, Redirect } from 'expo-router';
export default function RootLayout() {
const pathname = usePathname();
if (pathname && !isUserAllowed(pathname)) {
return <Redirect href="/home" />;
}
return <Slot />;
}
redirectSystemPath
In native apps, an alternative way to rewrite a URL is to handle it within the redirectSystemPath
method. This approach can be simpler for some use cases but comes with certain drawbacks:
+native-intent
is only available in native apps.+native-intent
is processed outside the context of your app. This means you won't have access to additional logic, such as user authentication status or the current route's state.Below is a basic example of how to send navigation events to an external service, such as an analytics or logging service. Consult with your provider for specific instructions.
import ThirdPartyService from 'third-party-sdk';
import { Slot, usePathname } from 'expo-router';
const thirdParty = new ThirdPartyService();
export default function RootLayout() {
const pathname = usePathname();
// Perform the service initiation logic
useEffect(() => {
thirdParty.register();
return () => {
thirdParty.deregister();
};
}, [thirdParty]);
// Send pathname changes to the third party
useEffect(() => {
thirdParty.sendEvent({ pathname });
}, [pathname]);
return <Slot />;
}
Expo Router does not require additional configuration for Universal Links and multiple domains. All URLs provided to your App will be evaluated. To customize the URL scheme(s) for your app, your should customize the scheme
value in your app config.
If you want a URL to be initially evaluated by the user's browser, write the address as a Fully Qualified URL (FQDN) with an http
/https
scheme. Using a complete URL ensures that the link is interpreted as a web URL and opened in the user's browser by default.
This approach is effective for directing users to external websites or Universal Links for other apps.
<Link href="https://my-website.com/router/introduction" />
legacy_subscribe
legacy_subscribe
is experimentally available in SDK 52.
If you're using a third-party provider that doesn't support Expo Router but does support React Navigation via the Linking.subscribe
function for existing projects, you can use legacy_subscribe
as an alternative API.
Using this API is not recommended for new projects or integrations. Its usage is incompatible with Server Side Routing and Static Rendering, and can be challenging to manage while offline or in a low network connectivity environment.