Skip to content

Instantly share code, notes, and snippets.

@faceyspacey
Last active April 25, 2019 13:03
Show Gist options
  • Save faceyspacey/7356dee86c74d30fa21964a35e80de2b to your computer and use it in GitHub Desktop.
Save faceyspacey/7356dee86c74d30fa21964a35e80de2b to your computer and use it in GitHub Desktop.
Respond Framework Walkthrough

Respond Framework Walkthrough

Respond Framework is what happens if you build Redux & first-class concerns for routing into React, plus take a page from the traditional server-side MVC playbook when it comes to side-effects.

Here's a quick overview of the features and usage in Respond Framework.

Installation

yarn add 'respond-framework'

Modular (did I hear you say "Redux Modules"??)

Respond Framework is modular and encapsulated like React components, but at a higher level. It gives you a birds-eye perspective of important portions of your app & enables separate developer teams to go off and build with confidence that what they bring back will plugin nicely.

Your app is composed of Respond Modules, which encapsulate routes, components, and reducers. moduleProps allow you to share state, actions, and types between parent modules and their children.

src/modules/app/index.js

import { createApp } from 'respond-framework'
import ReactDOM from 'react-dom'
import enhancer from './enhancer'
import reducer from './reducer'
import components from './components'

const { firstRoute, store } = createApp({
  enhancer,
  reducer,
  initialState: window.RESPOND_STATE,
  components,
  routes: {
    home: '/',
    login: '/login',
    dashboard: {
      path: '/dashboard',
      load: () => import('../modules/dashboard'),
      moduleProps: {
        state: {
          user: 'session'
        },
        actions: {
          login: 'login'
        }
      }
    }
  }
})

(async function() {
  await store.dispatch(firstRoute())

  ReactDOM.hydrate(
    <Provider store={store}>
      <App />
    </Provider>,
    document.getElementById('root')
  )
})()

src/modules/dashboard/index.js:

import { createModule } from 'respond-framework'
import { foo, bar } from './reducer'
import components from './components'

export default createModule({
  reducers: { foo, bar },
  components,
  routes: {
    main: '/',
    metrics: '/metrics',
    stats: '/stats'
  }
})

the configuration passed to createApp is also a module. Respond is "modules" all the way down.

Predictable Linear Effects

No longer leave side-effects up to random discovery in your component tree! "No surprises == better sleep". Instead orchestrate them linearly once per route.

Trust us, the React team has you brainwashed--side-effects don't belong in your components, even with hooks :)

export default createModule({
  reducers: { foo, bar },
  components,
  routes: {
    main: {
+      path: '/',
+      thunk: ({ api }) => api.fetch('items')
    },
    metrics: '/metrics',
    stats: '/stats'
  }
})

Or if you're old school:

export default createModule({
  reducers: { foo, bar },
  components,
  routes: {
    main: {
      path: '/',
+      thunk: async ({ api, dispatch, types }) => {
+        const payload = await api.fetch('items')
+        dispatch({ type: types.main.COMPLETE, payload })
      }
    },
    metrics: '/metrics',
    stats: '/stats'
  }
})

types, actions & state are supplied via dependency injection to avoid conflicts; that also reduces # of imports

Add callbacks that fire for all routes:

import mixpanel from 'mixpanel'

export default createModule({
  reducers: { foo, bar },
  components,
  routes: {
    main: {
      path: '/',
      thunk: thunk: ({ api }) => api.fetch('items')
    },
    metrics: '/metrics',
    stats: '/stats'
  }
}, {
+  onEnter: ({ location }) => mixpanel.track('dash', location)
})

they are run in parallel to route-specific ones

Bail out using redirects before route changes:

export default createModule({
  reducers: { foo, bar },
  components,
  routes: {
    main: {
      path: '/',
      thunk: thunk: ({ api }) => api.fetch('items')
    },
    metrics: '/metrics',
    stats: '/stats'
  }
}, {
+  beforeEnter: async ({ getState, actions }) => {
+    if (!getState().session) {
+      return actions.login() // remember, the login action was passed as a `moduleProp`
+    }
+  },
  onEnter: ({ location }) => mixpanel.track('dash', location)
})

