magic-ui
components
ui

dock

An implementation of the MacOS dock using react + tailwindcss + framer motion.

animated
flex
form
motion
transform
View Docs

Source Code

Files
dock
1"use client";
2 
3import { CalendarIcon, HomeIcon, MailIcon, PencilIcon } from "lucide-react";
4import Link from "next/link";
5import React from "react";
6 
7import { ModeToggle } from "@/components/mode-toggle";
8import { buttonVariants } from "@/components/ui/button";
9import { Separator } from "@/components/ui/separator";
10import {
11  Tooltip,
12  TooltipContent,
13  TooltipProvider,
14  TooltipTrigger,
15} from "@/components/ui/tooltip";
16import { cn } from "@/lib/utils";
17import { Dock, DockIcon } from "@/components/magicui/dock";
18 
19export type IconProps = React.HTMLAttributes<SVGElement>;
20 
21const Icons = {
22  calendar: (props: IconProps) => <CalendarIcon {...props} />,
23  email: (props: IconProps) => <MailIcon {...props} />,
24  linkedin: (props: IconProps) => (
25    <svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" {...props}>
26      <title>LinkedIn</title>
27      <path
28        fill="currentColor"
29        d="M20.447 20.452h-3.554v-5.569c0-1.328-.027-3.037-1.852-3.037-1.853 0-2.136 1.445-2.136 2.939v5.667H9.351V9h3.414v1.561h.046c.477-.9 1.637-1.85 3.37-1.85 3.601 0 4.267 2.37 4.267 5.455v6.286zM5.337 7.433c-1.144 0-2.063-.926-2.063-2.065 0-1.138.92-2.063 2.063-2.063 1.14 0 2.064.925 2.064 2.063 0 1.139-.925 2.065-2.064 2.065zm1.782 13.019H3.555V9h3.564v11.452zM22.225 0H1.771C.792 0 0 .774 0 1.729v20.542C0 23.227.792 24 1.771 24h20.451C23.2 24 24 23.227 24 22.271V1.729C24 .774 23.2 0 22.222 0h.003z"
30      />
31    </svg>
32  ),
33  x: (props: IconProps) => (
34    <svg viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" {...props}>
35      <title>X</title>
36      <path
37        fill="currentColor"
38        d="M18.901 1.153h3.68l-8.04 9.19L24 22.846h-7.406l-5.8-7.584-6.638 7.584H.474l8.6-9.83L0 1.154h7.594l5.243 6.932ZM17.61 20.644h2.039L6.486 3.24H4.298Z"
39      />
40    </svg>
41  ),
42  youtube: (props: IconProps) => (
43    <svg
44      width="32px"
45      height="32px"
46      viewBox="0 0 32 32"
47      fill="currentColor"
48      xmlns="http://www.w3.org/2000/svg"
49      {...props}
50    >
51      <title>youtube</title>
52      <path d="M29.41,9.26a3.5,3.5,0,0,0-2.47-2.47C24.76,6.2,16,6.2,16,6.2s-8.76,0-10.94.59A3.5,3.5,0,0,0,2.59,9.26,36.13,36.13,0,0,0,2,16a36.13,36.13,0,0,0,.59,6.74,3.5,3.5,0,0,0,2.47,2.47C7.24,25.8,16,25.8,16,25.8s8.76,0,10.94-.59a3.5,3.5,0,0,0,2.47-2.47A36.13,36.13,0,0,0,30,16,36.13,36.13,0,0,0,29.41,9.26ZM13.2,20.2V11.8L20.47,16Z" />
53    </svg>
54  ),
55  github: (props: IconProps) => (
56    <svg viewBox="0 0 438.549 438.549" {...props}>
57      <path
58        fill="currentColor"
59        d="M409.132 114.573c-19.608-33.596-46.205-60.194-79.798-79.8-33.598-19.607-70.277-29.408-110.063-29.408-39.781 0-76.472 9.804-110.063 29.408-33.596 19.605-60.192 46.204-79.8 79.8C9.803 148.168 0 184.854 0 224.63c0 47.78 13.94 90.745 41.827 128.906 27.884 38.164 63.906 64.572 108.063 79.227 5.14.954 8.945.283 11.419-1.996 2.475-2.282 3.711-5.14 3.711-8.562 0-.571-.049-5.708-.144-15.417a2549.81 2549.81 0 01-.144-25.406l-6.567 1.136c-4.187.767-9.469 1.092-15.846 1-6.374-.089-12.991-.757-19.842-1.999-6.854-1.231-13.229-4.086-19.13-8.559-5.898-4.473-10.085-10.328-12.56-17.556l-2.855-6.57c-1.903-4.374-4.899-9.233-8.992-14.559-4.093-5.331-8.232-8.945-12.419-10.848l-1.999-1.431c-1.332-.951-2.568-2.098-3.711-3.429-1.142-1.331-1.997-2.663-2.568-3.997-.572-1.335-.098-2.43 1.427-3.289 1.525-.859 4.281-1.276 8.28-1.276l5.708.853c3.807.763 8.516 3.042 14.133 6.851 5.614 3.806 10.229 8.754 13.846 14.842 4.38 7.806 9.657 13.754 15.846 17.847 6.184 4.093 12.419 6.136 18.699 6.136 6.28 0 11.704-.476 16.274-1.423 4.565-.952 8.848-2.383 12.847-4.285 1.713-12.758 6.377-22.559 13.988-29.41-10.848-1.14-20.601-2.857-29.264-5.14-8.658-2.286-17.605-5.996-26.835-11.14-9.235-5.137-16.896-11.516-22.985-19.126-6.09-7.614-11.088-17.61-14.987-29.979-3.901-12.374-5.852-26.648-5.852-42.826 0-23.035 7.52-42.637 22.557-58.817-7.044-17.318-6.379-36.732 1.997-58.24 5.52-1.715 13.706-.428 24.554 3.853 10.85 4.283 18.794 7.952 23.84 10.994 5.046 3.041 9.089 5.618 12.135 7.708 17.705-4.947 35.976-7.421 54.818-7.421s37.117 2.474 54.823 7.421l10.849-6.849c7.419-4.57 16.18-8.758 26.262-12.565 10.088-3.805 17.802-4.853 23.134-3.138 8.562 21.509 9.325 40.922 2.279 58.24 15.036 16.18 22.559 35.787 22.559 58.817 0 16.178-1.958 30.497-5.853 42.966-3.9 12.471-8.941 22.457-15.125 29.979-6.191 7.521-13.901 13.85-23.131 18.986-9.232 5.14-18.182 8.85-26.84 11.136-8.662 2.286-18.415 4.004-29.263 5.146 9.894 8.562 14.842 22.077 14.842 40.539v60.237c0 3.422 1.19 6.279 3.572 8.562 2.379 2.279 6.136 2.95 11.276 1.995 44.163-14.653 80.185-41.062 108.068-79.226 27.88-38.161 41.825-81.126 41.825-128.906-.01-39.771-9.818-76.454-29.414-110.049z"
60      ></path>
61    </svg>
62  ),
63};
64 
65const DATA = {
66  navbar: [
67    { href: "#", icon: HomeIcon, label: "Home" },
68    { href: "#", icon: PencilIcon, label: "Blog" },
69  ],
70  contact: {
71    social: {
72      GitHub: {
73        name: "GitHub",
74        url: "#",
75        icon: Icons.github,
76      },
77      LinkedIn: {
78        name: "LinkedIn",
79        url: "#",
80        icon: Icons.linkedin,
81      },
82      X: {
83        name: "X",
84        url: "#",
85        icon: Icons.x,
86      },
87      email: {
88        name: "Send Email",
89        url: "#",
90        icon: Icons.email,
91      },
92    },
93  },
94};
95 
96export function DockDemo() {
97  return (
98    <div className="relative flex h-[500px] w-full flex-col items-center justify-center overflow-hidden rounded-lg border bg-background md:shadow-xl">
99      <span className="pointer-events-none whitespace-pre-wrap bg-gradient-to-b from-black to-gray-300/80 bg-clip-text text-center text-8xl font-semibold leading-none text-transparent dark:from-white dark:to-slate-900/10">
100        Dock
101      </span>
102      <TooltipProvider>
103        <Dock direction="middle">
104          {DATA.navbar.map((item) => (
105            <DockIcon key={item.label}>
106              <Tooltip>
107                <TooltipTrigger asChild>
108                  <Link
109                    href={item.href}
110                    aria-label={item.label}
111                    className={cn(
112                      buttonVariants({ variant: "ghost", size: "icon" }),
113                      "size-12 rounded-full",
114                    )}
115                  >
116                    <item.icon className="size-4" />
117                  </Link>
118                </TooltipTrigger>
119                <TooltipContent>
120                  <p>{item.label}</p>
121                </TooltipContent>
122              </Tooltip>
123            </DockIcon>
124          ))}
125          <Separator orientation="vertical" className="h-full" />
126          {Object.entries(DATA.contact.social).map(([name, social]) => (
127            <DockIcon key={name}>
128              <Tooltip>
129                <TooltipTrigger asChild>
130                  <Link
131                    href={social.url}
132                    aria-label={social.name}
133                    className={cn(
134                      buttonVariants({ variant: "ghost", size: "icon" }),
135                      "size-12 rounded-full",
136                    )}
137                  >
138                    <social.icon className="size-4" />
139                  </Link>
140                </TooltipTrigger>
141                <TooltipContent>
142                  <p>{name}</p>
143                </TooltipContent>
144              </Tooltip>
145            </DockIcon>
146          ))}
147          <Separator orientation="vertical" className="h-full py-2" />
148          <DockIcon>
149            <Tooltip>
150              <TooltipTrigger asChild>
151                <ModeToggle className="rounded-full" />
152              </TooltipTrigger>
153              <TooltipContent>
154                <p>Theme</p>
155              </TooltipContent>
156            </Tooltip>
157          </DockIcon>
158        </Dock>
159      </TooltipProvider>
160    </div>
161  );
162}