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

@stack-spot/citron-navigator-cli

v1.2.0

Published

Citron Navigator is a navigator for React web applications written with Typescript.

Downloads

2,119

Readme

Citron Navigator

Citron Navigator is a navigator for React web applications written with Typescript.

Main features:

  • Render content based on the current URL.
  • Use hooks to get the current route and url parameters from anywhere.
  • Easily figure if a route or any of its sub-routes are active.
  • Concentrate all route definitions in a single file.
  • Automatic serialization/deserialization of variables in the URL.
  • Type-safe! No more guessing which route accepts which parameters.
  • Context-aware: navigating from /resourceA/{resourceAId} to /resourceA/{resourceAId}/resourceB/{resourceBId} should take only resourceBId as a required parameter, resourceAId should be optional in this context.
  • Support for modular applications (e.g. Module Federation).

Motivation

We, at StackSpot, have some very large web applications, with an enormous amount of routes. Navigation became a problem where we couldn't know for sure the contract of each page, i.e. what parameters they could receive from the URL (route and search parameters). Moreover, it was impossible to easily see every route that existed in the application, since they would be all spread.

The current solutions for navigating a React web application don't embrace Typescript for a type-safe way of navigating, we normally have to implement type-safe mechanisms on our own in order to guarantee a navigation will always be performed correctly. Unfortunately, it is hard to force a developer to always use the "correct way of navigating" that we created locally, creating a big mess over time. Some times we'd have no type-check at all and it can become very easy to go to page that requires a variable without passing this variable.

Another big problem we faced without typed-navigation was getting the search parameters in a page. How would the developer know what search parameters the page can receive, where are they defined? How does the programmer make changes to these parameters? What's the correct way to deserialize the string in the URL?

It can become quite complex to manage url variables in a large application, we needed a library that would take care of this for us, so we created Citron Navigator.

How does it work?

Citron Navigator is based in a yaml file that declares the whole navigation tree and the parameters for each of the routes. See the example below:

+ root (/):
  + account (/account):
    + profile: (/profile):
    + changePassword: (/password):
    + billing (/billing):
      year: number
  + photoAlbums (/albums):
    search: string
    year: number
    month: number (1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12)
    limit: number
    page: number
    type: string ('ownedByMe' | 'sharedWithMe' | 'all')
    + album (/{albumId}):
      limit: number
      page: number
      + photo (/{photoId}):

The file above defines that we have 8 routes in our application: root, account, profile, changePassword, billing, photoAlbums, album and photo.

Paths

  • root: /
  • account: /account
  • profile: /account/profile
  • changePassword: /account/password
  • billing: /account/billing
  • photoAlbums: /albums
  • album: /albums/{albumId}
  • photo: /albums/{albumId}/{photoId}

Route parameters

  • root, account, profile, changePassword, billing and photoAlbums are pages that don't accept any route parameter.
  • album accepts an albumId as a route parameter.
  • photo accepts a photoId and an albumId (from the parent) as route parameters.

Search parameters

  • root, account, profile, changePassword and photo are pages that don't accept any search parameter.
  • billing accepts one search parameter: "year", and it must be a number.
  • photoAlbums accepts many search parameters, they are:
    • search: must be a string;
    • year: must be a number;
    • month: must be a number and be typed as 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12, not any number;
    • limit: must be a number;
    • page: must be a number;
    • type: must be a string and be typed as 'ownedByMe' | 'sharedWithMe' | 'all', not any string.
  • album accepts 2 search parameters:
    • limit: must be a number;
    • page: must be a number.

Code generation

The code for the navigator is generated based on the yaml file. It uses the runtime library to instantiate a navigator based on your specifications.

Usage

Let's render some content according to the current route:

export const Page = () => {
  const [content, setContent] = useState<React.Element>(<></>)
  
  useNavigationContext((context) => {
    context
      .when('root', props => setContent(<Home {...props} />))
      .when('root.account', props => setContent(<AccountDashboard {...props} />))
      .when('root.account.profile', props => setContent(<Profile {...props} />))
      .when('root.account.changePassword', props => setContent(<ChangePassword {...props} />))
      .when('root.account.billing', props => setContent(<Billing {...props} />))
      .when('root.photoAlbums', props => setContent(<PhotoAlbums {...props} />))
      .when('root.photoAlbums.album', props => setContent(<Album {...props} />))
      .when('root.photoAlbums.album.photo', props => setContent(<Photo {...props} />))
  })

  return content
}

Let's see the implementation for Album which is rendered when the route is root.photoAlbums.album:

const Album = ({ route, params: { albumId, limit, page } }: ViewPropsOf<'root.photoAlbums.album'>) => (
  <p>Album {albumId}</p>
  <p>limit is {limit}</p>
  <p>page is {page}</p>
  <p><Link to={route.$parent}>Go back to albums</Link></p>
  <p><Link to={route.photo} params={{ photoId: '001' }}>Check out this picture</Link></p>
)

Notice that, from "album", we can easily create a link to "photo" by passing only the photoId, the albumId is implicit.

When creating links or navigating to other pages, the type will always be checked by Typescript. In the example above, if we didn't pass photoId when creating a link to "photo", we'd get a type error and the code wouldn't build.

Attention: we used the component Link from Citron Navigator. This is necessary if you're not using hash based URLs (flag --useHash=false). If you are using hash based URLs (/#/path), then you can safely use a simple a tag instead. Example: <a href={route.$parent.$link()}>Go back to albums</a>.

Installation

We're going to use PNPM throughout this documentation, but feel free to use either NPM or YARN.

pnpm add -D @stack-spot/citron-navigator-cli
pnpm add @stack-spot/citron-navigator

citron-navigator-cli is responsible for generating the code while @stack-spot/citron-navigator is the runtime dependency.

Configuration

  1. If you're using git, ignore the file generated by the CLI. In .gitignore, add: "src/generated".
  2. Create the file navigation.yaml in the root of your project. This file should contain the definition for the routes in your application as showed in the first code example of this document.
  3. This is optional, but to make it easier to import navigation related structures, create an alias to src/generated/navigation.ts. In your tsconfig.json, add:
"paths": {
  "navigation": ["./src/generated/navigation"],
}

You should do the same to whatever bundler you're using. In Vite, for instance, you should add the following to vite.config.ts:

{
  resolve: {
    alias: {
      navigation: resolve(__dirname, './src/generated/navigation'),
    },
  },
}

Source code generation

pnpm citron

By default it will get the definitions from navigation.yaml and output the generated code to src/generated/navigation.ts. If you need to change this, pass the options --src and --out.

The navigator uses hash-based URLs by default (/#/route). To change this behavior, you can pass the option --useHash=false to the command citron.

It's a good idea to call citron before running the application locally or building, check the example below for Vite:

package.json:

{
  "scripts": {
    "dev": "citron && vite",
    "build": "citron && tsc && vite build --mode production",
  }
}

Documentation