docs/README.md
This is the public documentation for Expo, its SDK, client, and services (EAS). This documentation is built using Next.js and you can access it online at https://docs.expo.dev/.
[!NOTE] For contributors: Please make sure that you edit the docs in the pages/versions/unversioned for SDK reference if you want your changes to apply to the next SDK version too!
[!TIP] If you are looking for Expo Documentation Writing Style guidelines, please refer Expo Documentation Style Guide.
git clone https://github.com/expo/expo.git
cd into the docs directory and install dependencies with:yarn
3002):yarn run dev
http://localhost:3002, and any changes you make to markdown or JavaScript files will automatically trigger reloads.yarn run export
yarn run export-server
All documentation-related content is inside the pages directory. We write docs in markdown with the help of custom React components that provide additional functionality, such as embedding Snack examples, representing commands inside a terminal component and so on.
The documentation is divided into four main sections:
[!NOTE] We are currently in the process of moving our API documentation to being auto-generated using
expotools'sGenerateDocsAPIDatacommand for some Expo libraries.
Each markdown page can be provided metadata in the heading, distinguished by:
---
metadata: goes here
---
These metadata items include:
title: Title of the page shown as the heading and in search results.description: Description of the page shown in search results and open graph descriptions when the page is shared on social media sites.hideFromSearch: Whether to hide the page from Algolia search results. Defaults to false.hideInSidebar: Whether to hide this page from the sidebar. Defaults to false.hideTOC: Whether to hide the table of contents (appears on the right sidebar). Defaults to false.sidebar_title: The title of the page to display in the sidebar. Defaults to the page title.maxHeadingDepth: The max level of headings shown in Table of Content on the right side. Defaults to 3.isNew: Whether to display the new badge for a page. Commonly used with API pages under Reference. Defaults to false.isDeprecated: Whether to display the deprecated badge for a page. Commonly used with API pages under Reference. Defaults to false.isAlpha: Whether to display the alpha badge for a page. Commonly used with API pages under Reference. Defaults to false.searchRank: A number between 0 and 100 that represents the relevance of a page. This value is mapped to Algolia's record.weight.pageRank property. Higher values indicate higher priority. We set this value to 5 by default, otherwise specified in the frontmatter.searchPosition: The position of a page in the search results. This value is mapped to Algolia's record.weight.position property. Algolia sets this value to 0 by default. Pages with lower values appear higher in the results. We set this value to 50 by default, otherwise specified in the frontmatter.hasVideoLink: To display a video link icon in the sidebar for the page that has a video tutorial link. Defaults to false.cliVersion: The CLI version to display for pages that include the CLI badge. Currently, this field is used for EAS CLI reference page and is populated automatically by yarn run eas-cli-sync.The docs are written with Next.js and TypeScript. If you need to make code changes, follow steps from the To run locally in development mode section, then open a separate terminal and run the TypeScript compiler in watch mode — it will watch your code changes and notify you about errors.
yarn watch
When you are done, you should run prettier to format your code. Also, don't forget to run tests and linter before committing your changes.
yarn prettier
yarn test
yarn lint
When you are done writing or editing docs, run the following script to lint your docs for style and grammar based on Expo's writing style guide:
yarn run lint-prose
We use Vale to lint our docs.
For exceptional cases, you can switch off the prose linter for a specific line or block of text by adding by using a comment delimiter:
This is some text that will be ignored by Vale.
<details> <summary>Alternative: Use Vale with VS Code</summary>[!NOTE] Ideally, to add new services or features, the Vale lint rules should upgrade accordingly when there's a pattern. If you want to update a rule, see the .vale directory for already established rules.
Alternatively, you can use Vale with VS Code. You need to:
Open the doc file (*.mdx) that you are working on and you'll may see suggested lines (yellow squiggly) in VS Code editor.
We use two layers of redirects:
public/_redirects using the Cloudflare Pages redirect format (source_path destination_path status_code, one rule per line). These are 301 permanent redirects for simple 1:1 path mappings and SEO-friendly behavior.common/client-redirects.ts that run on the 404 page for more complex rules (for example, stripping .html, version fallbacks) and to catch cases where server-side redirects do not apply (local/dev/preview or missed mappings).We currently do two client-side redirects, using meta tags with http-equiv="refresh":
/ -> /versions/latest//versions -> /versions/latestThis works by loading a page and then immediately navigating, which can confuse assistive tech (announced content disappears, focus resets) and gives developers less control. Treat this as a fallback and prefer server-side redirects or the 404-based client rules when possible.
We use Algolia as the main search results provider for our docs. This is set up in the @expo/styleguide library, which provides a universal search component that is used in the docs, expo.dev, and EAS dashboard.
Besides the query, the results are also filtered based on the version tag. This tag represents the user's current location. The tag is set in the components/DocumentationPage.tsx head.
Inside @expo/styleguide library, you can see the facetFilters set to [['version:none', 'version:{version}']] in packages/search-ui/src/components/CommandMenu.tsx. Translated to English, this means - search on all pages where version is none, or the currently selected version.
nonev51.0.0 or v50.0.0)hideFromSearch: true frontmatter entry don't have the version tagCurrently, the base results for Expo docs are combined with other results from multiple sources, such as:
ui/components/Search/expoEntries.tsYou can't have curly brace without quotes: `{}` -> {}.
The docs are deployed automatically via a GitHub Action each time a PR with docs changes is merged to main.
If you need to link from one MDX file to another, use the static/full path to this file (avoid relative links):
/introduction/expo/guides/errors/#tracking-javascript-errorsValidate all current links by running yarn lint-links script.
When we release a new SDK, we copy the unversioned directory, and rename it to the new version. Latest version of docs is read from package.json so make sure to update the version key there as well.
Make sure to also grab the upgrade instructions from the release notes blog post and put them in upgrading-expo-sdk-walkthrough.mdx.
The versions directory is listed on server start to find all available versions. The routes and navbar contents are automatically inferred from the directory structure within versions.
Since the navbar is automatically generated from the directory structure, the default ordering of the links under each section is alphabetical. However, for many sections, this is not ideal UX. So, if you want to override the alphabetical ordering, manipulate page titles in constants/navigation.js.
The API reference docs are generated from the TypeScript source code.
This section walks through the process of updating documentation for an Expo package. Throughout this document, we will assume we want to update TypeDoc definitions of property inside expo-constants as an example.
For more information on how TypeDoc/JSDoc parses comments, see Doc comments in TypeDoc documentation.
Before proceeding, make sure you:
direnv and run direnv allow at the root of the expo/ repo.et (Expotools) command locally.Once you have made sure the development setup is ready, proceed to the next section:
# Navigate to expo-constants package directory inside expo/ repo
cd expo/packages/expo-constants
Then, open .ts file in your code editor/IDE where you want to make changes/updates.
Start the TypeScript build compilation in watch mode using yarn build in the terminal window.
Make the update. For example, we want to update the TypeDoc description of expoConfig property
expoConfig property. It has a current description as shown below:/**
* The standard Expo config object defined in `app.json` and `app.config.js` files. For both
* classic and modern manifests, whether they are embedded or remote.
*/
expoConfig: ExpoConfig | null;
In the above example, let's fix the typo by changing confg to config:
/**
* The standard app config object defined in `app.json` and `app.config.js` files. For both
* classic and modern manifests, whether they are embedded or remote.
*/
expoConfig: ExpoConfig | null;
Ctrl + C from the keyboard.[!IMPORTANT]
If you are fixing issues in package's reference or after an SDK version is released, make sure to only update the
unversionedreference of that package. This way the changes will be reflected in the next SDK version from themainbranch. Updating the reference for a specific SDK version requires updating that SDK's branch (see collapsible below) and SDK team decides cherry-picking changes for the specific SDK branch (after an SDK version has been released).
In the terminal window and run the following command with to generate the JSON data file for the package (which is stored at the location expo/docs/public/static/data/[SDK-VERSION])
unversioned:et generate-docs-api-data --packageName expo-constants
#### NOTE ####
# To update a specific SDK reference, run the command by mentioning the SDK version
et gdad -p expo-constants --sdk 54
# For more information about et command, run: et gdad --help
Why update unversioned docs? If these are new changes/updates, apply them to unversioned to make sure that those changes are part of the next SDK version.
Now, in the terminal window, navigate to expo/docs repo and run the command yarn run dev to see the changes applied
When you need to update versioned documentation data late in the SDK lifecycle, follow these steps:
Ensure the related code change exists on the main branch and sdk * branch.
Make branch changes:
sdk * branch and make your changeset gdad -p expo-library --sdk 52, where expo-library is the library you want to update and 52 is the SDK version.git stash)main branch:main branchet gdad -p expo-libraryexpo-library.json to the main changesetAfter making changes, when you are opening the PR, consider adding <!-- disable:changelog-checks --> in the PR description if the changes you are making are docs-related changes (such as updating the field description or fixing a typo, and so on).
This will make sure that the ExpoBot on GitHub will not complain about updating the package's changelog (some of these changes, as described above, are not worth mentioning in the changelog).
Some of the packages have documentation spread over multiple pages. For example, expo-sensors package has a separate overview page in Expo Sensors reference, and rest of the information is separated into components such as, Accelerometer, Gyroscope, Magnetometer, and more. For such packages, always make sure to check the name of the package for et command.
To render the app config properties table, we currently store a local copy of the appropriate version of the schema.
If the schema is updated, to sync and rewrite our local copy, run yarn run schema-sync <SDK version integer> or yarn run schema-sync unversioned.
You can add images and assets to the public/static directory. They'll be served by the production and staging servers at static.
ffmpeg (brew install ffmpeg)ffmpeg -i your-video-name.mov -vcodec h264 -acodec mp2 your-video-name.mp4 to convert to mp4.ffmpeg -i your-video.mp4 -filter:v scale="1280:trunc(ow/a/2)*2" your-video-smaller.mp4public/static/videos and use it in your docs page MDX like this:import { ContentSpotlight } from '~/ui/components/ContentSpotlight';
// Change the path to point to the relative path to your video from within the `static/videos` directory
<ContentSpotlight file="guides/color-schemes.mp4" />;
To reference a video from Expo's YouTube channel, use the VideoBoxLink component. This component is imported from ~/ui/components/VideoBoxLink.
import { VideoBoxLink } from '~/ui/components/VideoBoxLink';
<VideoBoxLink videoId="Gk7RHDWsLsQ" title="Required title" description="Optional" />;
| Param | Description |
|---|---|
videoId | Required. The ID of the video from YouTube. You can find this in the URL of the video. For example, in https://www.youtube.com/watch?v=Gk7RHDWsLsQ, the ID is Gk7RHDWsLsQ. |
| title | Required. The title of the video. |
| description | Optional. The description of the video. |
Code blocks are a great way to add code snippets to our docs. We leverage the usual code block Markdown syntax, but it's expanded to support code block titles and additional params.
<!-- prettier-ignore --> ```js
// Your code goes in here
```
```js myFile.js
// Your code goes in here
```
```js Title for a code block
// Your code goes in here
```
```js myFile.js|collapseHeight=600
// Your code goes in here
```
```js collapseHeight=200
// Your code goes in here
```
| Param | Type | Description |
|---|---|---|
collapseHeight | number | The custom height that the code block uses to collapse automatically. The default value is 408 and is applied unless the collapseHeight param has been specified. |
Snacks are a great way to add instantly-runnable examples to our docs. The SnackInline component can be imported to any markdown file, and used like this:
import SnackInline from '~/components/plugins/SnackInline';
<SnackInline label='My Example Label' dependencies={['array of', 'packages', 'this Snack relies on']}>
```js
// All your code goes in here
// You can use:
/* @info Some text goes here */
const myVariable = SomeCodeThatDoesStuff();
/* @end */
// to create hoverable-text, which reveals the text inside of `@info` onHover.
// You can use:
/* @hide Content that is still shown, like a preview. */
Everything in here is hidden in the example Snack until
you open it in snack.expo.dev
/* @end */
// to shorten the length of code block shown in our docs.
// Hidden code will still be present when opening in Snack or using "Copy" action.
```
</SnackInline>
Sometimes it's useful to show multiple ways of doing something, for instance, maybe you'd like to have an example using a React class component, and also an example of a functional component. The Tabs plugin is useful for this, and this is how you'd use it in a markdown file:
import { Tabs, Tab } from '~/ui/components/Tabs';
<Tabs>
<Tab label="Add 1 One Way">
```js
addOne = async x => {
/* @info This text will be shown onHover */
return x + 1;
/* @end */
};
```
</Tab>
<Tab label="Add 1 Another Way">
```js
addOne = async x => {
/* @info This text will be shown onHover */
return x++;
/* @end */
};
```
</Tab>
</Tabs>
[!NOTE] The components should not be indented or they will not be parsed correctly.
To ignore a page from the search result, use hideFromSearch: true on that page. This removes the <meta name="docsearch:version"> tag from that page and filters it from our facet-based search.
Please note that hideFromSearch only prevents the page from showing up in the internal docs search (Algolia). The page will still show up in search engine results like Google. To hide a page from search engine results, you need to edit the sitemap that is generated via our Next.js config (next.config.js).
Certain directories are excluded from the sidebar to prevent it from getting too long and unnavigable. You can find a list of these directories, and add new ones, in constants/navigation.js under hiddenSections.
If you just want to hide a single page from the sidebar, set hideInSidebar: true in the page metadata.
Terminal component for shell commands snippetsWhenever shell commands are used or referred, use Terminal component to make the code snippets copy/pasteable. This component can be imported into any markdown file.
| Option | Type | Description |
|---|---|---|
cmd | string[] | Required. Lines to render. Use $ to mark commands, # for comments, and empty strings for spacing. |
cmdCopy | string | Optional. Overrides the auto-generated copy text. Helpful when multiple commands should be chained (for example with &&) or when you want to control the copied content. |
title | string | Optional. Overrides the default header label of "Terminal". |
browserAction | { href: string; label: string } | Optional. Adds a launch button in the header that opens the link in a new tab—ideal for flows that continue in a web UI. |
hideOverflow | boolean | Optional. Prevents horizontal scrollbars when you prefer the content to clip instead of scroll. |
className | string | Optional. Additional utility classes for adjusting layout or spacing around the snippet. |
import { Terminal } from '~/ui/components/Snippet';
<Terminal cmd={['$ npx expo install package']} />
<Terminal
cmd={['# Create a new Expo project', '$ npx create-expo-app --template bare-minimum', '']}
cmdCopy="npx create-expo-app --template bare-minimum"
/>
<Terminal
className="mt-6"
hideOverflow
cmd={[
'$ npx expo config --json',
'# Output is trimmed visually because hideOverflow is enabled.',
]}
/>
<Terminal
title="Deploy website with EAS"
cmd={['$ npx eas-cli deploy']}
browserAction={{ href: 'https://expo.dev/eas', label: 'Open in expo.dev' }}
/>
Four different types of callouts can be used with markdown syntax for > ... blockquote. Each callout represents a purpose.
> Normal callout that doesn't demand much attention but is required to add as a note.
> **info** Callout that is informative and demands attention is required to add as a note or a tip.
> **warning** Callout that is used for warnings and deprecation messages.
> **error** Callout that is used for errors and breaking changes or deprecated changes in the archive.
> **important** Callout that is used for presenting important information about state of package, service or tool.
All docs pages are automatically updated with the last update date of the file based on their Git commit history. This information is reflected in the footer of a docs page with Last updated on ....
If you need to add the date manually, add modificationDate to the frontmatter of the .mdx file. For example:
---
modificationDate: April 8th, 2024
---
This pattern is used for some of the pages where we manually update the modification date, such as Build server infrastructure.
Docs areas that are excluded or do not include an updated date are SDK API references and Tutorials sections under Learn.
Please commit any sizeable diffs that are the result of prettier separately to make reviews as easy as possible.
If you have a code block using /* @info */ highlighting, use `` on the block and take care to preview the block in the browser to ensure that the indentation is correct - the highlighting annotation will sometimes swallow newlines.
For procedural guides, use Step component:
import { Step } from '~/ui/components/Step';
<Step label="1">
This is some text.
</Step>