cult-ui
Input
Animation

family-button

Animated expansion inspired by Family.

animated
button
flex
gradient
hover
motion
positioning
text
transition

Source Code

Files
family-button
1"use client"
2 
3import { useMemo, useState } from "react"
4import { AnimatePresence, MotionConfig, motion } from "motion/react"
5import useMeasure from "react-use-measure"
6 
7import FamilyButton from "../ui/family-button"
8 
9export function FamilyButtonDemo() {
10  return (
11    <div className=" w-full h-full min-h-[240px]">
12      <div className="absolute bottom-4 right-4 ">
13        <FamilyButton>
14          <MusicPlayerExample />
15        </FamilyButton>
16      </div>
17    </div>
18  )
19}
20 
21let tabs = [
22  { id: 0, label: "Apple" },
23  { id: 1, label: "Spotify" },
24]
25 
26export function MusicPlayerExample() {
27  const [activeTab, setActiveTab] = useState(0)
28  const [direction, setDirection] = useState(0)
29  const [isAnimating, setIsAnimating] = useState(false)
30  const [ref, bounds] = useMeasure()
31 
32  const content = useMemo(() => {
33    switch (activeTab) {
34      case 0:
35        return (
36          <div className="flex items-center justify-center">
37            <svg
38              xmlns="http://www.w3.org/2000/svg"
39              xmlSpace="preserve"
40              viewBox="0 0 361 361"
41              width="5em"
42              height="5em"
43            >
44              <linearGradient
45                id="a"
46                x1={180}
47                x2={180}
48                y1={358.605}
49                y2={7.759}
50                gradientUnits="userSpaceOnUse"
51              >
52                <stop
53                  offset={0}
54                  style={{
55                    stopColor: "#fa233b",
56                  }}
57                />
58                <stop
59                  offset={1}
60                  style={{
61                    stopColor: "#fb5c74",
62                  }}
63                />
64              </linearGradient>
65              <path
66                d="M360 112.61c0-4.3 0-8.6-.02-12.9-.02-3.62-.06-7.24-.16-10.86-.21-7.89-.68-15.84-2.08-23.64-1.42-7.92-3.75-15.29-7.41-22.49a75.633 75.633 0 0 0-33.06-33.05c-7.19-3.66-14.56-5.98-22.47-7.41C287 .86 279.04.39 271.15.18c-3.62-.1-7.24-.14-10.86-.16-4.3-.02-8.6-.02-12.9-.02H112.61c-4.3 0-8.6 0-12.9.02-3.62.02-7.24.06-10.86.16C80.96.4 73 .86 65.2 2.27c-7.92 1.42-15.28 3.75-22.47 7.41A75.633 75.633 0 0 0 9.67 42.73c-3.66 7.2-5.99 14.57-7.41 22.49C.86 73.02.39 80.98.18 88.86.08 92.48.04 96.1.02 99.72 0 104.01 0 108.31 0 112.61v134.77c0 4.3 0 8.6.02 12.9.02 3.62.06 7.24.16 10.86.21 7.89.68 15.84 2.08 23.64 1.42 7.92 3.75 15.29 7.41 22.49a75.633 75.633 0 0 0 33.06 33.05c7.19 3.66 14.56 5.98 22.47 7.41 7.8 1.4 15.76 1.87 23.65 2.08 3.62.1 7.24.14 10.86.16 4.3.03 8.6.02 12.9.02h134.77c4.3 0 8.6 0 12.9-.02 3.62-.02 7.24-.06 10.86-.16 7.89-.21 15.85-.68 23.65-2.08 7.92-1.42 15.28-3.75 22.47-7.41a75.633 75.633 0 0 0 33.06-33.05c3.66-7.2 5.99-14.57 7.41-22.49 1.4-7.8 1.87-15.76 2.08-23.64.1-3.62.14-7.24.16-10.86.03-4.3.02-8.6.02-12.9V112.61z"
67                style={{
68                  fillRule: "evenodd",
69                  clipRule: "evenodd",
70                  fill: "url(#a)",
71                }}
72              />
73              <path
74                d="M254.5 55c-.87.08-8.6 1.45-9.53 1.64l-107 21.59-.04.01c-2.79.59-4.98 1.58-6.67 3-2.04 1.71-3.17 4.13-3.6 6.95-.09.6-.24 1.82-.24 3.62v133.92c0 3.13-.25 6.17-2.37 8.76s-4.74 3.37-7.81 3.99l-6.99 1.41c-8.84 1.78-14.59 2.99-19.8 5.01-4.98 1.93-8.71 4.39-11.68 7.51-5.89 6.17-8.28 14.54-7.46 22.38.7 6.69 3.71 13.09 8.88 17.82 3.49 3.2 7.85 5.63 12.99 6.66 5.33 1.07 11.01.7 19.31-.98 4.42-.89 8.56-2.28 12.5-4.61 3.9-2.3 7.24-5.37 9.85-9.11 2.62-3.75 4.31-7.92 5.24-12.35.96-4.57 1.19-8.7 1.19-13.26V142.81c0-6.22 1.76-7.86 6.78-9.08 0 0 88.94-17.94 93.09-18.75 5.79-1.11 8.52.54 8.52 6.61v79.29c0 3.14-.03 6.32-2.17 8.92-2.12 2.59-4.74 3.37-7.81 3.99l-6.99 1.41c-8.84 1.78-14.59 2.99-19.8 5.01-4.98 1.93-8.71 4.39-11.68 7.51-5.89 6.17-8.49 14.54-7.67 22.38.7 6.69 3.92 13.09 9.09 17.82 3.49 3.2 7.85 5.56 12.99 6.6 5.33 1.07 11.01.69 19.31-.98 4.42-.89 8.56-2.22 12.5-4.55 3.9-2.3 7.24-5.37 9.85-9.11 2.62-3.75 4.31-7.92 5.24-12.35.96-4.57 1-8.7 1-13.26V64.46c.02-6.16-3.23-9.96-9.02-9.46z"
75                style={{
76                  fillRule: "evenodd",
77                  clipRule: "evenodd",
78                  fill: "#fff",
79                }}
80              />
81            </svg>
82          </div>
83        )
84      case 1:
85        return (
86          <div className="flex items-center justify-center">
87            <svg
88              viewBox="0 0 256 256"
89              width="5em"
90              height="5em"
91              xmlns="http://www.w3.org/2000/svg"
92              preserveAspectRatio="xMidYMid"
93            >
94              <path
95                d="M128 0C57.308 0 0 57.309 0 128c0 70.696 57.309 128 128 128 70.697 0 128-57.304 128-128C256 57.314 198.697.007 127.998.007l.001-.006Zm58.699 184.614c-2.293 3.76-7.215 4.952-10.975 2.644-30.053-18.357-67.885-22.515-112.44-12.335a7.981 7.981 0 0 1-9.552-6.007 7.968 7.968 0 0 1 6-9.553c48.76-11.14 90.583-6.344 124.323 14.276 3.76 2.308 4.952 7.215 2.644 10.975Zm15.667-34.853c-2.89 4.695-9.034 6.178-13.726 3.289-34.406-21.148-86.853-27.273-127.548-14.92-5.278 1.594-10.852-1.38-12.454-6.649-1.59-5.278 1.386-10.842 6.655-12.446 46.485-14.106 104.275-7.273 143.787 17.007 4.692 2.89 6.175 9.034 3.286 13.72v-.001Zm1.345-36.293C162.457 88.964 94.394 86.71 55.007 98.666c-6.325 1.918-13.014-1.653-14.93-7.978-1.917-6.328 1.65-13.012 7.98-14.935C93.27 62.027 168.434 64.68 215.929 92.876c5.702 3.376 7.566 10.724 4.188 16.405-3.362 5.69-10.73 7.565-16.4 4.187h-.006Z"
96                fill="#1ED760"
97              />
98            </svg>
99          </div>
100        )
101      default:
102        return null
103    }
104  }, [activeTab])
105 
106  const handleTabClick = (newTabId: number) => {
107    if (newTabId !== activeTab && !isAnimating) {
108      const newDirection = newTabId > activeTab ? 1 : -1
109      setDirection(newDirection)
110      setActiveTab(newTabId)
111    }
112  }
113 
114  const variants = {
115    initial: (direction: number) => ({
116      x: 300 * direction,
117      opacity: 0,
118      filter: "blur(4px)",
119    }),
120    active: {
121      x: 0,
122      opacity: 1,
123      filter: "blur(0px)",
124    },
125    exit: (direction: number) => ({
126      x: -300 * direction,
127      opacity: 0,
128      filter: "blur(4px)",
129    }),
130  }
131 
132  return (
133    <div className="flex flex-col items-center pt-4 ">
134      <div className="flex space-x-1 border border-none rounded-[8px] cursor-pointer bg-neutral-700  px-[3px] py-[3.2px] shadow-inner-shadow">
135        {tabs.map((tab, i) => (
136          <button
137            key={`${tab.id}-i-${i}`}
138            onClick={() => handleTabClick(tab.id)}
139            className={`${
140              activeTab === tab.id ? "text-white " : "hover:text-neutral-300/60"
141            } relative rounded-[5px] px-3 py-1.5 text-xs sm:text-sm font-medium text-neutral-600  transition focus-visible:outline-1 focus-visible:ring-1 focus-visible:ring-blue-light focus-visible:outline-none`}
142            style={{ WebkitTapHighlightColor: "transparent" }}
143          >
144            {activeTab === tab.id && (
145              <motion.span
146                layoutId="family-bubble"
147                className="absolute inset-0 z-10 bg-neutral-800  mix-blend-difference shadow-inner-shadow"
148                style={{ borderRadius: 5 }}
149                transition={{ type: "spring", bounce: 0.19, duration: 0.4 }}
150              />
151            )}
152            {tab.label}
153          </button>
154        ))}
155      </div>
156      <MotionConfig transition={{ duration: 0.4, type: "spring", bounce: 0.2 }}>
157        <motion.div
158          className="relative mx-auto my-[10px] w-[60px] md:w-[150px] overflow-hidden"
159          initial={false}
160          animate={{ height: bounds.height }}
161        >
162          <div className="md:p-6 p-2" ref={ref}>
163            <AnimatePresence
164              custom={direction}
165              mode="popLayout"
166              onExitComplete={() => setIsAnimating(false)}
167            >
168              <motion.div
169                key={activeTab}
170                variants={variants}
171                initial="initial"
172                animate="active"
173                exit="exit"
174                custom={direction}
175                onAnimationStart={() => setIsAnimating(true)}
176                onAnimationComplete={() => setIsAnimating(false)}
177              >
178                {content}
179              </motion.div>
180            </AnimatePresence>
181          </div>
182        </motion.div>
183      </MotionConfig>
184    </div>
185  )
186}