xref: /expo/docs/README.md (revision fc13df4e)
1# Expo Documentation
2
3This is the public documentation for **Expo**, its SDK, client, and services, like **EAS**.
4
5This documentation is built using Next.js and you can access it online at https://docs.expo.dev/.
6
7> **Note** **Contributors:** Please make sure that you edit the docs in the `pages/versions/unversioned` directory if you want your changes to apply to the next SDK version too!
8
9> **Note**
10> If you are looking for Expo Documentation Writing Style guidelines, please refer [Expo Documentation Style Guide](https://github.com/expo/expo/blob/main/guides/Expo%20Documentation%20Writing%20Style%20Guide.md).
11
12## Running Locally
13
14Download the copy of this repository.
15
16```sh
17git clone https://github.com/expo/expo.git
18```
19
20Then `cd` into the `docs` directory and install dependencies with:
21
22```sh
23yarn
24```
25
26Then you can run the app with (make sure you have no server running on port `3002`):
27
28```sh
29yarn run dev
30```
31
32Now the documentation is running at http://localhost:3002, and any changes you make to markdown or JavaScript files will automatically trigger reloads.
33
34### To run locally in production mode
35
36```sh
37yarn run export
38yarn run export-server
39```
40
41## Editing Docs Content
42
43You can find the content source of the documentation inside the `pages/` directory. Documentation is mostly written in markdown with the help of some React components (for Snack embeds, etc). Our API documentation can all be found under `pages/versions/`; we keep separate versions of the documentation for each SDK version currently supported in Expo Go, see ["A note about versioning"](#a-note-about-versioning) for more info. The routes and navbar are automatically inferred from the directory structure within `versions`.
44
45> **Note**
46> We are currently in the process of moving our API documentation to being auto-generated using `expotools`'s `GenerateDocsAPIData` command.
47
48Each markdown page can be provided metadata in the heading, distinguished by:
49
50```
51---
52metadata: goes here
53---
54```
55
56These metadata items include:
57
58- `title`: Title of the page shown as the heading and in search results
59- `hideFromSearch`: Whether to hide the page from Algolia search results. Defaults to `false`.
60- `hideInSidebar`: Whether to hide this page from the sidebar. Defaults to `false`.
61- `hideTOC`: Whether to hide the table of contents (appears on the right sidebar). Defaults to `false`.
62- `sidebar_title`: The title of the page to display in the sidebar. Defaults to the page title.
63
64### Editing Code
65
66The docs are written with Next.js and TypeScript. If you need to make code changes, follow steps from the [Running locally](#running-locally) 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.
67
68```sh
69yarn watch
70```
71
72When you are done, you should run _prettier_ to format your code. Also, don't forget to run tests and linter before committing your changes.
73
74```sh
75yarn prettier
76yarn test
77yarn lint
78```
79
80## Redirects
81
82### Server-side redirects
83
84These redirects are limited in their expressiveness - you can map a path to another path, but no regular expressions or anything are supported. See client-side redirects for more of that. Server-side redirects are re-created on each run of **deploy.sh**.
85
86We currently do two client-side redirects, using meta tags with `http-equiv="refresh"`:
87
88- `/` -> `/versions/latest/`
89- `/versions` -> `/versions/latest`
90
91This method is not great for accessibility and should be avoided where possible.
92
93### Client-side redirects
94
95Use these for more complex rules than one-to-one path-to-path redirect mapping. For example, we use client-side redirects to strip the `.html` extension off, and to identify if the request is for a version of the documentation that we no longer support.
96
97You can add your own client-side redirect rules in `common/error-utilities.ts`.
98
99## Algolia Docsearch
100
101We use Algolia Docsearch as the search engine for our docs. Right now, it's searching for any keywords with the proper `version` tag based on the current location. This is set in the `components/DocumentationPage` header.
102
103In `components/plugins/AlgoliaSearch`, you can see the `facetFilters` set to `[['version:none', 'version:{currentVersion}']]`. Translated to English, this means "Search on all pages where `version` is `none`, or the currently selected version.".
104
105- All unversioned pages use the version tag `none`.
106- All versioned pages use the SDK version (e.g. `v40.0.0` or `v39.0.0`).
107- All `hideFromSearch: true` pages don't have the version tag.
108
109## Quirks
110
111- You can't have curly brace without quotes: \`{}\` -> `{}`
112- Make sure to leave an empty newline between a table and following content
113
114## A note about versioning
115
116Expo's SDK is versioned so that apps made on old SDKs are still supported
117when new SDKs are released. The website documents previous SDK versions too.
118
119Version names correspond to directory names under `versions`.
120
121`unversioned` is a special version for the next SDK release. It is not included in production output. Additionally, any versions greater than the package.json `version` number are not included in production output, so that it's possible to generate, test, and make changes to new SDK version docs during the release process.
122
123`latest` is an untracked folder which duplicates the contents of the folder matching the version number in **package.json**.
124
125Sometimes you want to make an edit in version `X` and have that edit also
126be applied in versions `Y, Z, ...` (say, when you're fixing documentation for an
127API call that existed in old versions too). You can use the
128`./scripts/versionpatch.sh` utility to apply your `git diff` in one version in
129other versions. For example, to update the docs in `unversioned` then apply it
130on `v8.0.0` and `v7.0.0`, you'd do the following after editing the docs in
131`unversioned` such that it shows up in `git diff`:
132
133`./scripts/versionpatch.sh unversioned v8.0.0 v7.0.0`
134
135Any changes in your `git diff` outside the `unversioned` directory are ignored
136so don't worry if you have code changes or such elsewhere.
137
138## Deployment
139
140The docs are deployed automatically via a GitHub Action each time a PR with docs changes is merged to `main`.
141
142## How-tos
143
144## Internal linking
145
146If you need to link from one MDX file to another, please use the path-reference to this file including extension.
147This allows us to automatically validate these links and see if the file and/or headers still exists.
148
149- from: `tutorial/button.md`, to: `/workflow/guides/` -> `../workflow/guides.md`
150- from: **index.md**, to: `/guides/errors/#tracking-js-errors` -> `./guides/errors.md#tracking-js-errors` (or without `./`)
151
152You can validate all current links by running `yarn lint-links`.
153
154### Updating latest version of docs
155
156When 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.
157
158Make sure to also grab the upgrade instructions from the release notes blog post and put them in `upgrading-expo-sdk-walkthrough.md`.
159
160That's all you need to do. 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`.
161
162Because 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.
163So, if you wish to override the alphabetical ordering, manipulate page titles in **constants/navigation.js**.
164
165### Syncing app.json / app.config.js with the schema
166
167To render the app.json / app.config.js properties table, we currently store a local copy of the appropriate version of the schema.
168
169If the schema is updated, in order to sync and rewrite our local copy, run `yarn run schema-sync <SDK version integer>` or `yarn run schema-sync unversioned`.
170
171### Importing from the React Native docs
172
173You can import the React Native docs in an automated way into these docs.
174
1751. Update the react-native-website submodule here
1762. `yarn run import-react-native-docs`
177
178This will write all the relevant RN doc stuff into the unversioned version directory.
179You may need to tweak the script as the source docs change; the script hackily translates between the different forms of markdown that have different quirks.
180
181The React Native docs are actually versioned but we currently read off of main.
182
183### Adding Images and Assets
184
185You can add images and assets to the `public/static` directory. They'll be served by the production and staging servers at `/static`.
186
187#### Adding videos
188
189- Record the video using QuickTime
190- Install `ffmpeg` (`brew install ffmpeg`)
191- Run `ffmpeg -i your-video-name.mov -vcodec h264 -acodec mp2 your-video-name.mp4` to convert to mp4.
192- If the width of the video is larger than ~1200px, then run this to shrink it: `ffmpeg -i your-video.mp4 -filter:v scale="1280:trunc(ow/a/2)*2" your-video-smaller.mp4`
193- Put the video in the appropriate location in `public/static/videos` and use it in your docs page MDX like this:
194
195```js
196import Video from '~/components/plugins/Video';
197
198// Change the path to point to the relative path to your video from within the `static/videos` directory
199<Video file="guides/color-schemes.mp4" />;
200```
201
202### Inline Snack examples
203
204Snacks 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:
205
206<!-- prettier-ignore -->
207```jsx
208import SnackInline from '~/components/plugins/SnackInline';
209
210<SnackInline label='My Example Label' dependencies={['array of', 'packages', 'this Snack relies on']}>
211
212// All your JavaScript code goes in here
213
214// You can use:
215/* @info Some text goes here */
216  const myVariable = SomeCodeThatDoesStuff();
217/* @end */
218// to create hoverable-text, which reveals the text inside of `@info` onHover.
219
220// You can use:
221/* @hide Content that is still shown, like a preview. */
222  Everything in here is hidden in the example Snack until
223  you open it in snack.expo.dev
224/* @end */
225// to shorten the length of the Snack shown in our docs. Common example are hiding useless code in examples, like StyleSheets
226
227</SnackInline>
228```
229
230### Embedding multiple options of code
231
232Sometimes 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.
233The `Tabs` plugin is really useful for this, and this is how you'd use it in a markdown file:
234
235<!-- prettier-ignore -->
236```jsx
237import { Tab, Tabs } from '~/components/plugins/Tabs';
238
239<Tabs>
240<Tab label="Add 1 One Way">
241
242    addOne = async x => {
243    /* @info This text will be shown onHover */
244    return x + 1;
245    /* @end */
246    };
247
248</Tab>
249<Tab label="Add 1 Another Way">
250
251    addOne = async x => {
252    /* @info This text will be shown onHover */
253    return x++;
254    /* @end */
255    };
256
257</Tab>
258</Tabs>
259```
260
261n.b. The components should not be indented or they will not be parsed correctly.
262
263### Excluding pages from DocSearch
264
265To 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.
266
267Please 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.
268For a page to be hidden even from search engine results, you need to edit the sitemap that is generated via our Next.js config (**next.config.js**).
269
270### Excluding directories from the sidebar
271
272Certain directories are excluded from the sidebar in order 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`.
273
274If you just want to hide a single page from the sidebar, set `hideInSidebar: true` in the page metadata.
275
276### Use `Terminal` component for shell commands snippets
277
278Whenever shell commands are used or referred, use `Terminal` component to make the code snippets copy/pasteable. This component can be imported in any markdown file.
279
280```jsx
281import { Terminal } from '~/ui/components/Snippet';
282
283// for single command and one prop
284<Terminal cmd={["$ npx expo install package"]} />
285
286// for multiple commands
287
288<Terminal cmd={[
289  "# Create a new native project",
290  "$ npx create-expo-app --template bare-minimum",
291  "",
292  "# If you don’t have expo-cli yet, get it",
293  "$ npm i -g expo-cli",
294  "",
295]} cmdCopy="npx create-expo-app --template bare-minimum && npm i -g expo-cli" />
296```
297
298### Prettier
299
300Please commit any sizeable diffs that are the result of `prettier` separately to make reviews as easy as possible.
301
302If you have a code block using `/* @info */` highlighting, use `{/* prettier-ignore */}` 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.
303
304## TODOs:
305
306- Handle image sizing in imports better
307- Make Snack embeds work; these are marked in some of the React Native docs but they are just imported as plain JS code blocks
308