Use HashRouter, BrowserRouter, Route, Link, Redirect, and Switch to react-router-DOM. But how does it work internally?

1, an overview of the

The HashRouter houses location and history, Route, Link, Redirect, and Switch implementations all rely on it. The HashRouter is their root component and holds the context’s data

  • context
import React from 'react';
const context = React.createContext();
export default context;
Copy the code

2. Implementation of HashRouter

Put the history and location objects in the context that the child components can call. The child component calls the push method to change location.hash and listens for the Hashchange event to render accordingly.


import React, {Component} from 'react';
import Context from './context';
export default class HashRouter extends Component {
  // Define an initialized state
 state = {  location: {pathname:window.location.hash.slice(1) | |'/'},  state:null  }  componentDidMount(){  // The core is to listen for hashChange events  window.addEventListener('hashchange', () = > { this.setState({  location: {. this.state.location, pathname:window.location.hash.slice(1), // #/a ---> /a  state:this.locationState  }  })  })  }  locationState = null;  render(){  let that = this;  let value = {  location:that.state.location,  history: { push(to){// Define a history object with a push method to jump to the path  if(typeof to === 'object') { let {pathname,state} = to;  that.locationState = state;  window.location.hash = pathname;  }else{  that.locationState = null;  window.location.hash = to;  }  }  }  }  return (  // The value attribute is specifically provided to the context to store shared data  <Context.Provider value={value}>  {this.props.children}  </Context.Provider>  )  } }  Copy the code

3. Implementation of Route


import React, {Component} from 'react';
import Context from './context';
import reg from 'path-to-regexp';
export default class Route extends Component{
 static contextType = Context;  render(){  let {pathname} = this.context.location;  let {path='/'.component:Component,exact=false} = this.props;  let paramNames = [];  // Match the route with the re  let regexp = reg(path,paramNames,{end:exact});  let result = pathname.match(regexp);   let props = {  location:this.context.location,  history:this.context.history,  }  if(result) {  paramNames = paramNames.map(item= >item.name);  let [url,...values] = result;  let params = {};  for(let i=0; i<paramNames.length; i++){ params[paramNames[i]] = values[i];  }  props.match = {  path,  url,  isExact: url===pathname,  params  }  return (<Component {. props} / >);  }  return null;  } } Copy the code

4. Implementation of Switch

And the switch here… The idea behind case is that it matches the child components and returns the matched component as soon as a match is found

import React, {Component} from 'react';
import Context from './context';
import PathToRegexp from 'path-to-regexp';
export default class Switch extends Component{
  static contextType = Context;
  render(){  let {pathname} =this.context.location;   let children = Array.isArray(this.props.children)?this.props.children:[this.props.children];   for(let i=0; i<children.length; i++){ let child = children[i];  let {path='/',exact} = child.props;  let paramNames = [];   let regexp = PathToRegexp(path,paramNames,{end:exact});  let result = pathname.match(regexp);   if(result){  return child;  }  }   return null  } } Copy the code

Redirect implementation

Its implementation is very simple, it is used in Switch, equivalent to Switch.. Default in case syntax. The idea isto change the location by calling the push method of history in the HashRouter.

import React, {Component} from 'react';
import Context from './context';

export default class Redirect extends Component{
  static contextType = Context
 componentDidMount(){  this.context.history.push(this.props.to)  }  render(){  return null;  } } -------------------------------------------- Use:<Redirect to="/home" from='/' /> Copy the code

6, Link implementation

Link in React replaces the A tag to Redirect. It works in a similar way to Redirect, in that it invokes the push method of history in HashRouter.

import React, {Component} from 'react';
import Context from './context';
export default class Route extends Component{
  static contextType = Context
  
 render(){   return (  // This can also be implemented  // <a to={`#{this.props.to}`}>{this.props.children}</a>  <a {. this.props} onClick={()= >{this.context.history.push(this.props.to)}}>{this.props.children}</a>  )  } } Copy the code

7, menuLink

import React, {Component} from 'react';
import Route from "./Route";
import Link from './Link'

export default ({to, children}) => {
 // If a match is found, the current component is given an active className  return <Route path={to} children={props= > (  <li className={props.match ? "active" :""} >  <Link to={to}>{children}</Link>  </li>  ) } / >}  Copy the code

7, withRouter

A normal component would also want to have properties and methods on Route, using withRouter, which is essentially a higher-order function.

import React, {Component} from 'react';
import Route from './Route';
export default function(WrappedComponent){
  return (a)= ><Route component={WrappedComponent}/>
} Copy the code

use

import React from 'react';
import {withRouter} from '.. /react-router-dom'
 class NavHeader extends React.Component{
  render(){
    return (
 <div className="navbar-heading"> // There is no history method for components not wrapped by Route <div onClick={()= >This. Props. History. Push ('/')} > XX science and technology</div>  </div>  )  } } export default withRouter(NavHeader) Copy the code

8, BrowserRouter implementation method

It is implemented by listening for popState and pushState.

import React, {Component} from 'react';
import Context from './context';
// let pushstate = window.history.pushState;
export default class BrowserRouter extends Component {
  state = {
 location: {pathname:window.location.pathname||'/'},  state:null  }   pushstate = window.history.pushState;   componentDidMount(){  // Since there is no onpushstate on the native method, we need to override pushstate  window.history.pushState = (state,title,url) = > {  // Call the native method first  this.pushstate.call(window.history,state,title,url)   window.onpushstate.call(this,state,url)  }   window.onpopstate = (event) = > {  this.setState({  location: {. this.state.location, pathname:window.location.pathname,  state:event.state  }  })  }   window.onpushstate = (state,pathname) = >{  this.setState({  location: {. this.state.location, pathname,  state  }  })  }   }   render(){  let that = this;  let value = {  location:that.state.location,  history: { push(to){// Define a history object with a push method to jump to the path  if(typeof to === 'object') { let {pathname,state} = to;  window.history.pushState(state,' ',pathname)  }else{  window.history.pushState(null.' ',to)  }  }  }  }  return (  // The value attribute is specifically provided to the context to store shared data  <Context.Provider value={value}>  {this.props.children}  </Context.Provider>  )  } } Copy the code

9,


This article is formatted using MDNICE