How to write a VanillaJS Router

rohanbagchi

Rohan Bagchi

Posted on September 5, 2022

How to write a VanillaJS Router

Photo by <a href=Juanjo Jaramillo on Unsplash
"/>

The ask is like this:

  1. React to link clicks by showing route specific content on the page.
  2. When user clicks on browser back or forward, the correct route and its content should be loaded.
  3. If user refreshes, content for the current route should be loaded.

To start with, make your HTML page look like this:



<body>
    <div id="nav"></div>
    <div id="app"></div>

    <script src="src/index.js"></script>
</body>


Enter fullscreen mode Exit fullscreen mode

Now in our index.js, lets add the following config:



const routes = {
  '/': {
    linkLabel: 'Home',
    content: `I am in home page`
  },
  '/about': {
    linkLabel: 'About',
    content: `I am in about page`
  },
  '/friends': {
    linkLabel: 'Friends',
    content: `I am in friends page`,
  },
};


Enter fullscreen mode Exit fullscreen mode

At this point it is clear that we are trying to create a config powered router. This means, we will need to create the nav items and then register the click handlers.

Let's add functions for them.



const app = document.querySelector('#app');
const nav = document.querySelector('#nav');

// function to create new nav items
const renderNavlinks = () => {
  const navFragment = document.createDocumentFragment();
  Object.keys(routes).forEach(route => {
    const { linkLabel } = routes[route];

    const linkElement = document.createElement('a')
    linkElement.href = route;
    linkElement.textContent = linkLabel;
    linkElement.className = 'nav-link';
    navFragment.appendChild(linkElement);
  });

  nav.append(navFragment);
};

// function to register click handlers
const registerNavLinks = () => {
  nav.addEventListener('click', (e) => {
    e.preventDefault();
    const { href } = e.target;
    history.pushState({}, "", href);
    navigate(e); // pending implementation
  });
};


Enter fullscreen mode Exit fullscreen mode

We now need to implement the navigate function. It will handle the actual navigation.



const renderContent = route => app.innerHTML = routes[route].content;

const navigate = e => {
    const route = e.target.pathname;
    // this is responsible for adding the new path name to the history stack
    history.pushState({}, "", route);
    renderContent(route);
};


Enter fullscreen mode Exit fullscreen mode

Now, all we have left to do is to is to handle the pop state event (handle browser back and forth) and to handle the initial page load.



const registerBrowserBackAndForth = () => {
  window.onpopstate = function (e) {
    const route = location.pathname;
    renderContent(route);
  };
};

const renderInitialPage = () => {
  const route = location.pathname;
  renderContent(route);
};


Enter fullscreen mode Exit fullscreen mode

Bringing it all together:



(function bootup() {
  renderNavlinks();
  registerNavLinks();
  registerBrowserBackAndForth();
  renderInitialPage();
})();


Enter fullscreen mode Exit fullscreen mode

Final demo:
This is what we are attempting to create: (check codesandbox demo)

...

Thank you for reading this far.

💖 💪 🙅 🚩
rohanbagchi
Rohan Bagchi

Posted on September 5, 2022

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related