Rethinking our Remix admin routes

โœ๏ธ

How can we have multiple nested layout routes in Remix

29 Apr, 2022 ยท 7 min read

In the previous article, we first looked at shared layouts in Remix.

But let's take a deeper look at how we can refactor this a little bit.

The previous example used a /posts/admin structure, which is fine if we only have posts in our admin section. But what happens if we add categories? Now we have to replicate the whole admin structure there.

What I want to achieve is a separate /admin/{something} structure.

Note: You would fully split your admin from the front-end part in most projects.

In our case, we will mix the front-end with the admin part to ease our showcase example.

Creating the admin routes

The first thing we want to change is to have a basic admin folder. Create this admin folder inside your routes directory.

We have learned from the previous article that we will need to create an admin.tsx file in our routes directory. This file will serve as the main layout for the admin part.

Go ahead and create this file and use the following markup.

import { Link, Outlet } from '@remix-run/react';

export default function PostAdmin() {
  return (
    <div className='mx-auto max-w-4xl'>
      <h1 className='my-6 mb-2 border-b-2 text-center text-3xl'>
        My super cool admin
      </h1>
      <header className='border-b-2 mb-2 p-2'>
        <ul className='flex gap-3'>
          <li>
            <Link to={'/admin'} className='text-blue-600 underline'>
              Admin
            </Link>
          </li>
          <li>
            <Link to={'posts'} className='text-blue-600 underline'>
              Posts
            </Link>
          </li>
        </ul>
      </header>
      <main className='p-4'>
        <Outlet />
      </main>
    </div>
  );
}

This will render a basic header that can link to our admin dashboard and our admin post section.

If we now create a super simple index.tsx inside our admin folder, we can see if this router outlet works.

export default function AdminIndex() {
  return <p>Welcome to the admin panel</p>;
}

Go ahead and run your app. At this point, you should see the following result.

Basic Remix admin panel

The image rendered the big header, the menu with the two links, and the index file inside the outlet.

Creating the admin post section

Now we can create a posts folder inside our admin folder. This will be the home for our existing post admin.

And again, we can leverage the specific layout methods to create a posts.tsx file in our admin folder. (So one level up from the admin/posts/ folder)

This will be our layout again.

If you are following along, you already have this layout as it was previously the posts/admin.tsx file.

Move and rename this file, or copy-paste the below code.

import { json } from "@remix-run/node";
import { Link, Outlet, useLoaderData } from "@remix-run/react";
import { getPosts } from "~/models/post.server";
import { LoaderFunction } from "@remix-run/node";

type LoaderData = {
  posts: Awaited<ReturnType<typeof getPosts>>;
};

export const loader: LoaderFunction = async () => {
  return json({ posts: await getPosts() });
};

export default function PostAdmin() {
  const { posts } = useLoaderData() as LoaderData;

  return (
      <div className="grid grid-cols-4 gap-6">
        <nav className="col-span-4 md:col-span-1">
          <ul>
            {posts.map((post) => (
              <li key={post.slug}>
                <Link
                  to={`/posts/${post.slug}`}
                  className="text-blue-600 underline"
                >
                  {post.title}
                </Link>
              </li>
            ))}
          </ul>
        </nav>
        <main className="col-span-4 md:col-span-3">
          <Outlet />
        </main>
      </div>
  );
}

Inside the admin/posts folder, create the index.tsx file, our welcome screen for the post section.

import { Link } from '@remix-run/react';

export default function AdminIndex() {
  return (
    <p>
      <Link to='new' className='text-blue-600 underline'>
        Create a New Post
      </Link>
    </p>
  );
}

And then we also want to use the new post page, which we also created, you can simply move the new.tsx file to this admin/posts folder or create a new one with this content:

import { Form } from '@remix-run/react';
import { redirect } from '@remix-run/node';
import { createPost } from '~/models/post.server';

const inputClassName = `w-full rounded border border-gray-500 px-2 py-1 text-lg`;

export const action = async ({ request }) => {
  const formData = await request.formData();

  const title = formData.get('title');
  const slug = formData.get('slug');
  const content = formData.get('content');

  await createPost({ title, slug, content });
  return redirect('/posts/admin');
};

export default function NewPost() {
  return (
    <Form method='post'>
      <p>
        <label>
          Post Title:{' '}
          <input type='text' name='title' className={inputClassName} />
        </label>
      </p>
      <p>
        <label>
          Post Slug:{' '}
          <input type='text' name='slug' className={inputClassName} />
        </label>
      </p>
      <p>
        <label>
          Content:{' '}
          <input type='text' name='content' className={inputClassName} />
        </label>
      </p>
      <p className='text-right'>
        <button
          type='submit'
          className='rounded bg-blue-500 py-2 px-4 text-white hover:bg-blue-600 focus:bg-blue-400 disabled:bg-blue-300'
        >
          Create Post
        </button>
      </p>
    </Form>
  );
}

And that's it. We now migrated our post admin section to a global admin section.

This makes it easier to add some more admin options at a later stage.

You can find the completed code on GitHub.

Thank you for reading, and let's connect!

Thank you for reading my blog. Feel free to subscribe to my email newsletter and connect on Facebook or Twitter

Spread the knowledge with fellow developers on Twitter
Tweet this tip
Powered by Webmentions - Learn more

Read next ๐Ÿ“–

Adding Markdown plugins in Remix

15 May, 2022 ยท 2 min read

Adding Markdown plugins in Remix

Remix Markdown overview page

14 May, 2022 ยท 2 min read

Remix Markdown overview page