Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/atolye15/crna-recipe

Step-by-step guide to bootstrap a React Native app from scratch
https://github.com/atolye15/crna-recipe

react react-native storybook typescript

Last synced: about 2 months ago
JSON representation

Step-by-step guide to bootstrap a React Native app from scratch

Awesome Lists containing this project

README

        

# React Native App Creation Recipe

This is a step-by-step guide to create React Native app for Atolye15 projects. You can review [React Native App Starter](https://github.com/atolye15/crna-starter) project to see how your application looks like when all steps followed.

You will get an application which has;

* TypeScript
* Linting
* Formatting
* Testing
* CI/CD
* Storybook

## Table of Contents

* [Step 1: Installing the React Native CLI](#step-1-installing-the-react-native-cli)
* [Step 2: Creating a new app](#step-2-creating-a-new-app)
* [Step 3: Make TypeScript more strict](#step-3-make-typescript-more-strict)
* [Step 4: Installing Prettier](#step-4-installing-prettier)
* [Step 5: Installing ESLint](#step-5-installing-eslint)
* [Step 6: Setting up our test environment](#step-6-setting-up-our-test-environment)
* [Step 7: Setting up config variables](#step-7-setting-up-config-variables)
* [Step 8: Organizing Folder Structure](#step-8-organizing-folder-structure)
* [Step 9: Adding Storybook](#step-9-adding-storybook)
* [Step 10: Adding CircleCI config](#step-10-adding-circleci-config)
* [Step 11: Github Settings](#step-11-github-settings)
* [Step 12 Final Touches](#step-12-final-touches)
* [Step 13: Starting to Development 🎉](#step-13-starting-to-development-)
* [Bonus: Npm Script Aliases](#bonus-npm-script-aliases)

## Step 1: Installing the React Native CLI

First of all, we need to install the React Native command line interface.

```bash
yarn global add react-native-cli
```

## Step 2: Creating a new app

Use the React Native command line interface to generate a new React Native project called "AwesomeProject":

```bash
react-native init AwesomeProject --template typescript
```

> _NOTE: Project name should be alphanumeric!_

## Step 3: Make TypeScript more strict

We want to keep type safety as strict as possibble. In order to do that, we update `tsconfig.json` with the settings below. Also we prefer to disable `isolatedModules` and activate `skipLibCheck`.

```json
"strict": true,
"noImplicitAny": true,
"noImplicitReturns": true,
"skipLibCheck": true,
"isolatedModules": false,
```

## Step 4: Installing Prettier

We want to format our code automatically. So, we need to install Prettier.

```bash
yarn add prettier --dev
```

```json
// .prettierrc

{
"printWidth": 100,
"singleQuote": true,
"trailingComma": "all"
}
```

Also, we want to enable format on save on VSCode.

> _React Native CLI adds `.vscode` to `.gitignore`, but we prefer not to ignore. So remove it from `.gitignore`._

```json
// .vscode/settings.json

{
"editor.formatOnSave": true
}
```

Finally, we update `package.json` with related format scripts.

```json
"format": "prettier --write 'src/**/*.{ts,tsx}'",
"format:check": "prettier -c 'src/**/*.{ts,tsx}'"
```

## Step 5: Installing ESLint

We want to have consistency in our codebase and also want to catch mistakes. So, we need to install ESLint.

```bash
yarn add eslint eslint-config-airbnb eslint-config-prettier eslint-plugin-eslint-comments eslint-plugin-import eslint-plugin-jest eslint-plugin-jsx-a11y eslint-plugin-prettier eslint-plugin-react eslint-plugin-react-native @typescript-eslint/eslint-plugin @typescript-eslint/parser --dev
```

```json
// .eslintrc

{
"parser": "@typescript-eslint/parser",
"extends": [
"airbnb",
"plugin:@typescript-eslint/recommended",
"plugin:prettier/recommended",
"plugin:eslint-comments/recommended",
"plugin:import/errors",
"plugin:import/warnings",
"plugin:import/typescript",
"plugin:jest/recommended"
],
"env": {
"browser": true,
"jest": true,
"react-native/react-native": true
},
"plugins": [
"react",
"react-native",
"@typescript-eslint",
"jsx-a11y",
"import",
"prettier",
"jest",
"eslint-comments"
],
"rules": {
"@typescript-eslint/indent": "off",
"@typescript-eslint/explicit-function-return-type": "off",
"@typescript-eslint/no-use-before-define": "off",
"react/jsx-filename-extension": [1, { "extensions": [".js", ".jsx", ".ts", ".tsx"] }],
"react/prop-types": "off",
"react/button-has-type": "off",
"no-use-before-define": "off",
"import/no-extraneous-dependencies": [
"error",
{
"devDependencies": [
"storybook/**/*.{ts,tsx,js}",
"config-overrides.js",
"src/setupTests.ts",
"src/components/**/*.stories.tsx",
"src/styles/**/*.stories.tsx",
"src/**/*.test.{ts,tsx}"
]
}
],
"react-native/no-unused-styles": "error",
"react-native/no-inline-styles": "error",
"react-native/no-color-literals": "error",
"react/jsx-one-expression-per-line": "off",
"@typescript-eslint/explicit-member-accessibility": "off",
"prettier/prettier": ["error"]
},
"overrides": [
{
"files": ["*.style.ts"],
"rules": {
"@typescript-eslint/camelcase": "off"
}
},
{
"files": ["*.stories.tsx", "*.test.tsx"],
"rules": {
"@typescript-eslint/no-explicit-any": "off",
"react-native/no-color-literals": "off",
"react-native/no-inline-styles": "off"
}
}
]
}
```

also ignore some files/folders;

```text
# .eslintignore

ios
android
build
coverage

# Storybook
storybook/storyLoader.js
```

We need to update `package.json` for ESLint scripts.

```json
"lint:eslint": "eslint 'src/**/*.{ts,tsx}'",
"lint:ts": "tsc && yarn lint:eslint",
"lint": "yarn lint:ts",
"format": "prettier --write 'src/**/*.{ts,tsx}' && yarn lint:eslint --fix",
```

Finally, we need to enable prettier ESLint integration on VSCode.

```json
// .vscode/settings.json

{
// ... ,
"eslint.validate": [
"javascript",
"javascriptreact",
{ "language": "typescript", "autoFix": true },
{ "language": "typescriptreact", "autoFix": true }
]
}
```

## Step 6: Setting up our test environment

We'll use `jest` with `react-native-testing-library`.

```bash
yarn add react-native-testing-library --dev
```

Add the following script into `package.json`

```json
"test": "jest",
"test:watch": "yarn test --watch",
"coverage": "yarn run test --coverage"
```

and then update `jest.config.js` as follows to complete jest configuration.

```js
{
// ... ,
collectCoverageFrom: [
'src/**/*.{ts,tsx}',
'!src/index.tsx',
'!src/setupTests.ts',
'!src/components/**/index.{ts,tsx}',
'!src/**/*.stories.{ts,tsx}',
'!src/**/*.style.ts',
'!src/styles/**/*',
],
coverageThreshold: {
global: {
branches: 80,
functions: 80,
lines: 80,
statements: 80,
},
},
}
```

Let's add a simple test to verify our setup.

```tsx
// src/App.test.tsx

import 'react-native';
import React from 'react';
import { shallow } from 'react-native-testing-library';

import App from './App';

it('renders correctly', () => {
const comp = shallow();

expect(comp.output).toMatchSnapshot();
});
```

Also, verify coverage report with `yarn coverage`.

When you run `yarn coverage`, a folder named `coverage` will be created in the root directory. This folder is auto-generated file. We should add it to `.gitignore`

```text
# .gitignore

...
# Test Coverage
coverage
```

## Step 7: Setting up config variables

We use the [react-native-config](https://github.com/luggit/react-native-config) package to expose config variables to our javascript code in React Native.

Follow [these steps](https://github.com/luggit/react-native-config#setup) to install.

## Step 8: Organizing Folder Structure

Our folder structure should look like this;

```text
src/
├── App.test.tsx
├── App.tsx
├── __snapshots__
│ └── App.test.tsx.snap
├── components
│ └── Button
│ ├── Button.style.ts
│ ├── Button.stories.tsx
│ ├── Button.test.tsx
│ ├── Button.tsx
│ ├── __snapshots__
│ │ └── Button.test.tsx.snap
│ └── index.ts
├── containers
│ └── Like
│ ├── Like.tsx
│ └── index.ts
├── index.tsx
├── screens
│ ├── Feed
│ │ ├── Feed.style.ts
│ │ ├── Feed.test.tsx
│ │ ├── Feed.tsx
│ │ ├── index.ts
│ │ └── tabs
│ │ ├── Discover
│ │ │ ├── Discover.style.ts
│ │ │ ├── Discover.test.tsx
│ │ │ ├── Discover.tsx
│ │ │ └── index.ts
│ │ └── MostLiked
│ │ ├── MostLiked.test.tsx
│ │ ├── MostLiked.tsx
│ │ └── index.ts
│ ├── Home
│ │ ├── Home.style.ts
│ │ ├── Home.test.tsx
│ │ ├── Home.tsx
│ │ └── index.ts
│ └── index.ts
├── styles
│ ├── Colors.ts
│ ├── Spacing.ts
│ ├── Typography.ts
│ └── index.ts
└── utils
├── location.test.ts
└── location.ts
```

## Step 9: Adding Storybook

We need to initialize the Storybook on our project. We'll use automatic setup with a few edits:

```bash
npx -p @storybook/cli sb init --type react_native
```

> _Warning: Probably after you have run the command above, you'll be asked to select a version. Cancel it._

Storybook CLI automatically installs `v5.0.x`, however `v5.0.x` is an unpublished version for react-native, therefore problems arise during installation. In order to avoid this problem we're going to fix our storybook packages in our `package.json` file to latest stable version `4.1.x`. (Check [this issue](https://github.com/storybooks/storybook/issues/5893) for more information.)

```json
"@storybook/addon-actions": "^4.1.16",
"@storybook/addon-links": "^4.1.16",
"@storybook/addons": "^4.1.16",
"@storybook/react-native": "^4.1.16",
```

thereafter in order to activate the changes and update `yarn.lock` file we'll run code below;

```bash
yarn
```

After completing steps above you'll notice that storybook CLI have created `storybook` folder on your project's root folder. We'll customize this folder structure according to our use case.

Firstly change the name of `index.js` file in `storybook` folder to `storybook.ts`. Also change file extensions of other files from `js` to `ts`, except the `addons.js` file (https://github.com/storybooks/storybook/issues/3970).

After that, we create a new file named `index.ts` to expose StorybookUI in your app.

```js
// storybook/index.ts

import StorybookUI from './storybook';

export default StorybookUI;
```

We finished the storybook installation but we are not done yet;

The stories for our app will be inside the `src/components` directory with the `.stories.tsx` extension.The React Native packager resolves all the imports at build-time, so it's not possible to load modules dynamically. we need to use a third party loader [react-native-storybook-loader](https://github.com/elderfo/react-native-storybook-loader) to automatically generate the import statements for all stories.

```bash
yarn add react-native-storybook-loader --dev
```

You need to update `storybook.ts` as follows:

> _Note: Do not forget to replace `%APP_NAME%` with your app name_

```js
// storybook/storybook.ts

import { AppRegistry } from 'react-native';
import { getStorybookUI, configure } from '@storybook/react-native';
import { loadStories } from './storyLoader';

import './rn-addons';

// import stories
configure(() => {
loadStories();
}, module);

// Refer to https://github.com/storybooks/storybook/tree/master/app/react-native#start-command-parameters
// To find allowed options for getStorybookUI
const StorybookUIRoot = getStorybookUI({});

// If you are using React Native vanilla write your app name here.
// If you use Expo you can safely remove this line.
AppRegistry.registerComponent('%APP_NAME%', () => StorybookUIRoot);

export default StorybookUIRoot;
```

The file `storyLoader.js` that we imported above is an auto-generated file. We should add it to `.gitignore`.

```text
# .gitignore

...
# Storybook
storybook/storyLoader.js
```

After you install storybook loader, you should run the following command once to avoid typescript errors.

```bash
yarn rnstl
```

Update the storybook script into `package.json` as follows:

```json
"storybook": "watch rnstl ./src --wait=100 | storybook start | yarn start --projectRoot storybook --watchFolders $PWD"
```

Add the following config into `package.json`:

```json
// package.json
{
"config": {
"react-native-storybook-loader": {
"searchDir": ["./src"],
"pattern": "**/*.stories.tsx",
"outputFile": "./storybook/storyLoader.js"
}
}
}
```

> _Warning: If you get typescript errors related with the storybook, you should disable `isolatedModules` in `tsconfig.json`_

Lastly, because we use typescript in the project, we need to install the type definition for storybook.

```bash
yarn add @types/storybook__react-native --dev
```

Let's create an example story for our Button component.

```tsx
// src/components/Button/Button.stories.tsx

import React from 'react';
import { storiesOf } from '@storybook/react-native';

import Button from './Button';

storiesOf('Button', module)
.add('Primary', () => Primary Button)
.add('Secondary', () => Secondary Button);
```

## Step 10: Adding CircleCI config

We can create a CircleCI pipeline in order to CI / CD.

```yaml
# .circleci/config.yml

version: 2
jobs:
build_dependencies:
docker:
- image: circleci/node:10
working_directory: ~/repo
steps:
- checkout
- attach_workspace:
at: ~/repo
- restore_cache:
keys:
- dependencies-{{ checksum "package.json" }}
- dependencies-
- run:
name: Install
command: yarn install
- save_cache:
paths:
- ~/repo/node_modules
key: dependencies-{{ checksum "package.json" }}
- persist_to_workspace:
root: .
paths: node_modules

test_app:
docker:
- image: circleci/node:10
working_directory: ~/repo
steps:
- checkout
- attach_workspace:
at: ~/repo
- run:
name: Generate Storyloader
command: yarn rnstl
- run:
name: Lint
command: yarn lint
- run:
name: Format
command: yarn format:check
- run:
name: Coverage
command: yarn coverage

workflows:
version: 2
build_app:
jobs:
- build_dependencies
- test_app:
requires:
- build_dependencies
```

After that we need to enable CircleCI for our repository.

## Step 11: Github Settings

We want to protect our `develop` and `master` branches. Also, we want to make sure our test passes and at lest one person reviewed the PR. In order to do that, we need to update branch protection rules like this in GitHub;

![github-branch-settings](https://user-images.githubusercontent.com/1801024/55028896-df2a3300-5019-11e9-9827-a984fcfa29af.png)

## Step 12 Final Touches

We are ready to develop our application. Just a final step, we need to update our `README.md` to explain what we add a script so far.

[EXAMPLE_README](EXAMPLE_README.md)

## Step 13: Starting to Development 🎉

Everything is done! You can start to develop your next awesome React Native application now on 🚀

## Bonus: Npm Script Aliases

### React-Native Alias

```bash
yarn rn
```

`rn` alias for `react-native` allows to run react-native CLI command via locally installed react-native.

```json
// package.json

"rn": "react-native",
```
**NOTE:** Only works with yarn.

### Run On Aliases

```bash
yarn ios
yarn run ios

yarn android
yarn run android

```

`ios` and `android` aliases are helpful when we need to pass different parameter for our project and provides single point entry.

```json
// package.json

"ios": "yarn rn run-ios",
"android": "yarn rn run-android",
```

#### Example

If we want to run our app on iPhone X as default and with scheme just specify that in the alias.

```json
// package.json

"ios": "yarn rn run-ios --simulator 'iPhone X' --scheme 'Production'",
```

### Clear React Native Cache Alias

```bash
yarn clear-rn-cache
```

```json
// package.json

"clear-rn-cache": "watchman watch-del-all && rm -rf $TMPDIR/react-* && rm -rf $TMPDIR/metro* && rm -rf $TMPDIR/haste-*"
```

## Related

* [cra-recipe](https://github.com/atolye15/cra-recipe/) - CRA Recipe