Handle multiple roles that access the same route
Unanswered
Barbary Lion posted this in #help-forum
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"
, 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?
Btw .. why is the youcode site down? And when do they open registration?
Barbary LionOP
i think it will be open in may
@West African Lion Well .. I'd much rather separate each block on its own rather than worrying about url
Barbary LionOP
hmm maybe you are right
@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
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
Yeah that switch statement looks better approach than my:
Have to give that a try!
{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
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.@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.
Barbary LionOP
It seems like also the other layouts are being triggered
unfortunately me also i don’t know how they performs under the hood