Skip to content

Latest commit

 

History

History
139 lines (115 loc) · 4.1 KB

how-to-implement-routing.md

File metadata and controls

139 lines (115 loc) · 4.1 KB

How to Implement Routing and Navigation img

Step 1: Basic Routing

In its simplest form the routing looks like a collection or URLs where each URL is mapped to a React component:

// app.js
import React from 'react';
import Layout from './components/Layout';
import HomePage from './components/HomePage';
import AboutPage from './components/AboutPage';
import NotFoundPage from './components/NotFoundPage';
import ErrorPage from './components/ErrorPage';

const routes = {
  '/':      <Layout><HomePage /></Layout>,
  '/about': <Layout><AboutPage /></Layout>
};

const container = document.getElementById('app');

function render() {
  try {
    const path = window.location.path.substr(1) || '/';
    const component = routes[path] || <NotFoundPage />;
    React.render(component, container);
  } catch (err) {
    React.render(<ErrorPage {...err} />, container);
  }
}

window.addEventListener('hashchange', () => render());
render();

Step 2: Asynchronous Routes

Just wrap React components inside your routes into asynchronous functions:

import React from 'react';
import http from './core/HttpClient';
import Layout from './components/Layout';
import HomePage from './components/HomePage';
import AboutPage from './components/AboutPage';
import NotFoundPage from './components/NotFoundPage';
import ErrorPage from './components/ErrorPage';

const routes = {
  '/': async () => {
    const data = await http.get('/api/data/home');
    return <Layout><HomePage {...data} /></Layout>
  },
  '/about': async () => {
    const data = await http.get('/api/data/about');
    return <Layout><AboutPage {...data} /></Layout>;
  }
};

const container = document.getElementById('app');

async function render() {
  try {
    const path = window.location.hash.substr(1) || '/';
    const route = routes[path];
    const component = route ? await route() : <NotFoundPage />;
    React.render(component, container);
  } catch (err) {
    React.render(<ErrorPage {...err} />, container);
  }
}

window.addEventListener('hashchange', () => render());
render();

Step 3: Parametrized Routes

(1) Convert the list of routes from hash table to an array, this way the order of routes will be preserved. (2) Wrap this collection into a Router class, where you can put .match(url) async method. (3) Use path-to-regexp to convert Express-like path strings into regular expressions which are used for matching URL paths to React components.

import React from 'react';
import Router from './core/Router';
import http from './core/HttpClient';
import Layout from './components/Layout';
import ProductListing from './components/ProductListing';
import ProductInfo from './components/ProductInfo';
import NotFoundPage from './components/NotFoundPage';
import ErrorPage from './components/ErrorPage';

const router = new Router([{
  path: '/products',
  action: async () => {
    const data = await http.get('/api/products');
    return <Layout><ProductListing {...data} /></Layout>
  }
}, {
  path: '/products/:id',
  action: async (id) => {
    const data = await http.get(`/api/products/${id}`);
    return <Layout><ProductInfo {...data} /></Layout>;
  }
}]);

const container = document.getElementById('app');

async function render() {
  try {
    const path = window.location.hash.substr(1) || '/';
    const component = await router.match(path) || <NotFoundPage />;
    React.render(component, container);
  } catch (err) {
    React.render(<ErrorPage {...err} />, container);
  }
}

window.addEventListener('hashchange', () => render());
render();

Step 4. Handling Redirects

Coming soon. Stay tuned!