Skip to content

Latest commit

 

History

History
115 lines (98 loc) · 2.83 KB

README.md

File metadata and controls

115 lines (98 loc) · 2.83 KB

React + TypeScript + Tanstack Router

This is a basic starter template for React + TypeScript + Tanstack Router.

it comprises of

Path Description
/ root layout + route
/profile profile route
/profile/$id dynamic profile route
/admin protected route
/auth/ login route (will redirect if already logged in)
/posts posts route + query loader
/post post route + query loader

tips

  • to use params in routes ( dynamic routes )
  const userRoute = new Route({
  getParentRoute: () => profileLayout,
  path: "$id",
  component: ProfileUser
})
  <Link to="/profile/$id" params={{ id: "marko" }}>
        Marko Profile
      </Link>
      <Link to="/profile/$id" params={{ id: "polo" }}>
        Polo Profile
      </Link>
  • to use search params
  export const authlayout = new Route({
  getParentRoute: () => rootLayout,
  path: "auth",
  component: AuthLayout,
  validateSearch: (search: Record<string, unknown>): AuthSearch => {
    // validate and parse the search params into a typed state
    return {
      // you can use zod or valibot to validate this 
      redirect: (search.redirect as string) ?? "/",
    };
  },
});
   import { authlayout } from "./config";
  const router = useRouter();
// include the route id for better typesafety
  const { redirect } = useSearch({ from: authlayout.id });

  //  if already , redirect fomr the login page to the recent page
  useEffect(() => {
    if (is_authed) {
      router.navigate({
        to: redirect,
        // @ts-expect-error
        search: (prev) => ({ redirect: prev?.redirect }),
      });
    }
  });
  • The above is being used in conjunction with a layout level auth guard
import { Outlet, useRouter } from "@tanstack/router";
import { useEffect } from "react";

interface AdminLayoutProps {}

export function AdminLayout({}: AdminLayoutProps) {
  const ls_is_authed = localStorage.getItem("is_authed");
  const is_authed = ls_is_authed === "true";
  const router = useRouter();
  useEffect(() => {
    if (!is_authed) {
      router.navigate({
        to: "/auth",
        search: {
          redirect: "/admin",
        },
      });
    }
  });
  return (
    <div className="w-full min-h-full flex flex-col items-center justify-center">
      <Outlet />
    </div>
  );
}
  • to get type safety the loaders you can import the route and use the loader from that
  const postsQuery = postsIndexRoute.useLoader()();
  • you can listen for navigation pending state and show a NProgress bar for better UX
  const router = useRouter();
  const status = router.state.status;
  ---- 
  ---- 
  <Nprogress isAnimating={status === "pending"} />