Svelte navigation

A "data-loading friendly", type safe, config based router for Svelte. Inspired by svelte-spa-router.

Installation

npm install svelte-navigation

Quickstart

<script lang="ts">
    import About from "./routes/About.svelte";
    import Home from "./routes/Home.svelte";
    import type { Routes } from "svelte-navigation/models/route";
    import Router, { link } from "svelte-navigation/Router.svelte";

    const routes: Routes = {
        "/": Home,
        "/about": About,
    }
</script>

<nav>
  <ul>
    <li><a href="/" use:link>Home</a></li>
    <li><a href="/about" use:link>About</a></li>
  </ul>
</nav>


<Router {routes}/>

Routing

Basic routing

Routes should be defined as a javascript object and passed to the Router component as the routes prop.

Keys of this object are strings representing the path (with parameters), and values are Svelte components or another configuration object for more complex use cases.

<script lang="ts">
...
    const routes: Routes = {
        "/": Home,
        "/about": About,
        // With a path param
        "/posts/:slug": Post,
        // With an optional path param
        "/posts/:from/:to?": PostsOverview,
        // With a wildcard
        "/page/*": Page,
        // Fallback component, must be at the end
        "*": Fallback,
    }
</script>

<Router {routes}/>

The Router passes path and search params as props to the matching component.

// Post component
<script lang="ts">
  export let pathParams: PathParams
  export let serarchParams: SearchParams
</script>

<h1>The slug of this post is {pathParams.slug}</h1>

Async routes

Components can be dynamically imported to enable code-splitting.

const routes: Routes = {
  "/": Home,
  "/about": {
    asyncComponent: () => import("./About.svelte"),
  },
};

Pre-conditions

You can define a pre-condition (or an array of preconditions) that act as a guard for the route.

A pre-condition is a function that return either a boolean, a Promise of a boolean, a redirection path, or a Promise of a redirection path. These functions can take an optionnal object as parameter that can contain the location object, the path parameters and the search parameters.

async function asyncCondition(options: {
  location: RouterLocation,
  pathParams: PathParams,
  searchParams: SearchParams,
}) {
  // Do something with options arguments
  return Math.random() > 0.5 || "/login";
}

const routes: Routes = {
  "/": Home,
  "/about": {
    asyncComponent: () => import("./About.svelte"),
    conditions: [
      () => Math.random() > 0.5,
      () => asyncCondition,
      () => Promise.resolve(Math.random() > 0.5),
      () => Promise.resolve(Math.random() > 0.5 || "/login"),
    ],
  },
};

If the returned value is false, the navigation is canceled. If the result is a redirection path, the router performs a redirection to that location.

Data loading

If a loading function is provided, the router will execute it and pass the result as a data prop to the component.

For a lazy-loaded route, export a loadData function in a context="module" script tag in the route component.

<script lang="ts" context="module">
  export async function loadData(): Promise<Post[]> {
    const posts = await fetch("https://jsonplaceholder.typicode.com/posts");
    return await posts.json();
  }
</script>

<script lang="ts">
  import type { Post } from "../models/post";

  export let data: Post[];
</script>

{#each data as post}
  <p>{post.title}</p>
{/each}

For a statically-imported component route, put the loading function in the loadData property of the route.

import About, { loadData } from "About.svelte";

const routes: Routes = {
  "/": Home,
  "/about": {
    component: () => About,
    loadData,
  },
};

The loadData function can take an optionnal object as parameter that can contain the location object, the path parameters and the search parameters.

Type safety

You can provide a zod schema to ensure type-safety on path params and search params.

import Posts, { loadData } from "About.svelte";
import z from "zod";

const pathParamsSchema = z.object({
  locale: z.enum(["en", "fr", "es"]),
});

const seachParamsSchema = z.object({
  from: z.string().datetime(),
  to: z.string().datetime().optional(),
});

const routes: Routes = {
  "/": Home,
  "/:locale/posts/": {
    component: () => Posts,
    loadData,
    pathParamsSchema,
    seachParamsSchema,
  },
};

Zod schemas will be used during matching phase of the navigation. If params parsing fails, the router won't match the route and will try next routes.

Loading component

You can provide a loading component for a specifique route, or for the whole router. It will be displayed durring the lazy-loading, during the execution of preconditions and during data loading.

<script lang="ts">
    const routes = {
        "/": Home,
        // Will display Loading.svelte during loading
        "/about": {
          asyncComponent: () => import("./About.svelte"),
          loadingComponent: Loading
        },
        // With display GlobalLoading.svelte during loading
        "/posts/:slug": {
          component: Post,
          loadData,
        },
    }
</script>

<Router {routes} loadingComponent={GlobalLoading}/>

Location, path params and search params are passed to the loading component just like for route's components.

Error component

Just like loading components, you can provide a error component for a specifique route, or for the whole router. It will be displayed if an exception is thrown during navigation.

<script lang="ts">
    const routes = {
        "/": Home,
        // Will display Error.svelte if an exception is thrown during navigation
        "/about": {
          asyncComponent: () => import("./About.svelte"),
          errorComponent: Error
        },
        // Witl display GlobalError.svelte if an exception is thrown during navigation
        "/posts/:slug": {
          component: Post,
          loadData,
        },
    }
</script>

<Router {routes} loadingComponent={GlobalError}/>

Location, path params and search params are passed to the loading component just like for route's components.

Router events

The link action should be add to <a> elements to trigger a client side navigation. We use raw html elements because it is easier for styling than a framework specific component.

<script lang="ts">
  import { link } from "svelte-navigation/Router.svelte"
</script>

<a href="/about" use:link>A client side link</a>

Programatic navigation

back, forward, push and replace can be used to programmatically navigate. push and replace functions take a location string as argument.

Top categories

svelte logo

Need a Svelte website built?

Hire a professional Svelte developer today.
Loading Svelte Themes