Next.js Discord

Discord Forum

Handle multiple roles that access the same route

Unanswered
Barbary Lion posted this in #help-forum
Open in Discord
Barbary LionOP
I am wondering what is the best approach to use if i have for example a page "/dashboard" but the content of the page will be different for each role "admin" or "employee", is it better to use an condition in the dashboard component to render the component based on the role (this will make the url path clean), or to use different routes for each role like "/dashboard/admin" and "/dashboard/employee" (this will give us the ability to use SSG) ??

32 Replies

West African Lion
I'm not knowledgeable enough to tell you which is best, but this is my approach
Barbary LionOP
@West African Lion your method is one of the two i mentioned which is good, but it let the url path not clean
i want instead of three paths "/teacher/dashboard" "/student/dashboard" "admin/dashboard"
, its better to have only "/dashboard"
and the content will be based on the role
West African Lion
Well .. I'd much rather separate each block on its own rather than worrying about url
Bro .. the URL doesn't matter .. no one is checking it
Btw .. why is the youcode site down? And when do they open registration?
Barbary LionOP
i think it will be open in may
@Barbary Lion and the content will be based on the role
ah it's to implement
please give me a moment
@Losti! ah it's to implement
Barbary LionOP
ok thanks
well, now i have to clear, you have to use parallel routes and intercepting routes.
it's very simple to use:

- app 
  - (dashboard)
    - dashboard
      @teacher
        - page.tsx
      @student
        - page.tsx
      layout.tsx
      page.tsx
Barbary LionOP
can you explain to me what is this
what the @ stand for
SUre
There are all details
but
Barbary LionOP
thanks, i will read about them and see if they are the best solution for my case
Believe me, i hope so.
West African Lion
Ohh Yeahh I remember reading about this .. thanks man .. it's actually a very good idea
Asiatic Lion
@Barbary Lion it would be great if you could be able to share your solution on this after getting one implemented? I've beent trying to get onve working for a while, but unfortunately I'm not quite getting this figured out.

There as attachment is my file structure, and here's code that I've in /dashboard/layout.tsx
// layout.tsx
import { redirect } from "next/navigation";
import { Suspense } from "react";
import FooterInfo from "~/app/_components/footer-info";
import MaxWidthWrapper from "~/app/_components/max-width-wrapper";
import LoadingSVG from "~/components/ui/loading-svg";
import { validateRequest } from "~/lib/auth";
import { TanstackProvider } from "~/providers/TanstackProvider";
import Navbar from "./_components/navbar";

export const dynamic = "force-dynamic";

export default async function DashboardLayout({
  modal,
  admin,
  contractor,
  customer,
  common,
}: {
  modal: React.ReactNode;
  admin: React.ReactNode;
  contractor: React.ReactNode;
  customer: React.ReactNode;
  common: React.ReactNode;
}) {
  const { user } = await validateRequest();

  // If no user, redirect to login
  if (!user) {
    return redirect("/login");
  }

  return (
    <TanstackProvider>
      <div className="grid h-full w-full grid-rows-[auto_1fr]">
        <Navbar />
        <main className="flex h-full flex-col overflow-y-auto pt-4">
          <div className="flex grow pb-8">
            <MaxWidthWrapper className="space-y-4">
              <Suspense fallback={<LoadingSVG size={64} />}>
                {user.role === "admin" && admin}
                {user.role === "contractor" && contractor}
                {user.role === "customer" && customer}
                {common}
              </Suspense>
            </MaxWidthWrapper>
          </div>
          <FooterInfo />
        </main>
        <Suspense fallback={<LoadingSVG size={64} />}>{modal}</Suspense>
      </div>
    </TanstackProvider>
  );
}
@Asiatic Lion <@612745907770884263> it would be great if you could be able to share your solution on this after getting one implemented? I've beent trying to get onve working for a while, but unfortunately I'm not quite getting this figured out. There as attachment is my file structure, and here's code that I've in `/dashboard/layout.tsx` typescript // layout.tsx import { redirect } from "next/navigation"; import { Suspense } from "react"; import FooterInfo from "~/app/_components/footer-info"; import MaxWidthWrapper from "~/app/_components/max-width-wrapper"; import LoadingSVG from "~/components/ui/loading-svg"; import { validateRequest } from "~/lib/auth"; import { TanstackProvider } from "~/providers/TanstackProvider"; import Navbar from "./_components/navbar"; export const dynamic = "force-dynamic"; export default async function DashboardLayout({ modal, admin, contractor, customer, common, }: { modal: React.ReactNode; admin: React.ReactNode; contractor: React.ReactNode; customer: React.ReactNode; common: React.ReactNode; }) { const { user } = await validateRequest(); // If no user, redirect to login if (!user) { return redirect("/login"); } return ( <TanstackProvider> <div className="grid h-full w-full grid-rows-[auto_1fr]"> <Navbar /> <main className="flex h-full flex-col overflow-y-auto pt-4"> <div className="flex grow pb-8"> <MaxWidthWrapper className="space-y-4"> <Suspense fallback={<LoadingSVG size={64} />}> {user.role === "admin" && admin} {user.role === "contractor" && contractor} {user.role === "customer" && customer} {common} </Suspense> </MaxWidthWrapper> </div> <FooterInfo /> </main> <Suspense fallback={<LoadingSVG size={64} />}>{modal}</Suspense> </div> </TanstackProvider> ); }
Barbary LionOP
oh sorry i forget to share what i did
I think its the same as your approach using the parallel routes
Asiatic Lion
Yeah that switch statement looks better approach than my:
    {user.role === "admin" && admin}
    {user.role === "contractor" && contractor}
    {user.role === "customer" && customer}
    {common}


Have to give that a try!
Asiatic Lion
I'm wondering how this solution performs, since I seems that despite of user role being admin, or manager or employee, all other layouts gets also triggered, despite of final output.

Example when I'm logged as contractor, I do get the page from @contractor folder, but server log also logs server action request from @admin/page.tsx which gives forbidded error, since my server actions checks user role before execution.