beforeEnter could just as easily been added to individual routes

Bail out on leave:

export default createModule({
+  reducers: { foo, bar, acceptedCookies },
  components,
  routes: {
    main: {
      path: '/',
      thunk: ({ api }) => api.fetch('items'),
+      onLeave: ({ getState }) => !getState().acceptedCookies // return false to block route change
    },
    metrics: '/metrics',
    stats: '/stats'
  }
}, {
  beforeEnter: async ({ getState, actions }) => {
    if (!getState().session) {
      return actions.login()
    }
  },
  onEnter: ({ location }) => mixpanel.track('dash', location)
})

100% Customizable Middleware

The backbone of Respond is our routing slash side-effects library, Rudy. Rudy offers an async middleware API similar to koa-compose with "rewind."

What this means is that each middleware will pause execution of the route change and asynchronously complete before passing the request to the next middleware in the chain. While classic Redux offers a synchronous middleware API, ours is async to meet the demands of today.

The above routes could be minimally served with this middleware pipeline:

export default createModule(config, options, [
+  transformAction, 
+  call('beforeEnter'),
+  enter,
+  call('onLeave', { prev: true }),
+  call('onEnter'),
+  call('thunk', { cache: true }),
])

All the callback names passed to call are available as keys on your routes and executed at the appropriate time during route transitions. The call middleware has many other super powers like automatically dispatching returns as actions.

Each middleware also has a 2nd chance to peform work as the chain "rewinds."

This gives us great control over route transitions. For example, we can bail out at any time (before or after enter). We have even figured out how to do this in bail-outs triggered by browser back/next buttons, thanks to our custom History package within core. By the way, our History is truly one of a kind--first to keep track of browser history entries, more on that to come...

If you're wondering, yes, you can still use the traditional Redux enhancer/middleware APIs like the Devtools and Sagas

Generated Action Creators & Types

Your routes object generates all the action types your application needs. For example in the current app, so for we have:

  • actions.home()
  • actions.login()
  • actions.dashboard()
  • actions.dashboard.metrics()
  • actions.dashboard.stats()

As well as the following for each route:

  • actions.home.complete()
  • actions.home.error()
  • actions.dashboard.metrics.complete()
  • etc

The types generated are just the capitalized snake_cased versions of these prefixed by their namespaces:

  • HOME
  • LOGIN
  • DASHBOARD
  • DASHBOARD/METRICS
  • DASHBOARD/STATS

And:

  • HOME_COMPLETE
  • HOME_ERROR
  • DASHBOARD/METRICS_COMPLETE
  • etc

They are accessible in their lowercased form at, for example: types.home or actions.home.type.

They are injected into callbacks:

thunk: ({ types, actions }) => 

Into reducers:

reducers: {
  foo: (state, action, types) => types.home ? true : state
}

And components:

const RespondComponent = (props, state, actions) => <Button onClick={actions.login} />

3 argument components will be covered shortly

