expo-horizon
Expo Horizon: Migrating Expo SDK to Meta Quest
Software Mansion's production guide for adding Meta Quest support to Expo apps using the expo-horizon packages.
This skill does not bundle a copy of the docs. For any task below, always webfetch the linked official README or Meta documentation page to get up-to-date installation steps, plugin options, API surface, and feature matrices. This skill only captures the decision tree, critical rules, and non-obvious gotchas that agents routinely miss.
Decision Tree
What do you need to do?
│
├── Starting from scratch or adding Quest support to an existing Expo app?
│ └── Follow the "Setup Workflow" below (do NOT auto-install location or
│ notifications packages) and webfetch: expo-horizon-core README
│ ├── Install expo-horizon-core
│ ├── Configure the config plugin (horizonAppId, panel size, supportedDevices)
│ ├── Add quest/mobile build scripts
│ ├── Add runtime device detection (isHorizonDevice, isHorizonBuild)
│ └── Detect expo-location / expo-notifications, then ASK before migrating
│
├── Need location services on Quest?
│ └── Webfetch: expo-horizon-location README
│ ├── Replace expo-location with expo-horizon-location
│ ├── Review the feature support matrix
│ └── Guard unsupported calls (heading, geocoding, geofencing, background)
│
├── Need push notifications on Quest?
│ └── Webfetch: expo-horizon-notifications README
│ ├── Replace expo-notifications with expo-horizon-notifications
│ ├── Configure horizonAppId in expo-horizon-core
│ ├── Use getDevicePushTokenAsync (Expo Push Service is not supported)
│ └── Skip badge counts (not supported on Quest)
│
└── Need to build, run, or publish for Quest?
└── Webfetch: expo-horizon-core README (build variants) + Meta docs below
├── Build variants: questDebug, questRelease, mobileDebug, mobileRelease
├── Meta Quest Developer Hub (device management, sideloading)
└── Meta Horizon Store manifest requirements
Setup Workflow (adding Quest support to an existing Expo app)
Follow these steps in order when the user asks to add Meta Quest support. Do not combine steps 2 and 3 into a single install command — the sibling packages require explicit user confirmation.
-
Install and configure
expo-horizon-core.- Run
npx expo install expo-horizon-core. - Ask the user for the config plugin values before writing them. Present all options in a single prompt so the user can paste custom values or accept defaults in one pass. Show each option on its own line with its default in brackets, e.g.:
I'll add the
expo-horizon-coreconfig plugin toapp.json. Please confirm or override each value (press Enter / reply "default" to accept the bracketed default):supportedDevices[quest2|quest3|quest3s] — pipe-separated Quest devices your app supports (required for Meta Horizon Store submission).horizonAppId[empty] — Meta Horizon application ID. Leave empty unless you plan to use push notifications; required byexpo-horizon-notificationsto issue device push tokens.defaultWidth[1024dp] — Default panel width. Leave blank to omit.defaultHeight[640dp] — Default panel height. Leave blank to omit. If you set width/height, make sure your Expoorientationmatches (use"landscape"for wide panels).disableVrHeadtracking[false] — Settrueto omit theandroid.hardware.vr.headtrackingmanifest entry.allowBackup[false] — Meta recommendsfalsefor sensitive data; settrueonly if you need Android backup in the Quest build.
- Only write the plugin config after the user replies. Omit any option the user left blank so the package's own default applies (don't write empty strings for
horizonAppId,defaultWidth, ordefaultHeight). - Add
quest/mobilebuild scripts topackage.json. - Run
npx expo prebuild --clean. - Webfetch the expo-horizon-core README for current option names and defaults before asking — the defaults above can change between releases.
- Run
-
Detect existing location / notification packages. Do NOT install the horizon equivalents yet.
- Read the project's
package.json. - Check
dependenciesanddevDependenciesforexpo-locationandexpo-notifications. - If neither is present, skip the rest of this workflow — the user has no migration to do.
- Read the project's
-
For each detected package, ask the user before migrating.
- If
expo-locationis found, ask:"I found
expo-locationin your project. Do you want me to replace it withexpo-horizon-locationso location works on Meta Quest? (Quest has no GPS, heading, geocoding, or geofencing — unsupported calls will need to be guarded withExpoHorizon.isHorizonDevice.)" - If
expo-notificationsis found, ask:"I found
expo-notificationsin your project. Do you want me to replace it withexpo-horizon-notificationsso push notifications work on Meta Quest? This requires ahorizonAppIdin the core config plugin, uses Meta's push service (not the Expo Push Service), and does not support badge counts orgetExpoPushTokenAsync." - Present the questions together if both packages are present.
- Wait for an explicit answer before running any install or edit for these packages.
- If
-
Only after the user confirms, perform the migration for the approved package(s):
- Install the horizon equivalent (
npx expo install expo-horizon-locationorexpo-horizon-notifications). - Uninstall the original (
npm uninstall expo-locationorexpo-notifications). - Update all
importstatements to the horizon package name. - For notifications: ensure
horizonAppIdis set in theexpo-horizon-coreplugin config and addexpo-horizon-notificationsto the plugins array. - Run
npx expo prebuild --cleanagain. - Webfetch the relevant README (location or notifications) for the full feature support matrix and guard unsupported calls behind
ExpoHorizon.isHorizonDevice.
- Install the horizon equivalent (
-
If the user declines a migration, leave the original package untouched and note in the summary that feature X (location / notifications) will not work on the
questbuild until migrated.
Critical Rules
-
Always install
expo-horizon-corefirst. It is required by all other expo-horizon packages and sets up thequest/mobilebuild flavors that other packages depend on. -
Never auto-install
expo-horizon-locationorexpo-horizon-notifications. When adding Quest support, detect existingexpo-location/expo-notificationsdependencies inpackage.jsonand ask the user whether to migrate each one. Install and configure only the packages the user explicitly approves. See the Setup Workflow above. -
Use
questbuild variants only on Meta Quest devices. RunningquestDebugorquestReleasebuilds on standard Android phones is unsupported and will behave unexpectedly. -
Set
supportedDevicesin the config plugin. This is required for Meta Horizon Store submission. Use pipe-separated values:"quest2|quest3|quest3s". -
Run
npx expo prebuild --cleanafter any plugin config change. The config plugin modifies native project files at prebuild time. Stale native projects will not reflect your changes. -
Replace imports, not just packages. When migrating from
expo-locationorexpo-notifications, update all import statements to use the new package names (expo-horizon-location,expo-horizon-notifications). -
Quest has no GPS, magnetic sensors, or Geocoder. Features like heading, geocoding, reverse geocoding, and geofencing are unavailable on Quest. Guard these calls with
ExpoHorizon.isHorizonDeviceorExpoHorizon.isHorizonBuild. -
Push notifications require
horizonAppId. Without it,getDevicePushTokenAsyncwill not return a valid token on Quest devices. UsegetDevicePushTokenAsync(notgetExpoPushTokenAsync) on Quest; send the returned{ type: 'horizon', data }token to your backend and deliver via Meta's push service. -
isHorizonDevicevsisHorizonBuild. UseisHorizonDevicefor runtime hardware checks (physical Quest detection). UseisHorizonBuildfor build-time feature gating (which native code was compiled in). -
Expo Go is not supported. You must use custom development builds via
npx expo prebuild.
Official References
Always webfetch the raw markdown (raw.githubusercontent.com/...) if the HTML view does not render the source; the raw URL is the source of truth.
| Topic | Official source |
|---|---|
| Repo overview and package list | expo-horizon README |
| Install, config plugin options, runtime API, native module access | expo-horizon-core README |
| Location migration, limitations, feature support matrix | expo-horizon-location README |
| Push notifications migration, token types, feature support matrix | expo-horizon-notifications README |
| Example app wiring for all three packages | expo-horizon example README |
| Panel sizing guidelines (dp values, orientation, letterboxing) | Meta Panel Sizing |
| Meta Horizon Store manifest checklist for publishing | Publish Mobile Manifest |
| Device management, casting, sideloading, ADB | Meta Quest Developer Hub |
| Server-side push delivery via Meta's push service | Horizon OS push notifications |