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

cerebral-router

v0.9.2

Published

An opinionated URL change handler for Cerebral

Downloads

378

Readme

cerebral-router

An opinionated URL change handler for Cerebral

NPM version Build status Test coverage bitHound Score

Install

npm install cerebral-router

How to use

When you build your Cerebral application you do not have to think about routing at all. You only trigger signals that brings your application into the correct state. Let us imagine an application that can open a messages page, and also open specific messages.


import controller from './controller.js';
import homeOpened from './signals/homeOpened';
import messagesOpened from './signals/messagesOpened';
import messageOpened from './signals/messageOpened';

controller.signal('messagesOpened', messagesOpened);
controller.signal('messageOpened', messageOpened);

When we want to open the messages we call the signal:


onMessagesClick() {
  this.props.signals.messagesOpened();
}

When we want to open a single message we call the signal and pass a payload:


onMessageClick(id) {
  this.props.signals.messageOpened({
    id: id
  });
}

The signature of a state change is the signal and the payload passed. We can bind this signature to a route. Lets imagine we have implemented our whole application and it works great, we just need to update the addressbar with a url representing the current state of the application.

So let us also add a homeOpened signal so that we handle the root url as well.


import Router from 'cerebral-router';
import controller from './controller.js';
import homeOpened from './signals/homeOpened';
import messagesOpened from './signals/messagesOpened';
import messageOpened from './signals/messageOpened';

controller.signal('homeOpened', homeOpened);
controller.signal('messagesOpened', messagesOpened);
controller.signal('messageOpened', messageOpened);

Router(controller, {
  '/': 'homeOpened',
  '/messages': 'messagesOpened',
  '/messages/:id': 'messageOpened'
}, {
  mapper: { query: true } // Read about this below
});

Initial url would be handled automatically during application bootstrap if you are using cerebral-react (in container's componentDidMount method) or cerebral-angular (module's run section) packages. Otherwise you should call trigger method to ensure that initial url is handled.

The router checks the url and fires the signal related to the url. The url will be parsed and any payload will be passed on the signal. That means if you go to example.com/messages/123 it will trigger the messageOpened signal with the payload {id: '123'}.

But if you click a message in the list it will also trigger the messageOpened signal with the payload {id: '456'} and now the url will also update to example.com/messages/456.

So it works both ways!

The important thing to understand here is that your application does not trigger urls to change its state. It triggers signals. Then you bind a route to a signal to allow a url to trigger the signal as well.

That means:


// Going to url
"example.com/messages/456"

// Is exactly the same as
this.props.signals.messageOpened({
  id: '456'
});

Diving into the app from a url

In the example above, when navigating in the app, you have to go to /messages before you can go to /messages/456. But when you expose urls you could go directly to /messages/456. So how do you handle that?


...
controller.signal('messageOpened', [...messagesOpened, ...messageOpened]);

Router(controller, {
  '/': 'homeOpened',
  '/messages': 'messagesOpened',
  '/messages/:id': 'messageOpened'
});

With Cerebral you are already used to composing chains and actions together and this is also effective when creating routes. Now you might say, "I do not want to load my messages every time I open a message!". There are multiple ways to handle this. It depends on when you want to load the messages.

But lets say you want to load them whenever you actually go to /messages. Inside your messagesOpened signal you can just check if there is an ID on the input. If there is an ID it means you are about to open a message, if not it means you are just opening the messages.

What about queries?

With Cerebral you get a very powerful way to use queries. But first we have to make a statement: "Queries are produced by your application, not by users". With this perspective we can do some wonderful things. Lets get back to opening our message. Inside the component opening the message we want to pass more than the ID of the message. We want to pass: {withComments: true}. So that when we load the message, we load it with comments.


onMessageClick(id) {
  this.props.signals.messageOpened({
    id: id,
    withComments: true
  });
}

Since this signal is bound to a url Cerebral router will automatically make this part of the query, turning your url into example.com/messages/123?withComments:true. That means if you refresh or pass the url to somebody else it will pass {id: '123', withComments: true} as the payload to the signal, opening the message in the exact same way, with the comments.

Notice here that we have withComments:true, not withComment=true. This is because Cerebral router uses the URLON project to create serializable queries. As you can see it is very powerful.

Tell me more about how routes matched

cerebral-router relies on url-mapper default behavior:

  • path-to-regexp library is used to define routes and payload parameters. It tweaked to preserve payload parameters with Number and Boolean types, prepending it with colon.
  • Payload parameters not defined in route part is mapped to query using URLON notation if mapper: { query: true } option was passed to router.

Routable part of url is extracted based on onlyHash and baseUrl options provided to router.

Router(controller, {
  '/': 'homeOpened',
  '/messages': 'messagesOpened',
  '/messages/:id': 'messageOpened'
}, {
  onlyHash: true,
  baseUrl: '/path',
  mapper: { query: true }
});

With given config and https://example.com/path#/messages/123?withComments:true url /messages/123?withComments:true is routable part. Then url-mapper matches it to route /messages/:id and extracts payload {id: '123', withComments: true}.

I want my own scheme for routes as well as queries

Feel free to implement compileFn which return your own parse and stringify methods for given route. The only recommendation is: parse method for previously stringified payload should result the same payload. Just like JSON.parse(JSON.stringify(object)). Create an issue if you need this now since cerebral-router had to be patched to support custom mapper.