npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

react-router-outlet

v0.2.4

Published

React Router Outlet - Angular-inspired router outlet for React Router

Downloads

17

Readme

React Router Outlet

React Router Outlet provides a very simple, declarative way to define your React Router routes, inspired by Angular's approach.

Prerequisite

This library requires React Router v5 as peer dependency.

To enable routing, the app would have a <Router> instance at a top-level component, ideally the <App>. You will only need to have one such instance for the entire app. This is provided by the standard React Router, not this library. You can check the React Router official documentation for details.

The Router Outlet

This library provides a router outlet component that dynamically changes content depending on the current route. This is a convenient drop-in replacement for React Router's combination of <Switch>, <Route> and <Redirect> components, and then adds a couple of neat features as well.

<RouterOutlet>

Place this component wherever you need routed views.

<RouterOutlet routes={routes} />

where routes defines the routes declaratively (see below).

Routing can apply to any level component, so you can place a <RouterOutlet> virtually anywhere in the component tree. Similarly, you can have as many nested router outlets as required in the application, particularly when your route paths go from wildcard to specific.

IMPORTANT: If your <RouterOutlet> and the <Router> are not in the same parent component, you would normally need to apply React Router's withRouter HOC to the component that renders the <RouterOutlet>. This is a requirement of React Router, not of Outlet, so please check their documentation for details.

Defining Routes Declaratively

We use this format when declaring routes:

const routes = [
  {
    path: '/login',
    component: LoginPage
  },
  {
    path: '/default',
    component: DefaultPage
  },
  {
    path: '/',
    exact: true,
    redirectTo: '/default'
  },
  {
    component: NotFound
  }
];

These are the common cases for routes:

  • Both path and component are defined in the route spec. This is straightforward routing.
  • No path is specified, which means this is the default route if none of the preceding routes matched.
  • If exact is set to true, the path has to match exactly.
  • If redirectTo path is specified instead of component, this is a straightforward redirection.

Protected Routes

Sometimes you need to restrict access to certain routes depending on some conditions. In such cases, you can do something like this:

  {
    path: '/default',
    component: DefaultPage,
    canEnter: isAuthenticated,
    fallback: '/login',
  },
const isAuthenticated = ({ auth }) => auth !== null;

If a canEnter guard function is defined, the router will only activate the matched route if this function evaluates to true. Otherwise, it will route to the fallback path. This is useful for apps that require user login to access specific features. Just make sure to define fallback whenever you have a canEnter.

The signature of such guard function is:

(outletProps: Object, route: Object) => condition: boolean

where outletProps contains all props that were passed to <RouterOutlet>, while route is a reference to the matched route.

In the example above, the outlet would look like this:

<RouterOutlet routes={routes} auth={token} />

so that the auth prop is passed to the isAuthenticated guard.

If you pass some state value (or an injected state prop, either internal or from store), the <RouterOutlet> will re-render, hence routing will happen, whenever this state value changes.

Lazy Evaluated Redirection/Fallback

Both redirectTo and fallback in your routes can alternatively be expressed as functions with the following signature:

(outletProps: Object, route: Object) => path: string

In this case, these path values are only evaluated when the <RouterOutlet> renders, and therefore you can dynamically determine the target paths based on props that were passed to the outlet, similar to how canEnter works with such props.

For example:

{
  path: '/login',
  redirectTo: ({ referrer }) => referrer
}

where referrer is a prop of the outlet:

  <RouterOutlet routes={routes} referrer={currentPath} />

Routing to Lazy Loaded Components

For optimization, feature components can be lazy loaded, i.e. they are only fetched by the browser if and when they are required by the app. This technique is also called code splitting.

Instead of the normal import, use React Lazy to import lazy-loaded components.

import React, { lazy } from 'react';

const WelcomePage = lazy(() => import('../welcome/WelcomePage'));

Then you can optionally tell <RouterOutlet> to display a placeholder (e.g. a spinner component) while the component is being loaded. For example:

<RouterOutlet routes={routes} placeholder={<div>...</div>} />

where placeholder can be any component.

The <RouterOutlet> supports routing to such lazy-loaded components by using React Suspense internally.

For more information on React Lazy and Suspense, check out the official documentation.