Creating a Reusable Component Library with Next.js & Tailwind CSS

Introduction Building a reusable component library can significantly improve development efficiency, maintainability, and consistency across projects. In this guide, we’ll walk through setting up a reusable component library using Next.js and Tailwind CSS, ensuring it’s modular, customizable, and easy to integrate into various projects. 1. Setting Up the Project Initialize a New Next.js Project First, create a new Next.js project and install dependencies: npx create-next-app@latest next-component-library --typescript cd next-component-library Install Tailwind CSS Set up Tailwind CSS by installing its dependencies: npm install -D tailwindcss postcss autoprefixer npx tailwindcss init -p Configure tailwind.config.js: module.exports = { content: ["./src/**/*.{js,ts,jsx,tsx}"], theme: { extend: {}, }, plugins: [], }; Add Tailwind to globals.css: @tailwind base; @tailwind components; @tailwind utilities; 2. Structuring the Component Library Create a Components Directory Inside the src/ folder, create a components/ directory to organize reusable UI components. Example: Button Component Create a Button.tsx file inside components/: import React from "react"; import clsx from "clsx"; type ButtonProps = { children: React.ReactNode; variant?: "primary" | "secondary" | "danger"; onClick?: () => void; }; const Button: React.FC = ({ children, variant = "primary", onClick }) => { const buttonClasses = clsx( "px-4 py-2 rounded-md font-semibold transition", { "bg-blue-500 text-white hover:bg-blue-600": variant === "primary", "bg-gray-500 text-white hover:bg-gray-600": variant === "secondary", "bg-red-500 text-white hover:bg-red-600": variant === "danger", } ); return ( {children} ); }; export default Button; 3. Making Components Reusable & Customizable Using Props for Customization Props allow us to make components flexible. The Button component above supports different variants and actions. Supporting Tailwind Overrides To allow additional styles, modify the component to accept class names: const Button: React.FC = ({ children, variant = "primary", onClick, className }) => { return ( {children} ); }; 4. Creating More Components Example: Card Component const Card: React.FC = ({ title, children }) => { return ( {title} {children} ); }; export default Card; Example: Input Component const Input: React.FC = ({ placeholder, type = "text", className }) => { return ( ); }; export default Input; 5. Exporting Components for Easy Import To make the library easier to use, create an index.ts inside the components/ folder: export { default as Button } from "./Button"; export { default as Card } from "./Card"; export { default as Input } from "./Input"; Now, importing components is simple: import { Button, Card, Input } from "@/components"; 6. Publishing as an NPM Package (Optional) To share your component library, you can publish it as an npm package. Prepare the Package Create a package.json in the root directory. Add exports to enable modular imports. Ensure your components are inside a src/ folder. Build the Package Add a build script in package.json: "scripts": { "build": "tsc && tailwindcss -o styles.css" } Run the build command: npm run build Publish to NPM npm login npm publish Once published, it can be installed in other projects: npm install my-nextjs-ui-library Conclusion By following this guide, you’ve built a reusable component library using Next.js and Tailwind CSS. This modular approach helps maintain design consistency across projects while keeping components highly customizable. Next Steps: Expand the library with more components like modals, dropdowns, and alerts. Add Storybook for component documentation. Implement Tailwind’s @apply for reusable utility styles. Optimize the package for tree shaking. By making your components reusable and shareable, you enhance efficiency in Next.js projects while maintaining a beautiful and consistent UI.

Mar 13, 2025 - 13:27
 0
Creating a Reusable Component Library with Next.js & Tailwind CSS

Introduction

Building a reusable component library can significantly improve development efficiency, maintainability, and consistency across projects. In this guide, we’ll walk through setting up a reusable component library using Next.js and Tailwind CSS, ensuring it’s modular, customizable, and easy to integrate into various projects.

1. Setting Up the Project

Initialize a New Next.js Project

First, create a new Next.js project and install dependencies:

