Dropdown Menu

A dropdown menu to display actions.

import { Button } from "@/components/ui/button/button";
import {
  DropdownMenu,
  DropdownMenuItem,
  DropdownMenuPopup,
  DropdownMenuPortal,
  DropdownMenuPositioner,
  DropdownMenuSeparator,
  DropdownMenuSpacer,
  DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu/dropdown-menu";

export default function DropdownMenuDemo() {
  return (
    <DropdownMenu>
      <DropdownMenuTrigger
        render={
          <Button variant="outline">
            <span>Actions</span>
          </Button>
        }
      />
      <DropdownMenuPortal>
        <DropdownMenuPositioner sideOffset={8}>
          <DropdownMenuPopup render={<ul />}>
            <DropdownMenuSpacer />
            <DropdownMenuItem render={<li />}>Edit</DropdownMenuItem>
            <DropdownMenuItem render={<li />}>Duplicate</DropdownMenuItem>
            <DropdownMenuItem render={<li />}>Share</DropdownMenuItem>
            <DropdownMenuSeparator />
            <DropdownMenuItem render={<li />}>Archive</DropdownMenuItem>
            <DropdownMenuItem render={<li />}>Delete</DropdownMenuItem>
            <DropdownMenuSpacer />
          </DropdownMenuPopup>
        </DropdownMenuPositioner>
      </DropdownMenuPortal>
    </DropdownMenu>
  );
}

npx shadcn@latest add @roiui/dropdown-menu
npx shadcn@latest add @roiui/dropdown-menu-tailwind

anatomy
<DropdownMenu>
  <DropdownMenuTrigger />
  <DropdownMenuPortal>
      <DropdownMenuPositioner>
          <DropdownMenuPopup>
              <DropdownMenuArrow />
              <DropdownMenuItem />
              <DropdownMenuSeparator />
              <DropdownMenuSubmenuRoot>
                  <DropdownMenuSubmenuTrigger />
              </DropdownMenuSubmenuRoot>
          </DropdownMenuPopup>
      </DropdownMenuPositioner>
  </DropdownMenuPortal>
</DropdownMenu>

import { Button } from "@/components/ui/button/button";
import {
  DropdownMenu,
  DropdownMenuItem,
  DropdownMenuPopup,
  DropdownMenuPortal,
  DropdownMenuPositioner,
  DropdownMenuSeparator,
  DropdownMenuSubmenuRoot,
  DropdownMenuSubmenuTrigger,
  DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu/dropdown-menu";
import styles from "./dropdown-menu-submenu.module.css";

export default function DropdownMenuSubmenu() {
  return (
    <DropdownMenu>
      <DropdownMenuTrigger render={<Button variant="outline">Open Menu</Button>} />
      <DropdownMenuPortal>
        <DropdownMenuPositioner sideOffset={8}>
          <DropdownMenuPopup>
            <div className={styles.spacer} />
            <DropdownMenuItem>
              <span className={styles.menuItemText}>Profile</span>
            </DropdownMenuItem>
            <DropdownMenuItem>
              <span className={styles.menuItemText}>Settings</span>
            </DropdownMenuItem>
            <DropdownMenuSeparator />
            <DropdownMenuSubmenuRoot>
              <DropdownMenuSubmenuTrigger>
                <span className={styles.menuItemText}>More Options</span>
              </DropdownMenuSubmenuTrigger>
              <DropdownMenuPortal>
                <DropdownMenuPositioner align="start" alignOffset={-4} side="right" sideOffset={-4}>
                  <DropdownMenuPopup>
                    <div className={styles.spacer} />
                    <DropdownMenuItem>
                      <span className={styles.menuItemText}>Export</span>
                    </DropdownMenuItem>
                    <DropdownMenuItem>
                      <span className={styles.menuItemText}>Import</span>
                    </DropdownMenuItem>
                    <DropdownMenuItem>
                      <span className={styles.menuItemText}>Share</span>
                    </DropdownMenuItem>
                    <div className={styles.spacer} />
                  </DropdownMenuPopup>
                </DropdownMenuPositioner>
              </DropdownMenuPortal>
            </DropdownMenuSubmenuRoot>
            <DropdownMenuSeparator />
            <DropdownMenuItem>
              <span className={styles.menuItemText}>Logout</span>
            </DropdownMenuItem>
            <div className={styles.spacer} />
          </DropdownMenuPopup>
        </DropdownMenuPositioner>
      </DropdownMenuPortal>
    </DropdownMenu>
  );
}

"use client";
import { BookOpen, Building2, Check, CreditCard, HelpCircle, LogOut, Zap } from "lucide-react";
import { useState } from "react";
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar/avatar";
import { Badge } from "@/components/ui/badge/badge";
import { Button } from "@/components/ui/button/button";
import {
  DropdownMenu,
  DropdownMenuItem,
  DropdownMenuPopup,
  DropdownMenuPortal,
  DropdownMenuPositioner,
  DropdownMenuRadioGroup,
  DropdownMenuRadioItem,
  DropdownMenuRadioItemIndicator,
  DropdownMenuSeparator,
  DropdownMenuSubmenuRoot,
  DropdownMenuSubmenuTrigger,
  DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu/dropdown-menu";
import styles from "./dropdown-menu-profile.module.css";

export default function DropdownMenuProfile() {
  const [selectedOrg, setSelectedOrg] = useState("acme");

  return (
    <DropdownMenu>
      <DropdownMenuTrigger
        render={
          <Button className={styles.triggerButton} data-slot="dropdown-trigger" variant="outline">
            <Avatar style={{ width: "32px", height: "32px", flexShrink: 0 }}>
              <AvatarImage alt="preetecool's profile photo" src="/preetecool.png" />
              <AvatarFallback>PC</AvatarFallback>
            </Avatar>
            <div className={styles.triggerText}>
              <div className={styles.triggerName}>preetecool</div>
              <div className={styles.triggerOrg}>Acme Corp</div>
            </div>
          </Button>
        }
      />
      <DropdownMenuPortal>
        <DropdownMenuPositioner sideOffset={8}>
          <DropdownMenuPopup className={styles.popup} render={<ul />}>
            <div className={styles.spacer} />

            {/* Profile Section */}
            <div className={styles.profileSection}>
              <div className={styles.profileInfo}>
                <Avatar className={styles.avatar}>
                  <AvatarImage alt="preetecool's profile photo" src="/preetecool.png" />
                  <AvatarFallback>PC</AvatarFallback>
                </Avatar>
                <div className={styles.profileText}>
                  <div className={styles.profileName}>preetecool</div>
                  <div className={styles.profileEmail}>example@roiui.com</div>
                </div>
              </div>
              <Badge className={styles.freeBadge} size="sm" variant="secondary">
                free
              </Badge>
            </div>

            <DropdownMenuSeparator />

            {/* Organizations Submenu */}
            <DropdownMenuSubmenuRoot>
              <DropdownMenuSubmenuTrigger className={styles.submenuTriggerWithIcon} render={<li />}>
                <Building2 className={styles.submenuIcon} size={14} />
                Organizations
              </DropdownMenuSubmenuTrigger>
              <DropdownMenuPortal>
                <DropdownMenuPositioner align="start" alignOffset={-4} side="right" sideOffset={-4}>
                  <DropdownMenuPopup render={<ul />}>
                    <div className={styles.spacer} />
                    <DropdownMenuRadioGroup onValueChange={setSelectedOrg} value={selectedOrg}>
                      <DropdownMenuRadioItem className={styles.radioItem} render={<li />} value="acme">
                        <span className={styles.menuItemText}>Acme Corp</span>
                        <DropdownMenuRadioItemIndicator>
                          <Check size={14} />
                        </DropdownMenuRadioItemIndicator>
                      </DropdownMenuRadioItem>
                      <DropdownMenuRadioItem className={styles.radioItem} render={<li />} value="tech">
                        <span className={styles.menuItemText}>Tech Startup</span>
                        <DropdownMenuRadioItemIndicator>
                          <Check size={14} />
                        </DropdownMenuRadioItemIndicator>
                      </DropdownMenuRadioItem>
                      <DropdownMenuRadioItem className={styles.radioItem} render={<li />} value="design">
                        <span className={styles.menuItemText}>Design Studio</span>
                        <DropdownMenuRadioItemIndicator>
                          <Check size={14} />
                        </DropdownMenuRadioItemIndicator>
                      </DropdownMenuRadioItem>
                    </DropdownMenuRadioGroup>
                    <DropdownMenuSeparator />
                    <DropdownMenuItem render={<li />}>
                      <span className={styles.menuItemText}>Add new...</span>
                    </DropdownMenuItem>
                    <div className={styles.spacer} />
                  </DropdownMenuPopup>
                </DropdownMenuPositioner>
              </DropdownMenuPortal>
            </DropdownMenuSubmenuRoot>

            <DropdownMenuSeparator />

            <DropdownMenuItem icon={<Zap size={14} />} render={<li />}>
              Upgrade Plan
            </DropdownMenuItem>

            <DropdownMenuItem icon={<CreditCard size={14} />} render={<li />}>
              Billing
            </DropdownMenuItem>

            <DropdownMenuSeparator />

            <DropdownMenuItem icon={<HelpCircle size={14} />} render={<li />}>
              Get Help
            </DropdownMenuItem>

            <DropdownMenuItem icon={<BookOpen size={14} />} render={<li />}>
              Learn More
            </DropdownMenuItem>

            <DropdownMenuSeparator />

            <DropdownMenuItem icon={<LogOut size={14} />} render={<li />}>
              Log Out
            </DropdownMenuItem>

            <div className={styles.spacer} />
          </DropdownMenuPopup>
        </DropdownMenuPositioner>
      </DropdownMenuPortal>
    </DropdownMenu>
  );
}