xref: /expo/docs/README.md (revision bb5069cd)
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- `description`: Description of the page shown in search results and open graph descriptions when the page is shared on social media sites.
60- `hideFromSearch`: Whether to hide the page from Algolia search results. Defaults to `false`.
61- `hideInSidebar`: Whether to hide this page from the sidebar. Defaults to `false`.
62- `hideTOC`: Whether to hide the table of contents (appears on the right sidebar). Defaults to `false`.
63- `sidebar_title`: The title of the page to display in the sidebar. Defaults to the page title.
64- `maxHeadingDepth`: The max level of headings shown in Table of Content on the right side. Defaults to `3`.
65
66### Editing Code
67
68The 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.
69
70```sh
71yarn watch
72```
73
74When you are done, you should run `prettier` to format your code. Also, don't forget to run tests and linter before committing your changes.
75
76```sh
77yarn prettier
78yarn test
79yarn lint
80```
81
82## Redirects
83
84### Server-side redirects
85
86These 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**.
87
88We currently do two client-side redirects, using meta tags with `http-equiv="refresh"`:
89
90- `/` -> `/versions/latest/`
91- `/versions` -> `/versions/latest`
92
93This method is not great for accessibility and should be avoided where possible.
94
95### Client-side redirects
96
97Use 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.
98
99You can add your own client-side redirect rules in `common/error-utilities.ts`.
100
101## Search
102
103We use Algolia as a main search results provider for our docs. Besides the query, results are also filtered based on the `version` tag which represents the user current location. The tag set in the `components/DocumentationPage.tsx` head.
104
105In `ui/components/CommandMenu/utils.ts`, you can see the `facetFilters` set to `[['version:none', 'version:{version}']]`. Translated to English, this means - search on all pages where `version` is `none`, or the currently selected version. Here are the rules we use to set this tag:
106
107- all unversioned pages use the version tag `none`,
108- all versioned pages use the SDK version (e.g. `v46.0.0` or `v47.0.0`),
109- all pages with `hideFromSearch: true` frontmatter entry don't have the version tag.
110
111Currently, the base results for Expo docs are combined with other results from multiple sources, like:
112
113- manually defined paths for Expo dashboard located in `ui/components/CommandMenu/expoEntries.ts`,
114- public Algolia index for React Native website,
115- React Native directory public API, see the directory [README.md](https://github.com/react-native-community/directory#i-dont-like-your-website-can-i-hit-an-api-instead-and-build-my-own-better-stuff) for more details.
116
117## Quirks
118
119- You can't have curly brace without quotes: \`{}\` -> `{}`
120
121## Deployment
122
123The docs are deployed automatically via a GitHub Action each time a PR with docs changes is merged to `main`.
124
125## How-tos
126
127### Internal linking
128
129If you need to link from one MDX file to another, please use the static/full path to this file (avoid relative links):
130
131- from: **tutorial/button.mdx**, to: **introduction/expo.mdx** -> `/introduction/expo`
132- from: **index.mdx**, to: **guides/errors.mdx#tracking-js-errors** -> `/guides/errors/#tracking-javascript-errors`
133
134You can validate all current links by running `yarn lint-links` script.
135
136### Updating latest version of docs
137
138When 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.
139
140Make sure to also grab the upgrade instructions from the release notes blog post and put them in **upgrading-expo-sdk-walkthrough.mdx**.
141
142That'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`.
143
144Because 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.
145So, if you wish to override the alphabetical ordering, manipulate page titles in **constants/navigation.js**.
146
147### Updating API reference docs
148
149The API reference docs are generated from the TypeScript source code.
150
151This 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.
152
153> For more information on how TypeDoc/JSDoc parses comments, see [**Doc comments in TypeDoc documentation**](https://typedoc.org/guides/doccomments/).
154
155#### Prerequisites
156
157Before proceeding, make sure you:
158
159- have [**expo/**](https://github.com/expo/expo) repo cloned on your machine
160  - make sure to [install `direnv`](https://direnv.net/docs/installation.html) and run `direnv allow` at the root of the **expo/** repo.
161- have gone through the steps mentioned in [**"Download and Setup" in the contribution guideline**](https://github.com/expo/expo/blob/main/CONTRIBUTING.md#-download-and-setup).
162- can run **expo/docs** app **[locally](https://github.com/expo/expo/tree/main/docs#running-locally)**.
163- can run [`et` (Expotools)](https://github.com/expo/expo/blob/main/tools/README.md) command locally.
164
165Once you have made sure the development setup is ready, proceed to the next section:
166
167#### Step 1: Update the package’s TypeDoc
168
169- After you have identified which package docs you want to update, open a terminal window and navigate to that package’s directory. For example:
170
171```shell
172# Navigate to expo-constants package directory inside expo/ repo
173cd expo/packages/expo-constants
174```
175
176- Then, open **.ts** file in your code editor/IDE where you want to make changes/updates.
177- Start the TypeScript build compilation in watch mode using `yarn build` in the terminal window.
178- Make the update. For example, we want to update the TypeDoc description of [`expoConfig` property](https://docs.expo.dev/versions/latest/sdk/constants/#nativeconstants)
179
180  - Inside the **src/** directory, open **Constants.types.ts** file.
181  - Search for `expoConfig` property. It has a current description as shown below:
182
183  ```ts
184  /**
185   * The standard Expo confg object defined in `app.json` and `app.config.js` files. For both
186   * classic and modern manifests, whether they are embedded or remote.
187   */
188  expoConfig: ExpoConfig | null;
189  ```
190
191- In the above example, let’s fix the typo by changing `confg` to `config`:
192
193```ts
194/**
195 * The standard Expo config object defined in `app.json` and `app.config.js` files. For both
196 * classic and modern manifests, whether they are embedded or remote.
197 */
198expoConfig: ExpoConfig | null;
199```
200
201- Before moving to the next step, make sure to exit the "watch mode" by pressing `Ctrl + C` from the keyboard.
202
203#### Step 2: Apply TypeDoc updates to expo/docs repo
204
205In 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]`)
206
207- Read the **NOTE** in the below snippet for updating the docs for `unversioned`:
208
209```shell
210et generate-docs-api-data --packageName expo-constants --sdk 47
211
212#### NOTE ####
213# To update unversioned docs, run the command without mentioning the SDK version
214et gdad -p expo-constants
215
216# For more information about et command, run: et gdad --help
217```
218
219**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.
220
221#### Step 3: See the changes in the docs repo
222
223Now, in the terminal window, navigate to **expo/docs** repo and run the command `yarn run dev` to see the changes applied
224
225- Open [http://localhost:3002/](http://localhost:3002/) in the browser and go to the API doc to see the changes you have made. Make sure to select the right SDK version to see the changes in the left sidebar.
226
227#### Tips
228
229##### Disabling changelog
230
231After 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, etc.)
232
233This 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).
234
235##### Using the correct package name
236
237Some of the packages have documentation spread over multiple pages. For example, `expo-av` package has a separate base interface, and some of the information is separated into `Audio` and `Video` components. For such packages, always make sure to check the [name of the package](https://github.com/expo/expo/blob/main/tools/src/commands/GenerateDocsAPIData.ts#L24) for `et` command.
238
239### Syncing app.json / app.config.js with the schema
240
241To render the app.json / app.config.js properties table, we currently store a local copy of the appropriate version of the schema.
242
243If 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`.
244
245### Adding Images and Assets
246
247You can add images and assets to the `public/static` directory. They'll be served by the production and staging servers at `/static`.
248
249#### Adding videos
250
251- Record the video using QuickTime
252- Install `ffmpeg` (`brew install ffmpeg`)
253- Run `ffmpeg -i your-video-name.mov -vcodec h264 -acodec mp2 your-video-name.mp4` to convert to mp4.
254- 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`
255- Put the video in the appropriate location in `public/static/videos` and use it in your docs page MDX like this:
256
257```js
258import Video from '~/components/plugins/Video';
259
260// Change the path to point to the relative path to your video from within the `static/videos` directory
261<Video file="guides/color-schemes.mp4" />;
262```
263
264### Inline Snack examples
265
266Snacks 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:
267
268<!-- prettier-ignore -->
269```jsx
270import SnackInline from '~/components/plugins/SnackInline';
271
272<SnackInline label='My Example Label' dependencies={['array of', 'packages', 'this Snack relies on']}>
273
274// All your JavaScript code goes in here
275
276// You can use:
277/* @info Some text goes here */
278  const myVariable = SomeCodeThatDoesStuff();
279/* @end */
280// to create hoverable-text, which reveals the text inside of `@info` onHover.
281
282// You can use:
283/* @hide Content that is still shown, like a preview. */
284  Everything in here is hidden in the example Snack until
285  you open it in snack.expo.dev
286/* @end */
287// to shorten the length of the Snack shown in our docs. Common example are hiding useless code in examples, like StyleSheets
288
289</SnackInline>
290```
291
292### Embedding multiple options of code
293
294Sometimes 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.
295The `Tabs` plugin is really useful for this, and this is how you'd use it in a markdown file:
296
297<!-- prettier-ignore -->
298```jsx
299import { Tabs, Tab } from '~/ui/components/Tabs';
300
301<Tabs>
302<Tab label="Add 1 One Way">
303
304    addOne = async x => {
305    /* @info This text will be shown onHover */
306    return x + 1;
307    /* @end */
308    };
309
310</Tab>
311<Tab label="Add 1 Another Way">
312
313    addOne = async x => {
314    /* @info This text will be shown onHover */
315    return x++;
316    /* @end */
317    };
318
319</Tab>
320</Tabs>
321```
322
323**Note:** The components should not be indented or they will not be parsed correctly.
324
325### Excluding pages from DocSearch
326
327To 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.
328
329Please 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.
330For 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**).
331
332### Excluding directories from the sidebar
333
334Certain 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`.
335
336If you just want to hide a single page from the sidebar, set `hideInSidebar: true` in the page metadata.
337
338### Use `Terminal` component for shell commands snippets
339
340Whenever 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.
341
342```jsx
343import { Terminal } from '~/ui/components/Snippet';
344
345// for single command and one prop
346<Terminal cmd={["$ npx expo install package"]} />
347
348// for multiple commands
349
350<Terminal cmd={[
351  "# Create a new native project",
352  "$ npx create-expo-app --template bare-minimum",
353  "",
354  "# If you don’t have expo-cli yet, get it",
355  "$ npm i -g expo-cli",
356  "",
357]} cmdCopy="npx create-expo-app --template bare-minimum && npm i -g expo-cli" />
358```
359
360### Prettier
361
362Please commit any sizeable diffs that are the result of `prettier` separately to make reviews as easy as possible.
363
364If 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.
365