npx create-next-app@latest next-component-library --typescript
cd next-component-library

Install Tailwind CSS

Set up Tailwind CSS by installing its dependencies:

npm install -D tailwindcss postcss autoprefixer
npx tailwindcss init -p

Configure tailwind.config.js:

module.exports = {
  content: ["./src/**/*.{js,ts,jsx,tsx}"],
  theme: {
    extend: {},
  },
  plugins: [],
};

Add Tailwind to globals.css:

@tailwind base;
@tailwind components;
@tailwind utilities;

2. Structuring the Component Library

Create a Components Directory

Inside the src/ folder, create a components/ directory to organize reusable UI components.

Example: Button Component

Create a Button.tsx file inside components/:

import React from "react";
import clsx from "clsx";

type ButtonProps = {
  children: React.ReactNode;
  variant?: "primary" | "secondary" | "danger";
  onClick?: () => void;
};

const Button: React.FC<ButtonProps> = ({ children, variant = "primary", onClick }) => {
  const buttonClasses = clsx(
    "px-4 py-2 rounded-md font-semibold transition",
    {
      "bg-blue-500 text-white hover:bg-blue-600": variant === "primary",
      "bg-gray-500 text-white hover:bg-gray-600": variant === "secondary",
      "bg-red-500 text-white hover:bg-red-600": variant === "danger",
    }
  );
  return (
    <button className={buttonClasses} onClick={onClick}>
      {children}
    button>
  );
};

export default Button;

3. Making Components Reusable & Customizable

Using Props for Customization

Props allow us to make components flexible. The Button component above supports different variants and actions.

Supporting Tailwind Overrides

To allow additional styles, modify the component to accept class names:

const Button: React.FC<ButtonProps & { className?: string }> = ({ children, variant = "primary", onClick, className }) => {
  return (
    <button className={clsx(buttonClasses, className)} onClick={onClick}>
      {children}
    button>
  );
};

4. Creating More Components

Example: Card Component

const Card: React.FC<{ title: string; children: React.ReactNode }> = ({ title, children }) => {
  return (
    <div className="border rounded-lg p-4 shadow-md">
      <h3 className="font-semibold text-lg mb-2">{title}h3>
      {children}
    div>
  );
};
export default Card;

Example: Input Component

const Input: React.FC<{ placeholder: string; type?: string; className?: string }> = ({ placeholder, type = "text", className }) => {
  return (
    <input
      type={type}
      placeholder={placeholder}
      className={clsx("px-3 py-2 border rounded-md w-full", className)}
    />
  );
};
export default Input;

5. Exporting Components for Easy Import

To make the library easier to use, create an index.ts inside the components/ folder:

export { default as Button } from "./Button";
export { default as Card } from "./Card";
export { default as Input } from "./Input";

Now, importing components is simple:

import { Button, Card, Input } from "@/components";

6. Publishing as an NPM Package (Optional)

To share your component library, you can publish it as an npm package.

Prepare the Package

  • Create a package.json in the root directory.
  • Add exports to enable modular imports.
  • Ensure your components are inside a src/ folder.

Build the Package

  1. Add a build script in package.json:
"scripts": {
  "build": "tsc && tailwindcss -o styles.css"
}
  1. Run the build command:
npm run build

Publish to NPM

npm login
npm publish

Once published, it can be installed in other projects:

npm install my-nextjs-ui-library

Conclusion

By following this guide, you’ve built a reusable component library using Next.js and Tailwind CSS. This modular approach helps maintain design consistency across projects while keeping components highly customizable.

Next Steps:

  • Expand the library with more components like modals, dropdowns, and alerts.
  • Add Storybook for component documentation.
  • Implement Tailwind’s @apply for reusable utility styles.
  • Optimize the package for tree shaking.

By making your components reusable and shareable, you enhance efficiency in Next.js projects while maintaining a beautiful and consistent UI.