Animated expansion inspired by Family.
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}