Native Tabs

PreviousNext

A sleek tabs component with a smooth sliding background indicator using Framer Motion layout animations. (Base UI)

Docs
uitripledcomponent

Preview

Loading preview…
components/native/baseui/native-tabs-baseui.tsx
"use client";

import { cn } from "@/lib/utils";
import { Tabs } from "@base-ui/react/tabs";
import { motion } from "framer-motion";
import { useState } from "react";

interface NativeTabsProps {
  items: {
    id: string;
    label: string;
    content: React.ReactNode;
  }[];
  defaultValue?: string;
  className?: string;
}

export function NativeTabs({
  items,
  defaultValue,
  className,
}: NativeTabsProps) {
  const [activeTab, setActiveTab] = useState(defaultValue || items[0].id);

  return (
    <Tabs.Root
      value={activeTab}
      onValueChange={(val) => setActiveTab(val as string)}
      className={cn("w-full max-w-md", className)}
    >
      <Tabs.List className="relative flex w-full items-center gap-1 rounded-xl bg-muted/50 p-1 border border-black/5 dark:border-white/5">
        {items.map((tab) => {
          const isActive = activeTab === tab.id;
          return (
            <Tabs.Tab
              key={tab.id}
              value={tab.id}
              className="relative z-10 flex-1 cursor-pointer rounded-lg px-4 py-2 text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 aria-selected:text-foreground text-muted-foreground hover:text-foreground bg-transparent border-0"
            >
              {isActive && (
                <motion.div
                  layoutId="active-tab-baseui"
                  className="absolute inset-0 z-[-1] rounded-lg bg-background shadow-sm border border-black/5 dark:border-white/5"
                  transition={{ type: "spring", bounce: 0.2, duration: 0.6 }}
                />
              )}
              {tab.label}
            </Tabs.Tab>
          );
        })}
      </Tabs.List>
      {items.map((item) => (
        <Tabs.Panel
          key={item.id}
          value={item.id}
          className="mt-4 overflow-hidden rounded-xl border bg-background p-6 shadow-sm focus:outline-none"
        >
          <motion.div
            initial={{ opacity: 0, x: -10 }}
            animate={{ opacity: 1, x: 0 }}
            exit={{ opacity: 0, x: 10 }}
            transition={{ duration: 0.2 }}
          >
            {item.content}
          </motion.div>
        </Tabs.Panel>
      ))}
    </Tabs.Root>
  );
}

Installation

npx shadcn@latest add @uitripled/native-tabs-baseui

Usage

import { NativeTabsBaseui } from "@/components/native-tabs-baseui"
<NativeTabsBaseui />