Simplify your stack and build anything. Or everything.
Build tomorrow’s web with a modern solution you truly own.
Code-based nature means you can build on top of it to power anything.
It’s time to take back your content infrastructure.
AuthorSandro Wegmann, 10xMedia

How to theme the Payload admin panel with Tailwind CSS 4

Community Guide
AuthorSandro Wegmann, 10xMedia

A step-by-step guide to theming Payload’s admin panel using Tailwind 4, custom logos, and CSS overrides.

What this guide will cover:

  • How to set up Tailwind CSS 4 in a Payload project?
  • Theming the Payload admin panel with your styles & colors
  • Adding custom logos and icons
  • Supporting Payload's dark mode
  • Fine-tuning components such as cards, buttons, and navigation

1. Set up a new Payload project

1
npx create-payload-app

Choose:

  • Template: blank
  • Database: MongoDB (can be local)

Once setup is complete, launch VS Code by running:

1
cd your-project-name
2
code .

And run pnpm dev to launch your dev server. Visit http://localhost:3000/admin and create your first user.

2. Install and configure Tailwind CSS 4

To install Tailwind, you will need to follow Tailwind installation instructions with Next.js, which are outlined below.

First, install the Tailwind peer dependencies. Go back into VS Code, and run:

1
pnpm install tailwindcss @tailwindcss/postcss postcss

The next thing is to create a postcss config file at the root: postcss.config.mjs

In this file, you'll paste the config from the Tailwind installation guide.

1
const config = {
2
plugins: {
3
"@tailwindcss/postcss": {},
4
},
5
};
6
export default config;

3. Split styles: global vs. Payload

Tailwind CSS 4 introduced a simpler setup with a single import line like @tailwind base; @tailwind components; @tailwind utilities;. This works great for your frontend — but it does not play nicely with Payload’s admin panel out of the box.

Why?

Payload uses its own internal CSS system to style the admin panel.

Tailwind’s default @tailwind base layer includes preflight styles, which reset browser styles globally — and these can unintentionally override Payload’s core styling.

So instead of importing Tailwind once for your entire project, we split styles into two separate files:

  • One for the frontend (e.g. your website)
  • One for the admin panel (Payload UI)

This lets you customize each environment without breaking the other.

Step-by-step

Create a src/styles folder and two files inside:

  • payloadStyles.css – for the Payload admin panel
  • global.css – for your frontend (optional)

payloadStyles.css (for admin panel):

1
/* src/styles/payloadStyles.css */
2
3
/* Define the Tailwind layers in the correct order */
4
@layer theme, base, components, utilities;
5
6
/* Import the Tailwind theme styles */
7
@import 'tailwindcss/theme.css' layer(theme);
8
9
/* Skip the preflight reset to avoid conflicts with Payload's admin styles */
10
/* @import 'tailwindcss/preflight.css' layer(base); */
11
12
/* Import Tailwind utilities */
13
@import 'tailwindcss/utilities.css' layer(utilities);

The above is a more advanced way to import TailwindCSS. It gives us the option to comment out the base layer and preflight to make it work with Payload. So it pretty much replaces the one-liner (@import "tailwindcss";), which we can then copy and use in globals.css for the frontend.

global.css (for frontend):

1
@import "tailwindcss";

Then import it inside src/app/payload/layout.tsx:

1
import '@/styles/payloadStyles.css';

This is everything we need to do to set up Tailwind 4 with a Payload project.

4. Enable Tailwind Dark Mode with Payload

Payload uses a data-theme="dark" attribute rather than relying on browser settings.

Add this to your payloadStyles.css:

1
@layer theme, base, components, utilities;
2
@import 'tailwindcss/theme.css' layer(theme);
3
/* @import 'tailwindcss/preflight.css' layer(base); */
4
@import 'tailwindcss/utilities.css' layer(utilities);
5
6
/* Support Payload’s dark mode toggle by adding a custom variant */
7
@custom-variant dark (&:where([data-theme=dark], [data-theme=dark] *));

Now Tailwind will respect the admin panel’s enforced light/dark mode.

5. Add custom logo and icon components

One of the great parts about Payload is the ability to white label the admin UI, essentially allowing you to brand the entire editing experience. So with Tailwind properly set, we'll now work on customizing the logo and icon components.

Note: You can see the dark and light mode icons & logos I've created here, but since they're not prepackaged for this guide, simply create your own to follow along.

You can’t import images directly into Payload’s admin config. Instead, you’ll need to create a custom React component that renders your logo or icon, and then reference that component in the Payload config to replace the default branding.

In order to do this, we're going to create a new folder structure: src/components/payload/

Inside, create logo.tsx and icon.tsx.

Logo.tsx:

1
import React from 'react';
2
import logo from '@/assets/lorem-logo.png'; // Make sure you have your correct images referenced here
3
import logoDark from '@/assets/lorem-logo-dark.png'; // Make sure you have your correct images referenced here
4
import Image from 'next/image';
5
6
export default function Logo() {
7
return (
8
<div>
9
<Image className="h-20 object-contain dark:hidden" src={logo} alt="" /> // This determines which logo would show based on user settings
10
<Image className="h-20 object-contain hidden dark:block" src={logoDark} alt="" />
11
</div>
12
);
13
}

Icon.tsx:

1
import React from 'react';
2
import logo from '@/assets/lorem-icon.png';
3
import Image from 'next/image';
4
5
export default function Icon() {
6
return (
7
<div>
8
<Image className="w-40" src={logo} alt="" /> // Note that we don't have a dark mode version because the icon is red
9
</div>
10
);
11
}

Next, update your payload.config.ts:

1
admin: {
2
components: {
3
graphics: {
4
Logo: '/components/payload/Logo',
5
Icon: '/components/payload/con',
6
},
7
},
8
}

Now, return to your admin panel to see if you can see your custom logo and icons.

If the assets do not appear right away, run:

1
pnpm payload generate:import-map

6. Customize colors, borders, and layout

All elements in the Payload admin UI have several utility classes. You can override them via CSS in payloadStyles.css.

For inspiration, we have a Payload UI theme generator, with a quick caveat: as of this guide, the theme generator doesn't reflect the latest version of Payload. So it will not work entirely. However, you can take a look at the generated CSS code and pick a few out that might be relevant for you, such as the hamburger wrapper or the theme accent color among others.

To style this, simply go back into payloadStyles.css and paste in whatever styles you'd like to apply. In the example below, we're changing theme elevation colors—in this case, just three variants of it. We'll also apply a background color to the cards in our dashboard.

Example:

1
@layer theme, base, components, utilities;
2
3
@import 'tailwindcss/theme.css' layer(theme);
4
/* @import 'tailwindcss/preflight.css' layer(base); */
5
@import 'tailwindcss/utilities.css' layer(utilities);
6
7
@custom-variant dark (&:where([data-theme=dark], [data-theme=dark] *));
8
9
:root {
10
--theme-elevation-100: #f06c9e;
11
--theme-elevation-200: #d06979;
12
--theme-elevation-800: #e31f3c;
13
}
14
15
.card {
16
background-color: var(--theme-elevation-100);
17
border: 1px solid var(--theme-elevation-200);
18
border-radius: 1px;
19
}

As you return to your admin panel, you should see your updates! Of course, you will want to make any additional adjustments to match your own brand, but I hope that has given you an idea of how to white label your Payload admin panel.