Because Respond Modules are guaranteed to be unaware of the outside world (even though they're conveniently using the same store), actions, types and state must be injected by the framework. This allows Respond to transparently normalize namespace access under the hood via proxies, so you only have to use namespaces where you absolutely must, which brings up an important point:

Actions, state and types from child modules is available in parent components by their namespace. Whereas child modules must use moduleProps to access the same from the parent. In other words, parents get to know whats up with their children, but not the other way around (kind of like in real life :)

Automatic Code Splitting

Respond automatically code splits your app. There's nothing you have to do about it, it just happens through normal use of modules.

It's provided through one of our middleware. If you don't provide a middleware array, here's the default one:

export default createModule(config, options, [
  serverRedirect,           // short-circuiting middleware       
  anonymousThunk,
  pathlessRoute('thunk')   
  transformAction,          // pipeline starts here
  codeSplitModule('load'),  
  call('beforeLeave', { prev: true }),
  call('beforeEnter'),
  enter,
  changePageTitle,
  call('onLeave', { prev: true }),
  call('onEnter'),
  call('thunk', { cache: true }),
  call('onComplete')
])

The codeSplitModule middleware is responsible for insuring modules (including their components, reducers and route side-effects) are loaded before the route executes. In other words, the plane is built while flying.

Routes can also be prefetched, including both their Webpack chunks and data dependencies which get cached.

To prefetch potential subsequent routes, return a list of them from the current route using the prefetch handler:

routes: {
  login: '/login',
  home: {
    path: '/',
    prefetch: ({ actions, types, getState }) => [
      types.login,
      actions.profile({ params: { id: getState().user.slug }})
    ]
  },
  profile: {
    path: '/profile/:slug',
    thunk: ({ api, params }) => api.fetch(`users/${params.slug}`)
  }
}

If you provide a precise action, both the chunk and callbacks such as thunks will be called (with their results cached, aka stored in Redux); if you, supply just a string as in types.login, only the chunk for the matching route will be called (since thunks wouldn't know what to fetch without precise params/etc).

Serve Split Chunks w/ SSR

SSR is challenging. Code Splitting is challenging.

SSR + Splitting unfortunately is greater than the sum of its parts, which is to say combination SSR + Splitting is many times more challenging.

With Respond it's just a matter of passing the request url and awaiting your firstRoute().

server/configureStore.js:

import { createApp } from 'respond-framework'

export default async function configureStore(request) {
  const options = {
    initialEntries: [request.url]
  }

  const { firstRoute, store } = createApp(config, options) // default middleware used

  await store.dispatch(firstRoute())

  return store
}

And then extracting used chunks from state:

server/serverRender.js:

import ReactDOM from 'react-dom/server'
import { Provider } from 'respond-framework'
import configureStore from './configureStore'
import App from '../src/components/App'

export default async function serverRender(req, res) {
  const store = await configureStore(req)

  const appString = ReactDOM.renderToString(<Provider store={store}><App /></Provider>)
  const stateJson = JSON.stringify(store.getState())

  // like this:

  const scripts = store.getState().location.chunks.map(chunk => {
    return `<script src="/static/${chunk}.js" />`
  }).join(' ')

  return res.send(
    `<!doctype html>
      <html>
        <body>
          <div id="root">${appString}</div>
          <script>window.RESPOND_STATE = ${stateJson}</script>
          <script src="/static/bootstrap.js" />
          <script src="/static/vendors.js" />
          ${scripts}
        </body>
      </html>`
  )
}

server/index.js:

import express from 'express'
import serverRender from './serverRender'

const app = express()
app.get('*', serverRender)
http.createServer(app).listen(3000)

Yes, we wrote the book when it comes to routing, splitting and SSR in a Redux world. Respond Framework is the direct heir to:

Baked-in Redux (check out our sweet Components!)

Internally, our state management library is called Remixx, but it's ok if you continue to call it "Redux" :)

"Redux Modules"--you know the ones the community never figured out how to make--was always about components. The idea is that you can provide a pairing of Redux assets (reducers, actions, types) and components in a format that you can share with 3rd parties, such as on NPM, without naming conflicts. In other words: mini apps. It would have been nice, but when was the last time you saw Redux-based components on NPM?? Never until now.

While making this possible, we took the liberty to build in exactly what you might expect into React. Here's what Respond components look like:

const LoginButon = (props, state, actions) => !state.session && <Button onClick={actions.login} />

state and actions are passed as arguments in addition to props!

Actions are automatically bound to dispatch and there's never any need for mapStateToProps. Under the hood proxies are used to track your actual usage of state so re-renderings only occur if the precise nested piece of state you accessed has changed. This applies both to reducers and selectors. More on selectors below.

Yes, Respond is built for the era where you assume your users' browsers support proxies.

The transformation of your components to support this interface is done via Babel, therefore if you don't happen to use state or actions, your components will be left untouched. You're free to use hooks, side effects, you name it (though we recommend keeping your side-effects in Respond routes).

Automatic Namespacing

If you saw:

const MetricsButton = (props, state, actions) => !state.foo && <Button onClick={actions.metrics} />

export default createModule({
  reducers: { foo, bar },
  components: {
    Dash: (props) => {
      return (
        <div>
          <MetricsButton />
        </div>
      )
    }
  },
  routes: {
    main: '/',
    metrics: '/metrics',
    stats: '/stats'
  }
})

and were wondering how actions.login and state.session was guaranteed to be unique if this component was part of a module on NPM, you'd be a keen observer.

Under the hood (within the state and actions proxies) here's what's actually being called:

const LoginButon = (props, state, actions) => {
  return !state.dashboard.foo && <Button onClick={actions.dashboard.metrics} />
}

That's because the component knows what module it's part of. It doesn't need to provide its own namespace. It wouldn't even work if it tried. Respond components have no awareness of the outside world unless it's told about it.

Like import aliasing in ES6 modules, the namespace is assigned in the parent module. Remember this:

src/modules/app:

dashboad: {
  path: '/dashboard',
  load: () => import('../modules/dashboard'),
}

The parent route type, DASHBOARD doubles as the module's namespace for nested routes

Here's how you tell the dashboard module about pre-existing state and actions in the parent:

dashboad: {
  path: '/dashboard',
  load: () => import('../modules/dashboard'),
+  moduleProps: {
+    state: {
+      user: 'session'
+    },
+    actions: {
+      login: 'login'
+    }
+  }
}

So getState().user will be made available through the state proxy at state.session and the login action will simply be passed down by the same name (since that's what the child module's documentation also said was the name).

Modules, Nesting, Splitting, Oh My!

The cornerstone of Respond's flawless interface is one thing: the collapsing of many capabilities into our core "modules" feature.

Let's take a look at how dynamically imported routes look like after being merged:

before:

// parent module:
routes: {
  dashboard: {
    path: '/dashboard',
    load: () => import('../modules/dashboard'),
  }
}

// child module:
routes: {
  metrics: '/metrics',
  stats: '/stats'
}

after:

routes: {
  dashboard: {
    path: '/dashboard',
    thunk: ({ api }) => api.fetch('user'),
    routes: {
      metrics: '/metrics',  // reified path /dashboard/metrics
      stats: '/stats'       // reified path /dashboard/stats
    }
  }
}

As you can see, modules, path nesting, and code splitting are all collaped into a single unified interface.

But that's not all, the thunk above has a special characterstic we call "callback nesting":

  • it's called even if you visit /dashboard/metrics directly
  • it's not called if you navigate from /dashboard/metrics to /dashboard/stats though

It's only called on first entrance of the given group of nested routes/module. The common use case for this is to insure that the user or session object exists for all dashboard routes, without having to code the call for every route.

Let's check out a few more scenarios:

Use a different path prefix than the parent:

routes: {
  dashboard: {
    path: '/dashboard',
    pathPrefix: '/something-else',
    thunk: ({ api }) => api.fetch('user'),
    routes: {
      metrics: '/metrics',  // reified path /something-else/metrics
      stats: '/stats'       // reified path /something-else/stats
    }
  }
}

Leaving out the parent module path:

routes: {
  dashboard: {
    thunk: ({ api }) => api.fetch('user'),
    routes: {
      metrics: '/metrics',  // reified path /metrics
      stats: '/stats'       // reified path /stats
    }
  }
}

In essence, the route nesting is being used just for callback nesting + module namespacing; actions.dashboard() doesn't exist!

Leaving out the parent module path:

routes: {
  dashboard: {
    path: '/dashboard',
    pathPrefix: false,
    thunk: ({ api }) => api.fetch('user'),
    routes: {
      metrics: '/metrics',  // reified path /metrics
      stats: '/stats'       // reified path /stats
    }
  }
}

This allows you to dispatch actions.dashboard() and reach an independent route as you originally could, while allowing for child routes to be unprefixed. It's a hybrid of the previous ones.

Merge child route into parent:

// pre-import:
dashboard: {
  path: '/dashboard',
  load: () => import('../modules/dashboard'),
}

// dashboard module:
routes: {
  main: {
    path: '/',
    thunk: ({ api }) => api.fetch('user'),
  }
}

//post-import:
routes: {
  dashboard: {
    path: '/dashboard',
    thunk: ({ api }) => api.fetch('user'),
    alias: 'main',
  }
}

If a child route has a path of '/' it's assumed that the goal of code-splitting in this case is to merge the route capabilities into the parent. The child will have refered to this action by, in this case, main, so it's specified as an alias, resulting in actions.dashboard() being the action creator available in parent routes and actions.main() within the child module itself.

This pattern can be very useful for splitting single routes as above, but could have also brought along additional nested routes as before.

selectors

  • selectors

Routing Components

Location Location Location

  • transformation
  • tons of info in state
  • automatic basename handling

Misc Features

  • History Entries Sync (world first!)
  • Caching
  • Prefetching
  • pathless routes
  • anonymous thunks

Big Picture

Modularity + Linear Side Effects

Conclusion

So there you go--long standing ecosystem problems finally solved and just about every effective React concept consolidated/collapsed under one beautiful interface. GraphQL/Apollo middleware coming soon. Welcome to the Rails of React.

Ambiguous Paths and Ranking

Even though two paths might be ambiguous--like "/:invoiceId" and "/invoices"--Router ranks the paths and renders the one that makes the most sense.

render(
  <Router>
    <Home path="/" />
    <Invoice path=":invoiceId" />
    <InvoiceList path="invoices" />
  </Router>
)

The URL "/invoices" will render <InvoiceList/> and "/123" will render <Invoice invoiceId={123}/>. Same thing with the Home component. Even though it’s defined first, and every path will match "/", Home won't render unless the path is exactly "/". So don't worry about the order of your paths.

Nested Component Paths

You can nest components inside of a Router, and the paths will nest too. The matched child component will come in as the children prop, the same as if you'd rendered it directly. (Internally Router just renders another Router, but I digress...)

const Dash = ({ children }) => (
  <div>
    <h1>Dashboard</h1>
    <hr />
    {children}
  </div>
)

render(
  <Router>
    <Home path="/" />
    <Dash path="dashboard">
      <Invoices path="invoices" />
      <Team path="team" />
    </Dash>
  </Router>
)

If the URL is "/dashboard/invoices" then the Router will render <Dash><Invoices/></Dash>. If it's just "/dashboard", children will be null and we’ll only see <Dash/>.

Most apps probably have some sort of global chrome/navigation, that works out just fine:

const Main = ({ children }) => (
  <div>
    <h1>Welcome to the App!</h1>
    <ul>
      <li>
        <Link to="dashboard">Dashboard</Link>
      </li>
      <li>
        <Link to="invoices">Invoices</Link>
      </li>
    </ul>
    <hr />
    {children}
  </div>
)

render(
  <Router>
    <Main path="/">
      <Invoices path="invoices" />
      <Dash path="dashboard" />
    </Main>
  </Router>
)

Relative Links

You can link to relative paths. The relativity comes from the path of the component that rendered the Link. These two links will link to "/dashboard/invoices" and "/dashboard/team" because they're rendered inside of <Dash/>. This is really nice when you change a parent's URL, or move the components around.

render(
  <Router>
    <Home path="/" />
    <Dash path="dashboard">
      <Invoices path="invoices" />
      <Team path="team" />
    </Dash>
  </Router>
)

const Dash = ({ children }) => (
  <div>
    <h1>Dashboard</h1>
    <nav>
      <Link to="invoices">Invoices</Link>{" "}
      <Link to="team">Team</Link>
    </nav>
    <hr />
    {children}
  </div>
)

This also makes it trivial to render any section of your app as its own application with its own router. If all your links are relative, it can be embedded inside any other router and just work.

"Index" Paths

Nested components can use the path / to signify they should render at the path of the parent component, like an index.html file inside a folder on a static server. If this app was at "/dashboard" we'd see this component tree: <Dash><DashboardGraphs/></Dash>

render(
  <Router>
    <Home path="/" />
    <Dash path="dashboard">
      <DashboardGraphs path="/" />
      <InvoiceList path="invoices" />
    </Dash>
  </Router>
)

Not Found "Default" Components

Put a default prop on a component and Router will render it when nothing else matches.

const NotFound = () => (
  <div>Sorry, nothing here.</div>
)

render(
  <Router>
    <Home path="/" />
    <Dash path="dashboard">
      <DashboardGraphs path="/" />
      <InvoiceList path="invoices" />
    </Dash>
    <NotFound default />
  </Router>
)

Multiple Routers

If you want to match the same path in two places in your app, just render two Routers. Again, a Router picks a single child to render based on the URL, and then ignores the rest.

Just makes sure to mark the non-primary router(s) as primary={false} so that it doesn't manage the focus on those components.

render(
  <div>
    <Sidebar>
      <Router primary={false}>
        <HomeNav path="/" />
        <DashboardNav path="dashboard" />
      </Router>
    </Sidebar>

    <MainScreen>
      <Router>
        <Home path="/">
          <About path="about" />
          <Support path="support" />
        </Home>
        <Dash path="dashboard">
          <Invoices path="invoices" />
          <Team path="team" />
        </Dash>
      </Router>
    </MainScreen>
  </div>
)

Embedded Routers

You can render a router anywhere you want in your app, even deep inside another Router, just makes sure to use a splat (*) on the parent component so nested paths match it.

render(
  <Router>
    <Home path="/" />
    <Dash path="dashboard/*" />
  </Router>
)

const Dash = () => (
  <div>
    <p>A nested router</p>
    <Router>
      <DashboardGraphs path="/" />
      <InvoiceList path="invoices" />
    </Router>
  </div>
)

This allows you to have all of your routes configured at the top of the app, or to configure only where you need them, which is really helpful for code-splitting and very large apps. You can even render Dash as an independent application.

Navigating Programmatically

If you need to navigate programmatically (like after a form submits), import navigate.

import { navigate } from "@reach/router"

const Invoices = () => (
  <div>
    <NewInvoiceForm
      onSubmit={async event => {
        const newInvoice = await createInvoice(
          event.target
        )
        navigate(`/invoices/${newInvoice.id}`)
      }}
    />
  </div>
)

Or better, yet, use props.navigate passed to your route components and then you can navigate to relative paths:

const Invoices = ({ navigate }) => (
  <div>
    <NewInvoiceForm
      onSubmit={async event => {
        const newInvoice = await createInvoice(
          event.target
        )
        // can navigate to relative paths
        navigate(newInvoice.id)
      }}
    />
  </div>
)
;<Router>
  <Invoices path="invoices" />
  <Invoice path="invoices/:id" />
</Router>

Navigate returns a promise so you can await it. It resolves after React is completely finished rendering the next screen, even with React Suspense.

class Invoices extends React.Component {
  state = {
    creatingNewInvoice: false
  }

  render() {
    return (
      <div>
        <LoadingBar
          animate={this.state.creatingNewInvoice}
        />
        <NewInvoiceForm
          onSubmit={async event => {
            this.setState({
              creatingNewInvoice: true
            })
            const newInvoice = await createInvoice(
              event.target
            )
            await navigate(
              `/invoice/${newInvoice.id}`
            )
            this.setState({
              creatingNewInvoice: false
            })
          }}
        />
        <InvoiceList />
      </div>
    )
  }
}

That's it!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment