From b7722b9ba9b0c33b3ce75d872dab767bb5d70d2f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E5=8C=97=E6=9E=B3?= <7854742+wang_rumeng@user.noreply.gitee.com> Date: Mon, 23 Jun 2025 20:47:46 +0800 Subject: [PATCH] =?UTF-8?q?=E7=AC=AC=E4=B8=80=E7=89=88?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/create/layout.tsx | 9 + app/create/page.tsx | 9 +- app/create/script-to-video/page.tsx | 5 + app/create/video-to-video/page.tsx | 5 + app/globals.css | 24 +- app/layout.tsx | 4 +- app/page.tsx | 18 +- components/layout/dashboard-layout.tsx | 8 +- components/layout/sidebar.tsx | 6 +- components/layout/top-bar.tsx | 8 +- components/pages/create-video-workflow.tsx | 66 - components/pages/home-page2.tsx | 46 + components/pages/script-to-video.tsx | 7 + components/pages/style/home-page2.css | 75 + components/pages/style/video-to-video.css | 29 + components/pages/video-to-video.tsx | 783 ++++++++++ components/theme-provider.tsx | 2 +- .../workflow/generate-chapters-step.tsx | 6 +- components/workflow/generate-shots-step.tsx | 1265 ++++++++--------- components/workflow/input-script-step.tsx | 401 +++++- package-lock.json | 70 + package.json | 2 + plugins/liquid-glass/index.d.ts | 27 + plugins/liquid-glass/index.d.ts.map | 1 + plugins/liquid-glass/index.esm.js | 586 ++++++++ plugins/liquid-glass/index.js | 607 ++++++++ plugins/liquid-glass/shader-utils.d.ts | 25 + plugins/liquid-glass/shader-utils.d.ts.map | 1 + plugins/liquid-glass/utils.d.ts | 4 + plugins/liquid-glass/utils.d.ts.map | 1 + tailwind.config.js | 74 + 31 files changed, 3360 insertions(+), 814 deletions(-) create mode 100644 app/create/layout.tsx create mode 100644 app/create/script-to-video/page.tsx create mode 100644 app/create/video-to-video/page.tsx create mode 100644 components/pages/home-page2.tsx create mode 100644 components/pages/script-to-video.tsx create mode 100644 components/pages/style/home-page2.css create mode 100644 components/pages/style/video-to-video.css create mode 100644 components/pages/video-to-video.tsx create mode 100644 plugins/liquid-glass/index.d.ts create mode 100644 plugins/liquid-glass/index.d.ts.map create mode 100644 plugins/liquid-glass/index.esm.js create mode 100644 plugins/liquid-glass/index.js create mode 100644 plugins/liquid-glass/shader-utils.d.ts create mode 100644 plugins/liquid-glass/shader-utils.d.ts.map create mode 100644 plugins/liquid-glass/utils.d.ts create mode 100644 plugins/liquid-glass/utils.d.ts.map create mode 100644 tailwind.config.js diff --git a/app/create/layout.tsx b/app/create/layout.tsx new file mode 100644 index 0000000..8ccba09 --- /dev/null +++ b/app/create/layout.tsx @@ -0,0 +1,9 @@ +import { DashboardLayout } from '@/components/layout/dashboard-layout'; + +export default function CreateLayout({ + children, +}: { + children: React.ReactNode; +}) { + return {children}; +} \ No newline at end of file diff --git a/app/create/page.tsx b/app/create/page.tsx index 016734b..ade8853 100644 --- a/app/create/page.tsx +++ b/app/create/page.tsx @@ -1,10 +1,5 @@ -import { DashboardLayout } from '@/components/layout/dashboard-layout'; -import { CreateVideoWorkflow } from '@/components/pages/create-video-workflow'; +import { redirect } from 'next/navigation'; export default function CreatePage() { - return ( - - - - ); + redirect('/create/video-to-video'); } \ No newline at end of file diff --git a/app/create/script-to-video/page.tsx b/app/create/script-to-video/page.tsx new file mode 100644 index 0000000..2e38915 --- /dev/null +++ b/app/create/script-to-video/page.tsx @@ -0,0 +1,5 @@ +import { ScriptToVideo } from '@/components/pages/script-to-video'; + +export default function ScriptToVideoPage() { + return ; +} \ No newline at end of file diff --git a/app/create/video-to-video/page.tsx b/app/create/video-to-video/page.tsx new file mode 100644 index 0000000..9a8685b --- /dev/null +++ b/app/create/video-to-video/page.tsx @@ -0,0 +1,5 @@ +import { VideoToVideo } from '@/components/pages/video-to-video'; + +export default function VideoToVideoPage() { + return ; +} \ No newline at end of file diff --git a/app/globals.css b/app/globals.css index 52f5692..3fe7533 100644 --- a/app/globals.css +++ b/app/globals.css @@ -19,12 +19,12 @@ @layer base { :root { --background: 0 0% 100%; - --foreground: 0 0% 3.9%; + --foreground: 0 0% 98%; --card: 0 0% 100%; --card-foreground: 0 0% 3.9%; --popover: 0 0% 100%; --popover-foreground: 0 0% 3.9%; - --primary: 0 0% 9%; + --primary: 0 0% 98%; --primary-foreground: 0 0% 98%; --secondary: 0 0% 96.1%; --secondary-foreground: 0 0% 9%; @@ -44,6 +44,7 @@ --chart-5: 27 87% 67%; --radius: 0.5rem; } + .dark { --background: 0 0% 3.9%; --foreground: 0 0% 98%; @@ -72,6 +73,20 @@ } } +body { + height: 100vh; + overflow: hidden; + background: radial-gradient(circle at 50% 0, + rgba(255, 0, 0, 0.3), + rgba(255, 0, 0, 0) 70.71%), + radial-gradient(circle at 6.7% 75%, + rgba(0, 0, 255, 0.3), + rgba(0, 0, 255, 0) 70.71%), + radial-gradient(circle at 93.3% 75%, + rgba(0, 0, 255, 0.3), + rgba(0, 0, 255, 0) 70.71%) beige !important; +} + .hide-scrollbar::-webkit-scrollbar { display: none; } @@ -80,7 +95,8 @@ * { @apply border-border; } + body { - @apply bg-background text-foreground; + @apply text-foreground; } -} +} \ No newline at end of file diff --git a/app/layout.tsx b/app/layout.tsx index 61e4a5e..7c41c63 100644 --- a/app/layout.tsx +++ b/app/layout.tsx @@ -7,7 +7,7 @@ import { Toaster } from '@/components/ui/sonner'; const inter = Inter({ subsets: ['latin'] }); export const metadata: Metadata = { - title: 'AI Video Studio - Create Amazing Videos with AI', + title: 'AI Movie Flow - Create Amazing Videos with AI', description: 'Professional AI-powered video creation platform with advanced editing tools', }; @@ -21,7 +21,7 @@ export default function RootLayout({ diff --git a/app/page.tsx b/app/page.tsx index 872a03e..8b36bcd 100644 --- a/app/page.tsx +++ b/app/page.tsx @@ -1,10 +1,22 @@ import { DashboardLayout } from '@/components/layout/dashboard-layout'; -import { HomePage } from '@/components/pages/home-page'; +// import { HomePage } from '@/components/pages/home-page'; +import { HomePage2 } from '@/components/pages/home-page2'; export default function Home() { return ( - + ); -} \ No newline at end of file +} + +// import LiquidGlass from '@/plugins/liquid-glass' + +// export default function Home() { +// return ( +// // +//
+//

Hello World

+//
+// ); +// } \ No newline at end of file diff --git a/components/layout/dashboard-layout.tsx b/components/layout/dashboard-layout.tsx index 81f93d3..ddb9e57 100644 --- a/components/layout/dashboard-layout.tsx +++ b/components/layout/dashboard-layout.tsx @@ -9,14 +9,14 @@ interface DashboardLayoutProps { } export function DashboardLayout({ children }: DashboardLayoutProps) { - const [sidebarCollapsed, setSidebarCollapsed] = useState(false); + const [sidebarCollapsed, setSidebarCollapsed] = useState(true); return ( -
+
- -
+ +
{children}
diff --git a/components/layout/sidebar.tsx b/components/layout/sidebar.tsx index e37b9fd..2e99395 100644 --- a/components/layout/sidebar.tsx +++ b/components/layout/sidebar.tsx @@ -53,18 +53,18 @@ export function Sidebar({ collapsed, onToggle }: SidebarProps) { return (
{/* Logo */} -
+
{!collapsed && (
)} diff --git a/components/layout/top-bar.tsx b/components/layout/top-bar.tsx index c18d8fc..022ef6d 100644 --- a/components/layout/top-bar.tsx +++ b/components/layout/top-bar.tsx @@ -18,14 +18,14 @@ import { Bell, } from 'lucide-react'; -export function TopBar() { +export function TopBar({ collapsed }: { collapsed: boolean }) { const { theme, setTheme } = useTheme(); return ( -
+
-
-

AI Video Studio

+
+

Movie Flow

diff --git a/components/pages/create-video-workflow.tsx b/components/pages/create-video-workflow.tsx index 82632d5..88ae5d7 100644 --- a/components/pages/create-video-workflow.tsx +++ b/components/pages/create-video-workflow.tsx @@ -70,74 +70,8 @@ export function CreateVideoWorkflow() { } }; - const progress = ((currentStep - 1) / (steps.length - 1)) * 100; - return (
- {/* Header */} -
-
- -
- -
-

Create AI Video

-

- Follow these steps to create your AI-powered video. You can navigate back and forth to make changes. -

-
-
- - {/* Steps Navigation */} - - -
- {steps.map((step, index) => ( -
-
handleStepClick(step.id)} - > -
- {completedSteps.includes(step.id) ? ( - - ) : ( - - )} -
-
-
{step.name}
-
{step.description}
-
-
- {index < steps.length - 1 && ( - - )} -
- ))} -
-
-
- {/* Step Content */}
{renderStepContent()} diff --git a/components/pages/home-page2.tsx b/components/pages/home-page2.tsx new file mode 100644 index 0000000..bcc38d4 --- /dev/null +++ b/components/pages/home-page2.tsx @@ -0,0 +1,46 @@ +"use client"; + +import { useState, useRef } from "react"; +import { ArrowRight } from "lucide-react"; +import "./style/home-page2.css"; +import { useRouter } from "next/navigation"; + +export function HomePage2() { + const router = useRouter(); + return ( +
+
+
+
{ + router.push("/create/video-to-video"); + }}> +
+
Video To Video
+
Transform your video into a new style
+
+
+ {/* 图标 右箭头 */} + +
Create
+
+
+
+
+
+
+
Script To Video
+
Transform your script into a video
+
+
+ {/* 图标 右箭头 */} + +
Create
+
+
+
+
+
+
+
+ ); +} \ No newline at end of file diff --git a/components/pages/script-to-video.tsx b/components/pages/script-to-video.tsx new file mode 100644 index 0000000..8a8f28b --- /dev/null +++ b/components/pages/script-to-video.tsx @@ -0,0 +1,7 @@ +export function ScriptToVideo() { + return ( +
+

Script To Video

+
+ ); +} \ No newline at end of file diff --git a/components/pages/style/home-page2.css b/components/pages/style/home-page2.css new file mode 100644 index 0000000..0a8eb5f --- /dev/null +++ b/components/pages/style/home-page2.css @@ -0,0 +1,75 @@ +.tab-item:first-child { + position: relative; +} +.tab-item { + position: relative; + cursor: pointer; + --tw-bg-opacity: .2; + background-color: rgb(255 255 255 / var(--tw-bg-opacity)); + --tw-backdrop-blur: blur(15px); + backdrop-filter: var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia); + flex: 1 1 0%; + border-radius: 16px; +} + +.tab-item .content-wrapper { + position: relative; + height: 220px; + background-repeat: no-repeat; + background-size: 100% 100%; + overflow: hidden; + border-radius: 16px; + padding: 2rem; + background-position: left center, right center; +} +.tab-item .tab-title { + font-size: 36px; + font-weight: 600; + line-height: 64px; + --tw-text-opacity: 1; + color: rgb(255 255 255 / var(--tw-text-opacity)); +} +.tab-item .tab-description { + font-size: 16px; + font-weight: 300; + line-height: 22px; + color: rgba(255, 255, 255, 0.8); +} +.tab-item .tab-btn { + display: flex; + height: 40px; + min-width: 168px; + flex-direction: row; + align-items: center; + justify-content: center; + --tw-bg-opacity: -0.5; + background-color: rgb(89 0 255 / var(--tw-bg-opacity)); + padding-left: 16px; + padding-right: 16px; + --tw-text-opacity: 1; + color: rgb(255 255 255 / var(--tw-text-opacity)); + --tw-backdrop-blur: blur(10px); + backdrop-filter: var(--tw-backdrop-blur) var(--tw-backdrop-brightness) var(--tw-backdrop-contrast) var(--tw-backdrop-grayscale) var(--tw-backdrop-hue-rotate) var(--tw-backdrop-invert) var(--tw-backdrop-opacity) var(--tw-backdrop-saturate) var(--tw-backdrop-sepia); + gap: 6px; + overflow: hidden; + border-radius: 10px; +} +.tab-item .tab-btn .icon { + /* --tw-text-opacity: 1; + color: rgb(29 33 41 / var(--tw-text-opacity)); */ + transform: translate(-100px); + transition-property: transform; + transition-duration: 0.3s; +} +.tab-item .tab-btn .btn-text { + transform: translate(-14px); + transition-property: transform; + transition-duration: 0.3s; + font-weight: 600; +} +.tab-item:hover .hover-btn { + --tw-bg-opacity: 0.5; +} +.tab-item:hover .hover-btn .btn-text, .tab-item:hover .hover-btn .icon { + transform: translate(0px); +} \ No newline at end of file diff --git a/components/pages/style/video-to-video.css b/components/pages/style/video-to-video.css new file mode 100644 index 0000000..9e49b96 --- /dev/null +++ b/components/pages/style/video-to-video.css @@ -0,0 +1,29 @@ +.video-tool-component { + position: fixed; + left: 50%; + bottom: 1rem; + z-index: 99; + --tw-translate-x: calc(-50% + 34.5px); + transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skew(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y)); +} +.video-storyboard-tools .tool-submit-button { + display: flex; + height: 36px; + width: 120px; + cursor: pointer; + align-items: center; + justify-content: center; + gap: 2px; + border-radius: 10px; + font-size: .875rem; + line-height: 1.25rem; + font-weight: 600; + --tw-text-opacity: 1; + color: rgb(29 33 41 / var(--tw-text-opacity)); + background-color: #fff; +} +.video-storyboard-tools .tool-submit-button.disabled { + background-color: #fff; + opacity: .3; + cursor: not-allowed; +} \ No newline at end of file diff --git a/components/pages/video-to-video.tsx b/components/pages/video-to-video.tsx new file mode 100644 index 0000000..a4faf72 --- /dev/null +++ b/components/pages/video-to-video.tsx @@ -0,0 +1,783 @@ +"use client"; + +import { useState, useEffect, useRef } from 'react'; +import { Card } from '@/components/ui/card'; +import { Button } from '@/components/ui/button'; +import { ArrowLeft, ChevronDown, ChevronUp, Video, ListOrdered, Play, Loader2 } from 'lucide-react'; +import { useRouter } from 'next/navigation'; +import './style/video-to-video.css'; + +// 添加自定义滚动条样式 +const scrollbarStyles = ` + .custom-scrollbar::-webkit-scrollbar { + width: 4px; + } + .custom-scrollbar::-webkit-scrollbar-track { + background: rgba(255, 255, 255, 0.05); + border-radius: 2px; + } + .custom-scrollbar::-webkit-scrollbar-thumb { + background: rgba(255, 255, 255, 0.1); + border-radius: 2px; + } + .custom-scrollbar::-webkit-scrollbar-thumb:hover { + background: rgba(255, 255, 255, 0.2); + } +`; + +interface SceneVideo { + id: number; + video_url: string; + script: any; +} + +export function VideoToVideo() { + const router = useRouter(); + const [isUploading, setIsUploading] = useState(false); + const [isExpanded, setIsExpanded] = useState(false); + const [videoUrl, setVideoUrl] = useState(''); + const [isLoading, setIsLoading] = useState(false); + const [loadingText, setLoadingText] = useState('Generating...'); + const [generateObj, setGenerateObj] = useState({ + scripts: null, + frame_urls: null, + video_info: null, + scene_videos: null, + cut_video_url: null, + audio_video_url: null, + final_video_url: null + }); + const [showAllFrames, setShowAllFrames] = useState(false); + const containerRef = useRef(null); + const [showScrollNav, setShowScrollNav] = useState(false); + const [selectedVideoIndex, setSelectedVideoIndex] = useState(null); + const videosContainerRef = useRef(null); + const scriptsContainerRef = useRef(null); + + // 监听内容变化,自动滚动到底部 + useEffect(() => { + if (containerRef.current && (generateObj.scripts || generateObj.frame_urls || generateObj.video_info)) { + setTimeout(() => { + containerRef.current?.scrollTo({ + top: containerRef.current.scrollHeight, + behavior: 'smooth' + }); + }, 100); // 给一个小延迟,确保内容已经渲染 + } + }, [generateObj.scripts, generateObj.frame_urls, generateObj.video_info, generateObj.scene_videos, generateObj.cut_video_url, generateObj.audio_video_url, generateObj.final_video_url]); + + // 计算每行可以显示的图片数量(基于图片高度100px和容器宽度) + const imagesPerRow = Math.floor(1080 / (100 * 16/9 + 8)); // 假设图片宽高比16:9,间距8px + // 计算三行可以显示的最大图片数量 + const maxVisibleImages = imagesPerRow * 3; + + const handleUploadVideo = () => { + console.log('upload video'); + // 打开文件选择器 + const input = document.createElement('input'); + input.type = 'file'; + input.accept = 'video/*'; + input.onchange = (e) => { + const file = (e.target as HTMLInputElement).files?.[0]; + if (file) { + setVideoUrl(URL.createObjectURL(file)); + } + } + input.click(); + } + + const generateSences = async () => { + try { + generateObj.scene_videos = []; + const videoUrls = [ + 'https://cdn.qikongjian.com/videos/1750385931_99a8fb42-af89-4ae9-841a-a49869f026bd_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750389908_37d4fffa-8516-43a3-a423-fc0274f40e8a_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750384661_d8e30b79-828e-48cd-9025-ab62a996717c_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750320040_4b47996e-7c70-490e-8433-80c7df990fdd_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750303377_8c3c4ca6-c4ea-4376-8583-de3afa5681d8_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' + ]; + setGenerateObj({...generateObj, scene_videos: []}); + + // 使用 Promise.all 和 Array.map 来处理异步操作 + const promises = generateObj.scripts.map((element: any, index: number) => { + return new Promise((resolveVideo) => { + setTimeout(() => { + generateObj.scene_videos.push({ + id: index, + video_url: videoUrls[index], + script: element + }); + setGenerateObj({...generateObj}); + setLoadingText(`生成第 ${index + 1} 个分镜视频...`); + resolveVideo(); + }, index * 2000); // 每个视频间隔2秒 + }); + }); + + // 等待所有视频生成完成 + await Promise.all(promises); + } catch (error) { + console.error('生成分镜视频失败:', error); + throw error; + } + } + + const handleCreateVideo = async () => { + try { + // 清空所有数据 + setGenerateObj({ + scripts: null, + frame_urls: null, + video_info: null, + scene_videos: null, + cut_video_url: null, + audio_video_url: null, + final_video_url: null + }); + console.log('create video'); + setIsLoading(true); + setIsExpanded(true); + + // 提取帧 + await new Promise(resolve => setTimeout(resolve, 1000)); + setLoadingText('提取帧...'); + + // 生成帧 + await new Promise(resolve => setTimeout(resolve, 1000)); + generateObj.frame_urls = [ + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000000.jpg/1750507510_tmphfb431oc_000000.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000001.jpg/1750507511_tmphfb431oc_000001.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000002.jpg/1750507511_tmphfb431oc_000002.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000003.jpg/1750507511_tmphfb431oc_000003.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000004.jpg/1750507512_tmphfb431oc_000004.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000005.jpg/1750507512_tmphfb431oc_000005.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000006.jpg/1750507513_tmphfb431oc_000006.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000007.jpg/1750507514_tmphfb431oc_000007.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000008.jpg/1750507515_tmphfb431oc_000008.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000009.jpg/1750507515_tmphfb431oc_000009.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000010.jpg/1750507516_tmphfb431oc_000010.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000011.jpg/1750507516_tmphfb431oc_000011.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000012.jpg/1750507516_tmphfb431oc_000012.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000013.jpg/1750507517_tmphfb431oc_000013.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000014.jpg/1750507517_tmphfb431oc_000014.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000015.jpg/1750507517_tmphfb431oc_000015.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000016.jpg/1750507517_tmphfb431oc_000016.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000017.jpg/1750507517_tmphfb431oc_000017.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000018.jpg/1750507518_tmphfb431oc_000018.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000019.jpg/1750507520_tmphfb431oc_000019.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000020.jpg/1750507521_tmphfb431oc_000020.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000021.jpg/1750507523_tmphfb431oc_000021.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000022.jpg/1750507523_tmphfb431oc_000022.jpg", + "https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000023.jpg/1750507524_tmphfb431oc_000023.jpg", + ]; + setGenerateObj({...generateObj}); + setLoadingText('分析视频...'); + + // 生成视频信息 + await new Promise(resolve => setTimeout(resolve, 2000)); + generateObj.video_info = { + roles: [ + { + name: '雪 (YUKI)', + core_identity: '一位接近二十岁或二十出头的年轻女性,东亚裔,拥有深色长发和刘海,五官柔和且富有表现力。', + avatar: 'https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000000.jpg/1750507510_tmphfb431oc_000000.jpg', + },{ + name: '春 (HARU)', + core_identity: '一位接近二十岁或二十出头的年轻男性,东亚裔,拥有深色、发型整洁的中长发和深思的气质。', + avatar: 'https://smart-video-ai.oss-cn-beijing.aliyuncs.com/frames/d877fa43-4856-4acb-9a3b-627c28275343/frame_000000.jpg/1750507510_tmphfb431oc_000000.jpg', + } + ], + sence: '叙事在两个不同的时间段展开。现在时空设定在一个安静的乡下小镇,冬季时被大雪覆盖。记忆则设定在春夏两季,一个日本高中的校园内外,天气温暖而晴朗。', + style: '电影感,照片般逼真,带有柔和、梦幻般的质感。其美学风格让人联想到日本的浪漫剧情片。现在时空的场景使用冷色、蓝色调的调色板,而记忆序列则沐浴在温暖的黄金时刻光晕中。大量使用浅景深、微妙的镜头光晕,以及平滑、富有情感的节奏。8K分辨率。' + }; + setGenerateObj({...generateObj}); + setLoadingText('提取分镜脚本...'); + + // 生成分镜脚本 + await new Promise(resolve => setTimeout(resolve, 2000)); + generateObj.scripts = [ + { + shot: '面部特写,拉远至广角镜头。', + frame: '序列以雪(YUKI)面部的特写开场,她闭着眼睛躺在一片纯净的雪地里。柔和的雪花轻轻飘落在她的黑发和苍白的皮肤上。摄像机缓慢拉远,形成一幅令人惊叹的广角画面,揭示出在黄昏时分,她是在一片广阔、寂静、白雪覆盖的景观中的一个渺小、孤独的身影。', + atmosphere: '忧郁、宁静、寂静且寒冷。' + }, { + shot: '切至室内中景,随后是一个透过窗户的主观视角镜头。', + frame: '我们切到雪(YUKI)在一个舒适、光线温暖的卧室里醒来。她穿着一件舒适的毛衣,从床上下来,走向一扇窗户。她的呼吸在冰冷的玻璃上凝成雾气。她的主观视角镜头揭示了外面的雪景,远处有一座红色的房子,以及一封信被放入邮箱的记忆,从而触发了一段闪回。', + atmosphere: '怀旧、温暖、内省。' + }, { + shot: '跟踪镜头,随后是一系列切出镜头和特写。', + frame: '记忆开始。一个跟踪镜头跟随着一群学生,包括雪(YUKI),他们在飘落的樱花花瓣构成的华盖下走路上学。场景切到一个阳光普照的教室。雪(YUKI)穿着校服,坐在她的课桌前,害羞地瞥了一眼坐在前几排的春(HARU)。他仿佛感觉到她的凝视,巧妙地转过头来。', + atmosphere: '充满青春气息、怀旧、温暖,带有一种萌芽的、未言明的浪漫感。' + }, { + shot: '静态中景,切到另一个中景,营造出共享空间的感觉。', + frame: '记忆转移到学校图书馆,充满了金色的光束。一个中景镜头显示春(HARU)靠在一个书架上,全神贯注地读一本书。然后摄像机切到雪(YUKI),她坐在附近的一张桌子旁,专注于画架上的一幅小画,当她感觉到他的存在时,嘴唇上泛起一丝淡淡的、秘密的微笑。', + atmosphere: '平和、亲密、书卷气、温暖。' + }, { + shot: '雪(YUKI)的中景,过渡到通过相机镜头的特写主观视角。', + frame: '在一个阳光明媚的运动日,雪(YUKI)站在运动场边缘,拿着一台老式相机。她举起相机,镜头推进到她透过取景器看的眼睛的特写。我们切到她的主观视角:除了站在运动场上、表情专注而坚定的春(HARU)之外,整个世界都是失焦的。', + atmosphere: '充满活力、专注,一种投入的观察和遥远的钦佩感。' + }, + ]; + setGenerateObj({...generateObj}); // 使用展开运算符创建新对象,确保触发更新 + setLoadingText('生成分镜视频...'); + + // 生成分镜视频 + await new Promise(resolve => setTimeout(resolve, 2000)); + await generateSences(); + setLoadingText('分镜剪辑中...'); + + // 生成剪辑后的视频 + await new Promise(resolve => setTimeout(resolve, 2000)); + generateObj.cut_video_url = 'https://cdn.qikongjian.com/videos/1750385931_99a8fb42-af89-4ae9-841a-a49869f026bd_text_to_video_0.mp4'; + setGenerateObj({...generateObj}); + setLoadingText('口型同步中...'); + + // 口型同步后生成视频 + await new Promise(resolve => setTimeout(resolve, 2000)); + generateObj.audio_video_url = 'https://cdn.qikongjian.com/videos/1750385931_99a8fb42-af89-4ae9-841a-a49869f026bd_text_to_video_0.mp4'; + setGenerateObj({...generateObj}); + setLoadingText('一致化处理中...'); + + // 最终完成 + await new Promise(resolve => setTimeout(resolve, 2000)); + generateObj.final_video_url = 'https://cdn.qikongjian.com/videos/1750385931_99a8fb42-af89-4ae9-841a-a49869f026bd_text_to_video_0.mp4'; + setGenerateObj({...generateObj}); + setLoadingText('完成'); + setIsLoading(false); + } catch (error) { + console.error('视频生成过程出错:', error); + setLoadingText('生成失败,请重试'); + setIsLoading(false); + } + } + + // 处理视频选中 + const handleVideoSelect = (index: number) => { + setSelectedVideoIndex(index); + // 滚动脚本到对应位置 + const scriptElement = document.getElementById(`script-${index}`); + scriptElement?.scrollIntoView({ behavior: 'smooth', block: 'nearest', inline: 'center' }); + }; + + // 处理脚本选中 + const handleScriptSelect = (index: number) => { + setSelectedVideoIndex(index); + // 滚动视频到对应位置 + const videoElement = document.getElementById(`video-${index}`); + videoElement?.scrollIntoView({ behavior: 'smooth', block: 'nearest', inline: 'center' }); + }; + + return ( +
+ {/* 展示创作详细过程 */} + {generateObj && ( +
+ {generateObj.frame_urls && ( +
+
+ 提取帧 +
+
+
+ {generateObj.frame_urls.map((frame: string, index: number) => ( +
+ {`frame +
+ Frame {index + 1} +
+
+ ))} +
+ + {generateObj.frame_urls.length > maxVisibleImages && ( +
setShowAllFrames(!showAllFrames)} + > +
+ {showAllFrames ? ( + <> + + 收起 + + ) : ( + <> + + 展开全部 ({generateObj.frame_urls.length} 帧) + + )} +
+
+ )} +
+
+ )} + + {/* 视频信息 */} + {generateObj.video_info && ( +
+
+ 视频信息 +
+ {/* 展示:角色档案卡(头像、姓名、核心身份);场景;风格 */} +
+ {/* 角色档案卡 */} +
+ 角色档案: +
+ {generateObj.video_info.roles.map((role: any, index: number) => ( +
+
+
+ {role.name} +
+
+
+
{role.name}
+
{role.core_identity}
+
+
+ ))} +
+
+ + {/* 场景和风格 */} +
+
+ 场景 +

{generateObj.video_info.sence}

+
+
+ 风格 +

{generateObj.video_info.style}

+
+
+
+
+ )} + + {/* 分镜脚本 */} + {generateObj.scripts && ( +
+
+ 分镜脚本 +
+
+
+ {generateObj.scripts.map((script: any, index: number) => ( +
+ {/* 序号 */} +
+ Scene {index + 1} +
+ #{String(index + 1).padStart(2, '0')} +
+
+ + {/* 滚动内容区域 */} +
+
+ 镜头 +

{script.shot}

+
+ +
+ 场景 +

{script.frame}

+
+ +
+ 氛围 +

{script.atmosphere}

+
+
+
+ ))} +
+
+
+ )} + + {/* 分镜视频 */} + {generateObj.scene_videos && ( +
+
+ 分镜视频 +
+
+ {/* 视频展示区 */} +
+ {generateObj.scripts.map((script: any, index: number) => { + const video = generateObj.scene_videos.find((v: any) => v.id === index); + const isSelected = selectedVideoIndex === index; + + return ( +
handleVideoSelect(index)} + > + {video ? ( + <> +
+ ); + })} +
+ + {/* 脚本展示区 */} +
+ {generateObj.scripts.map((script: any, index: number) => { + const isSelected = selectedVideoIndex === index; + + return ( +
handleScriptSelect(index)} + > +
Scene {index + 1}
+
{script.frame}
+
+ ); + })} +
+
+
+ )} + + {/* 剪辑后的视频 */} + {generateObj.cut_video_url && ( +
+
+ 剪辑后的视频 +
+
+
+
+
+
+
+
+ )} + + {/* 口型同步后的视频 */} + {generateObj.audio_video_url && ( +
+
+ 口型同步后的视频 +
+
+
+
+
+
+
+
+ )} + + {/* 最终视频 */} + {generateObj.final_video_url && ( +
+
+ 最终视频 +
+
+
+
+
+
+
+
+ )} + + +
+ )} + + {/* 回滚条 */} +
setShowScrollNav(true)} + onMouseLeave={() => setShowScrollNav(false)} + > + {/* 悬浮按钮 */} + + + {/* 展开的回滚导航 */} +
+
+ {generateObj && ( + <> + {/* 进度条背景 */} +
+ + {/* 动态进度条 */} +
+ + {/* 步骤按钮 */} +
+ {generateObj.frame_urls && ( + + )} + + {generateObj.video_info && ( + + )} + + {generateObj.scripts && ( + + )} + + {generateObj.scene_videos && ( + + )} + + {generateObj.cut_video_url && ( + + )} + + {generateObj.audio_video_url && ( + + )} + + {generateObj.final_video_url && ( + + )} +
+ + )} +
+
+
+ + {/* 工具栏 */} +
+
+ {isExpanded ? ( +
setIsExpanded(false)}> + {/* 图标 展开按钮 */} + + Click to create +
+ ) : ( +
setIsExpanded(true)}> + {/* 图标 折叠按钮 */} + +
+ )} + +
+
+
+
+
+ {/* 图标 添加视频 */} +
+
+ Add Video +
+
+ {videoUrl && ( +
+
+
+
+ )} +
+
+ +
+
+
Stop
+
Create
+
+
+
+
+
+ + {/* Loading动画 */} + {isLoading && ( +
+
+ {/* 外圈动画 */} +
+ {/* 中圈动画 */} +
+ {/* 内圈动画 */} +
+
+
+
+ {/* Loading文字 */} +
+ {loadingText} + + . + . + . + +
+
+
+ )} +
+ ); +} \ No newline at end of file diff --git a/components/theme-provider.tsx b/components/theme-provider.tsx index 4203e6a..de1730b 100644 --- a/components/theme-provider.tsx +++ b/components/theme-provider.tsx @@ -5,5 +5,5 @@ import { ThemeProvider as NextThemesProvider } from "next-themes" import { type ThemeProviderProps } from "next-themes/dist/types" export function ThemeProvider({ children, ...props }: ThemeProviderProps) { - return {children} + return {children} } \ No newline at end of file diff --git a/components/workflow/generate-chapters-step.tsx b/components/workflow/generate-chapters-step.tsx index c0b0f55..2530d2b 100644 --- a/components/workflow/generate-chapters-step.tsx +++ b/components/workflow/generate-chapters-step.tsx @@ -116,11 +116,7 @@ export function GenerateChaptersStep({ onNext, onPrevious }: GenerateChaptersSte - -

- AI has automatically split your script into chapters and suggested actors for each section. - You can edit the content and assign multiple actors as needed. -

+
{chapters.map((chapter, index) => ( diff --git a/components/workflow/generate-shots-step.tsx b/components/workflow/generate-shots-step.tsx index 3f558c5..636ad56 100644 --- a/components/workflow/generate-shots-step.tsx +++ b/components/workflow/generate-shots-step.tsx @@ -19,7 +19,434 @@ interface GenerateShotsStepProps { onPrevious: () => void; } -const mockChapters = [ +interface Shot { + id: string; + type: string; + duration: number; + shotVideo: string; + generatedVideos: string[]; + description: string; + transition: string; + volume: number; + mediaNumber: number; +} + +interface Chapter { + id: number; + title: string; + shots: Shot[]; +} + +// 时间轴组件 +const TimelineView = ({ + chapters, + selectedShot, + onShotSelect, + onVideoCheck +}: { + chapters: Chapter[]; + selectedShot: string; + onShotSelect: (shotId: string) => void; + onVideoCheck?: () => void; +}) => ( +
+ {chapters.map((chapter) => ( +
+
+
Chapter {chapter.id}
+
+
+ {chapter.shots.map((shot) => ( +
onShotSelect(shot.id)} + > +
+
+
+ ))} +
+
+ ))} +
+); + +// 媒体信息项组件 +const MediaInfoItem = ({ + icon, + text, + popoverContent +}: { + icon: React.ReactNode; + text: string; + popoverContent?: React.ReactNode; +}) => ( +
+ {icon} + {text} + {popoverContent && ( + + + + + + {popoverContent} + + + )} +
+); + +// 查看视频弹窗 +const CheckVideoDialog = ({ + isOpen, + onClose, + currentShot +}: { + isOpen: boolean; + onClose: (open: boolean) => void; + currentShot?: Shot; +}) => ( + + + + Media history + +
+ {currentShot?.generatedVideos.map((video, index) => ( +
+
+ ))} +
+
+ + +
+
+
+); + +// 替换媒体弹窗 +const ReplaceMediaDialog = ({ + isOpen, + onClose, + chapters, + selectedShot, + onShotSelect, + activeTab, + setActiveTab +}: { + isOpen: boolean; + onClose: (open: boolean) => void; + chapters: Chapter[]; + selectedShot: string; + onShotSelect: (shotId: string) => void; + activeTab: string; + setActiveTab: (tab: string) => void; +}) => { + const replaceMediaTabs = [ + { value: 'uploaded', label: 'Uploaded media' }, + { value: 'stock', label: 'Stock media' }, + { value: 'generative', label: 'Generative media' }, + ]; + + return ( + + + + Replace media + +
+ + + + + {replaceMediaTabs.map((tab) => ( + {tab.label} + ))} + + +
+
+ + +
+
+
+

No media uploaded

+
+
+
+ +
+
+ +
+
+
+
+
+ + +
+
+
+ ); +}; + +// 媒体属性弹窗 +const MediaPropertyDialog = ({ + isOpen, + onClose, + chapters, + selectedShot, + onShotSelect, + currentShot, + activeTab, + setActiveTab +}: { + isOpen: boolean; + onClose: (open: boolean) => void; + chapters: Chapter[]; + selectedShot: string; + onShotSelect: (shotId: string) => void; + currentShot?: Shot; + activeTab: string; + setActiveTab: (tab: string) => void; +}) => { + const mediaPropertyTabs = [ + { value: 'media', label: 'Media' }, + { value: 'audio', label: 'Audio & SFX' }, + ]; + + return ( + + + + Media properties + +
+
+ + + {mediaPropertyTabs.map((tab) => ( + {tab.label} + ))} + + + +
+ +
00m : 10s : 500ms / 00m : 17s : 320ms
+
+ +
+ +
+ + +
+
+
+ + +
+
+ + +
+
+
+ +
+ +
+
+ + +
+
+ + +
+
+
+ +
+ +
+
+ + +
+
+ + +
+
+
+ +
+ + +
+ +
+ +
This part of the script is 21.00 seconds long.
+
There are 2 media attached to this part of the script:
+ + +
+
+ + +
+ +
Airplane Rocket Fire Close
+
+ +
+ + +
+ +
+ +
+ + + +
+
+
+
+
+ +
+ {currentShot && ( +
+
+
+ +
+ Chapter 1 / media 1 / People gathered in a city square to watch a fireworks display +
+ +
+
+
+ + 0:00 +
+ 0:12 +
+
+ {Array.from({ length: 40 }).map((_, i) => ( +
+ ))} +
+ +
+ Chapter 1 / Audio & SFX / Airplane Rocket Fire Close +
+
+
+ )} +
+
+ +
+ + +
+ +
+ ); +}; + +const mockChapters: Chapter[] = [ { id: 1, title: 'Chapter 1', @@ -57,85 +484,67 @@ const mockChapters = [ }, { id: 2, - title: 'Chapter 2', + title: 'Chapter 2', shots: [ { - id: '2-1', + id: '2-1', type: 'talking-head', - duration: 8, - shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - generatedVideos:[ - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' - ], - description: 'Opening welcome shot with character', - transition: 'Selected Automatically by Preset', - volume: 55, - mediaNumber: 1, - }, - { - id: '2-2', - type: 'b-roll', - duration: 9, - shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - generatedVideos:[ - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' - ], - description: 'Technology overview montage', - transition: 'Selected Automatically by Preset', - volume: 55, - mediaNumber: 2, - }, - { - id: '2-3', + duration: 8, + shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + generatedVideos:[ + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' + ], + description: 'Opening welcome shot with character', + transition: 'Selected Automatically by Preset', + volume: 55, + mediaNumber: 1, + }, + { + id: '2-2', + type: 'b-roll', + duration: 9, + shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + generatedVideos:[ + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' + ], + description: 'Technology overview montage', + transition: 'Selected Automatically by Preset', + volume: 55, + mediaNumber: 2, + }, + { + id: '2-3', type: 'animation', - duration: 10, - shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - generatedVideos:[ - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' - ], - description: 'Animation sequence', - transition: 'Selected Automatically by Preset', - volume: 55, - mediaNumber: 3, - }, - { - id: '2-4', - type: 'talking-head', - duration: 8, - shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - generatedVideos:[ - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', - 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' - ], - description: 'Character dialogue', - transition: 'Selected Automatically by Preset', - volume: 55, - mediaNumber: 4, - } - ] - } -]; - -const transitionTypes = [ - { value: 'fade', label: 'Fade' }, - { value: 'slide', label: 'Slide' }, - { value: 'zoom', label: 'Zoom' }, - { value: 'cut', label: 'Cut' }, -]; - -const replaceMediaTabs = [ - { value: 'uploaded', label: 'Uploaded media' }, - { value: 'stock', label: 'Stock media' }, - { value: 'generative', label: 'Generative media' }, -]; - -const mediaPropertyTabs = [ - { value: 'media', label: 'Media' }, - { value: 'audio', label: 'Audio & SFX' }, + duration: 10, + shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + generatedVideos:[ + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' + ], + description: 'Animation sequence', + transition: 'Selected Automatically by Preset', + volume: 55, + mediaNumber: 3, + }, + { + id: '2-4', + type: 'talking-head', + duration: 8, + shotVideo: 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + generatedVideos:[ + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4', + 'https://cdn.qikongjian.com/videos/1750162598_db834807-3f1c-4c52-8f50-b25396bd73ef_text_to_video_0.mp4' + ], + description: 'Character dialogue', + transition: 'Selected Automatically by Preset', + volume: 55, + mediaNumber: 4, + } + ] + } ]; export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps) { @@ -144,14 +553,15 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps const [chapters, setChapters] = useState(mockChapters); const videoRef = useRef(null); const [isPlaying, setIsPlaying] = useState(false); - const currentChapter = chapters.find(ch => ch.id === selectedChapter); - const currentShot = currentChapter?.shots.find(shot => shot.id === selectedShot); const [isCheckVideoOpen, setIsCheckVideoOpen] = useState(false); const [isReplaceMediaOpen, setIsReplaceMediaOpen] = useState(false); const [isMediaPropertyOpen, setIsMediaPropertyOpen] = useState(false); const [activeTabReplaceMedia, setActiveTabReplaceMedia] = useState('uploaded'); const [activeTabMediaProperty, setActiveTabMediaProperty] = useState('media'); + const currentChapter = chapters.find(ch => ch.id === selectedChapter); + const currentShot = currentChapter?.shots.find(shot => shot.id === selectedShot); + const handlePlayPause = () => { if (videoRef.current) { if (videoRef.current.paused) { @@ -213,473 +623,43 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps return (
- {/* 分镜视频列表 弹窗 */} - - - - Media history - -
- {currentShot?.generatedVideos.map((video, index) => ( -
-
- ))} -
- {/* Apply 按钮 Cancel 按钮 */} -
- - -
-
-
- {/* 替换媒体 弹窗;点击弹窗以外 不触发关闭弹窗 */} - - - - Replace media - - {/* 占剩余高度 溢出滚动 */} -
- {/* 章节列表 */} -
- {chapters.map((chapter, index) => ( -
- {/* Chapter: index + 1 文字竖着显示 */} -
-
Chapter {chapter.id}
-
- {/* flex: 1 */} -
- {chapter.shots.map((shot, index) => ( -
setSelectedShot(shot.id)} - > -
-
-
- ))} -
-
- ))} -
+ {/* 弹窗组件 */} + + + + + - {/* 标签页 */} - - - {replaceMediaTabs.map((tab) => ( - {tab.label} - ))} - - -
- {/* 上传媒体 */} - {/* 上传按钮;筛选下拉框 视频 图片;一行展示 两边对齐 */} -
- - -
- {/* 媒体库 */} -
- {/* 媒体库 网格展示 */} -
- {/* 没有数据 提示 请上传媒体 */} -
-

No media uploaded

-
-
-
- -
- {/* 素材库 */} -
-
- -
- {/* 生成媒体 */} -
-
-
-
- {/* Apply 按钮 Cancel 按钮 */} -
- - -
-
-
- {/* 媒体属性 弹窗 高度全屏 */} - - - - Media properties - -
- {/* 左侧内容区域 */} -
- {/* 标签页 */} - - - {mediaPropertyTabs.map((tab) => ( - {tab.label} - ))} - - - - {/* Duration */} -
- -
00m : 10s : 500ms / 00m : 17s : 320ms
-
- - {/* Trim */} -
- -
- - -
-
-
- - -
-
- - -
-
-
- - {/* Center point */} -
- -
-
- - -
-
- - -
-
-
- - {/* Zoom & Rotation */} -
- -
-
- - -
-
- - -
-
-
- - {/* Transition */} -
- - -
- - {/* Script */} -
- -
This part of the script is 21.00 seconds long.
-
There are 2 media attached to this part of the script:
- - {/* 章节列表 */} -
- {chapters.map((chapter, index) => ( -
- {/* Chapter: index + 1 文字竖着显示 */} -
-
Chapter {chapter.id}
-
- {/* flex: 1 */} -
- {chapter.shots.map((shot, index) => ( -
setSelectedShot(shot.id)} - > -
-
-
- ))} -
-
- ))} -
-
-
- - - {/* SFX name */} -
- -
Airplane Rocket Fire Close
-
- - {/* SFX volume */} -
- - -
- - {/* Replace audio */} -
- -
- - - -
-
-
-
-
- - {/* 右侧预览区域 */} -
- {currentShot && ( -
- {/* 视频预览 */} -
-
- -
- Chapter 1 / media 1 / People gathered in a city square to watch a fireworks display -
- - {/* 音频波形 */} -
-
-
- - 0:00 -
- 0:12 -
- {/* 简单的音频波形显示 */} -
- {Array.from({ length: 40 }).map((_, i) => ( -
- ))} -
- -
- Chapter 1 / Audio & SFX / Airplane Rocket Fire Close -
-
-
- )} -
-
- - {/* 底部按钮 */} -
- - -
- -
{/* Timeline Header */} -
- {/* 章节列表 */} -
- {chapters.map((chapter, index) => ( -
- {/* Chapter: index + 1 文字竖着显示 */} -
-
Chapter {chapter.id}
-
- {/* flex: 1 */} -
- {chapter.shots.map((shot, index) => ( -
setSelectedShot(shot.id)} - > -
-
-
- ))} -
-
- ))} -
+
+ setIsCheckVideoOpen(true)} + /> - {/* Chinese Text 不换行 溢出可左右滚动 隐藏滚动条 */}
但我决心要改变它。我的翅膀展开,在千星的光芒中翱翔,降临人生。和星光铸就,我是凤青楗,这就是我重生的故事。不......?
@@ -687,9 +667,7 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps {/* Main Content */}
- {/* Left Panel */}
- {/* Replace Media Section */} {currentShot && (
@@ -698,7 +676,7 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps
- - - -
-
Delete media
-
-
Delete and add blank media
+ } + text={`Chapter 1 / media ${currentShot.mediaNumber} / Generated media`} + popoverContent={ +
+
Delete media
+
+
Delete and add blank media
+
+
+ } + /> + + } + text="00m : 08s : 070ms / 00m : 08s : 070ms" + popoverContent={ +
+
Trim
+
+
+ + +
+
+ + + + +
+
+ +
- - -
+
+ } + /> - {/* Duration */} -
- {/* 标尺图标 */} - - - 00m : 08s : 070ms / 00m : 08s : 070ms - - - - - - -
-
Trim
-
- {/* checkbox */} -
- - -
-
- - - - -
- {/* 按钮 */} -
- - -
+ } + text={`Transition: ${currentShot.transition}`} + popoverContent={ +
+
Transition
+
+
+ +
+
+ +
- - -
+
+ } + /> - {/* Transition */} -
- {/* 图标 */} - - - Transition: {currentShot.transition} - - - - - - -
-
Transition
-
- {/* select */} -
- -
- {/* 按钮 */} -
- - -
-
-
-
-
-
- - {/* Audio Volume */}
@@ -845,10 +785,17 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps
+ + {/* Media Properties */} +
+
+

handleOpenMediaProperty('media')}>Media properties

+
+
- {/* Right Panel - Preview */} -
+ +
{currentShot && (
@@ -862,7 +809,6 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps loop={false} />
- {/* 刷新图标 */} @@ -871,24 +817,17 @@ export function GenerateShotsStep({ onNext, onPrevious }: GenerateShotsStepProps
)} - - {/* Media Properties */} -
-
-

handleOpenMediaProperty('media')}>Media properties

-
-
{/* Action Buttons */} -
- -
diff --git a/components/workflow/input-script-step.tsx b/components/workflow/input-script-step.tsx index 94ce9fd..65667fd 100644 --- a/components/workflow/input-script-step.tsx +++ b/components/workflow/input-script-step.tsx @@ -1,6 +1,6 @@ "use client"; -import { useState } from 'react'; +import { useState, useEffect } from 'react'; import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'; import { Button } from '@/components/ui/button'; import { Textarea } from '@/components/ui/textarea'; @@ -9,41 +9,372 @@ import { Label } from '@/components/ui/label'; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from '@/components/ui/select'; import { Badge } from '@/components/ui/badge'; import { Separator } from '@/components/ui/separator'; -import { ArrowRight, Sparkles, Users, FileText } from 'lucide-react'; +import { ArrowRight, Sparkles, Users, FileText, Play, Pause, RefreshCw, Palette, Volume2 } from 'lucide-react'; +import { Progress } from '@/components/ui/progress'; +import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs'; +import { motion } from 'framer-motion'; interface InputScriptStepProps { onNext: () => void; } +interface Character { + id: string; + name: string; + description: string; + personality: string; + appearance: string; + voice: string; + avatar: string; + fullBodyImage: string; + audioSample: string; + styles: string[]; + currentStyle: number; +} + const aiModels = [ { id: 'gpt-4', name: 'GPT-4 Turbo', description: 'Most advanced model with superior creativity' }, { id: 'gpt-3.5', name: 'GPT-3.5 Turbo', description: 'Fast and efficient for most tasks' }, { id: 'claude-3', name: 'Claude 3 Opus', description: 'Excellent for narrative and storytelling' }, ]; +const loadingSteps = [ + { text: "分析脚本内容...", progress: 20 }, + { text: "提取角色信息...", progress: 40 }, + { text: "生成角色形象...", progress: 60 }, + { text: "匹配音色特征...", progress: 80 }, + { text: "完成角色创建...", progress: 100 }, +]; + +const mockCharacters: Character[] = [ + { + id: "1", + name: "凤青楗", + description: "重生的凤凰,拥有强大的意志力,决心改变自己的命运", + personality: "坚强、勇敢、充满希望", + appearance: "优雅的凤凰形象,金色羽毛,炯炯有神的眼睛", + voice: "温暖而坚定的女声", + avatar: "https://images.pexels.com/photos/3861969/pexels-photo-3861969.jpeg?auto=compress&cs=tinysrgb&w=300", + fullBodyImage: "https://images.pexels.com/photos/3861969/pexels-photo-3861969.jpeg?auto=compress&cs=tinysrgb&w=400", + audioSample: "https://www.soundjay.com/misc/sounds/bell-ringing-05.wav", + styles: [ + "https://images.pexels.com/photos/3861969/pexels-photo-3861969.jpeg?auto=compress&cs=tinysrgb&w=300", + "https://images.pexels.com/photos/3184287/pexels-photo-3184287.jpeg?auto=compress&cs=tinysrgb&w=300", + "https://images.pexels.com/photos/1222271/pexels-photo-1222271.jpeg?auto=compress&cs=tinysrgb&w=300" + ], + currentStyle: 0 + }, + { + id: "2", + name: "星光使者", + description: "掌控星辰力量的神秘角色,与凤青楗一同战斗", + personality: "智慧、冷静、神秘", + appearance: "星光环绕的身影,深邃的蓝色长袍", + voice: "低沉磁性的男声", + avatar: "https://images.pexels.com/photos/3184287/pexels-photo-3184287.jpeg?auto=compress&cs=tinysrgb&w=300", + fullBodyImage: "https://images.pexels.com/photos/3184287/pexels-photo-3184287.jpeg?auto=compress&cs=tinysrgb&w=400", + audioSample: "https://www.soundjay.com/misc/sounds/bell-ringing-05.wav", + styles: [ + "https://images.pexels.com/photos/3184287/pexels-photo-3184287.jpeg?auto=compress&cs=tinysrgb&w=300", + "https://images.pexels.com/photos/1222271/pexels-photo-1222271.jpeg?auto=compress&cs=tinysrgb&w=300", + "https://images.pexels.com/photos/3184465/pexels-photo-3184465.jpeg?auto=compress&cs=tinysrgb&w=300" + ], + currentStyle: 0 + } +]; + +// 新的Loading组件 +const CharacterLoading = ({ step }: { step: string }) => { + return ( +
+ {/* 旋转粒子环 */} + + + {/* 中心波动光圈 */} + + + {/* 扫光线条 */} + + + {/* 核心文本 */} + + {step} + +
+ ); +}; + +// 角色卡片组件 +const CharacterCard = ({ + character, + onStyleChange, + onPlayAudio, + isPlaying +}: { + character: Character; + onStyleChange: (id: string, styleIndex: number) => void; + onPlayAudio: (id: string) => void; + isPlaying: string | null; +}) => ( + + + {/* 角色头像区域 */} +
+
+ {character.name} + {/* 渐变遮罩 */} +
+ + {/* 角色名字 */} +
+

{character.name}

+ + {character.voice} + +
+ + {/* 音频播放按钮 */} + +
+ + {/* 详细信息 */} +
+
+ +

{character.description}

+
+ +
+ +

{character.personality}

+
+ +
+ +

{character.appearance}

+
+ + {/* 样式切换 */} +
+
+ + +
+ +
+ {character.styles.map((style, index) => ( + + ))} +
+
+
+
+ + +); + +// 角色生成结果组件 +const CharacterGenerationResult = ({ + characters, + onStyleChange, + onPlayAudio, + isPlaying, + onContinue +}: { + characters: Character[]; + onStyleChange: (id: string, styleIndex: number) => void; + onPlayAudio: (id: string) => void; + isPlaying: string | null; + onContinue: () => void; +}) => ( +
+
+ {/* 标题区域 */} +
+
+ + 角色生成完成 + +
+

您的故事角色

+

+ AI已经根据您的脚本生成了{characters.length}个独特的角色,每个角色都有专属的形象和音色。 + 您可以试听音色、切换形象样式,满意后继续下一步。 +

+
+ + {/* 角色网格 */} +
+ {characters.map((character) => ( +
+ +
+ ))} +
+ + {/* 操作按钮 */} +
+ + +
+
+
+); + export function InputScriptStep({ onNext }: InputScriptStepProps) { const [script, setScript] = useState(''); const [chapters, setChapters] = useState('4'); + const [shots, setShots] = useState('8'); const [showActorsPanel, setShowActorsPanel] = useState(false); + const [isGenerating, setIsGenerating] = useState(true); + const [showCharacters, setShowCharacters] = useState(false); + const [loadingStep, setLoadingStep] = useState(0); + const [characters, setCharacters] = useState(mockCharacters); + const [playingAudio, setPlayingAudio] = useState(null); + + // 模拟生成过程 + useEffect(() => { + if (isGenerating) { + const timer = setInterval(() => { + setLoadingStep((prev) => { + if (prev >= loadingSteps.length - 1) { + clearInterval(timer); + setTimeout(() => { + setIsGenerating(false); + setShowCharacters(true); + }, 500); + return prev; + } + return prev + 1; + }); + }, 1500); + + return () => clearInterval(timer); + } + }, [isGenerating]); const handleSubmit = () => { if (script.trim() && chapters) { - onNext(); + setIsGenerating(true); + setLoadingStep(0); } }; + const handleStyleChange = (characterId: string, styleIndex: number) => { + setCharacters(prev => + prev.map(char => + char.id === characterId + ? { ...char, currentStyle: styleIndex, fullBodyImage: char.styles[styleIndex] } + : char + ) + ); + }; + + const handlePlayAudio = (characterId: string) => { + if (playingAudio === characterId) { + setPlayingAudio(null); + } else { + setPlayingAudio(characterId); + // 模拟音频播放,3秒后自动停止 + setTimeout(() => setPlayingAudio(null), 3000); + } + }; + + const handleContinue = () => { + setShowCharacters(false); + onNext(); + }; + + // 显示角色生成结果 + if (showCharacters) { + return ( + + ); + } + + // 原始的脚本输入界面 return (
- - - - Script Input & Configuration - - - - {/* Script Input */}
- - - - {/* Chapters Configuration */} -
-
- - setChapters(e.target.value)} - placeholder="4" - /> -

- AI will automatically split your script into this many chapters -

-
- -
- - -

- Choose existing actors or create new ones for character reference -

+ {/* Loading Animation - 显示在输入框下方 */} + {isGenerating && ( + + )} + {/* Action Buttons */}
diff --git a/package-lock.json b/package-lock.json index cd47866..2488e7f 100644 --- a/package-lock.json +++ b/package-lock.json @@ -48,8 +48,10 @@ "embla-carousel-react": "^8.3.0", "eslint": "8.49.0", "eslint-config-next": "13.5.1", + "framer-motion": "^12.18.1", "input-otp": "^1.2.4", "lucide-react": "^0.446.0", + "motion": "^12.18.1", "next": "13.5.1", "next-themes": "^0.3.0", "postcss": "8.4.30", @@ -4117,6 +4119,33 @@ "url": "https://github.com/sponsors/rawify" } }, + "node_modules/framer-motion": { + "version": "12.18.1", + "resolved": "https://registry.npmmirror.com/framer-motion/-/framer-motion-12.18.1.tgz", + "integrity": "sha512-6o4EDuRPLk4LSZ1kRnnEOurbQ86MklVk+Y1rFBUKiF+d2pCdvMjWVu0ZkyMVCTwl5UyTH2n/zJEJx+jvTYuxow==", + "license": "MIT", + "dependencies": { + "motion-dom": "^12.18.1", + "motion-utils": "^12.18.1", + "tslib": "^2.4.0" + }, + "peerDependencies": { + "@emotion/is-prop-valid": "*", + "react": "^18.0.0 || ^19.0.0", + "react-dom": "^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/is-prop-valid": { + "optional": true + }, + "react": { + "optional": true + }, + "react-dom": { + "optional": true + } + } + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -5096,6 +5125,47 @@ "node": ">=16 || 14 >=14.17" } }, + "node_modules/motion": { + "version": "12.18.1", + "resolved": "https://registry.npmmirror.com/motion/-/motion-12.18.1.tgz", + "integrity": "sha512-w1ns2hWQ4COhOvnZf4rg4mW0Pl36mzcShpgt0fSfI6qJxKUbi3kHho/HSKeJFRoY0TO1m5/7C8lG1+Li0uC9Fw==", + "license": "MIT", + "dependencies": { + "framer-motion": "^12.18.1", + "tslib": "^2.4.0" + }, + "peerDependencies": { + "@emotion/is-prop-valid": "*", + "react": "^18.0.0 || ^19.0.0", + "react-dom": "^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "@emotion/is-prop-valid": { + "optional": true + }, + "react": { + "optional": true + }, + "react-dom": { + "optional": true + } + } + }, + "node_modules/motion-dom": { + "version": "12.18.1", + "resolved": "https://registry.npmmirror.com/motion-dom/-/motion-dom-12.18.1.tgz", + "integrity": "sha512-dR/4EYT23Snd+eUSLrde63Ws3oXQtJNw/krgautvTfwrN/2cHfCZMdu6CeTxVfRRWREW3Fy1f5vobRDiBb/q+w==", + "license": "MIT", + "dependencies": { + "motion-utils": "^12.18.1" + } + }, + "node_modules/motion-utils": { + "version": "12.18.1", + "resolved": "https://registry.npmmirror.com/motion-utils/-/motion-utils-12.18.1.tgz", + "integrity": "sha512-az26YDU4WoDP0ueAkUtABLk2BIxe28d8NH1qWT8jPGhPyf44XTdDUh8pDk9OPphaSrR9McgpcJlgwSOIw/sfkA==", + "license": "MIT" + }, "node_modules/ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", diff --git a/package.json b/package.json index a0806d1..315d421 100644 --- a/package.json +++ b/package.json @@ -49,8 +49,10 @@ "embla-carousel-react": "^8.3.0", "eslint": "8.49.0", "eslint-config-next": "13.5.1", + "framer-motion": "^12.18.1", "input-otp": "^1.2.4", "lucide-react": "^0.446.0", + "motion": "^12.18.1", "next": "13.5.1", "next-themes": "^0.3.0", "postcss": "8.4.30", diff --git a/plugins/liquid-glass/index.d.ts b/plugins/liquid-glass/index.d.ts new file mode 100644 index 0000000..21eb614 --- /dev/null +++ b/plugins/liquid-glass/index.d.ts @@ -0,0 +1,27 @@ +interface LiquidGlassProps { + children: React.ReactNode; + displacementScale?: number; + blurAmount?: number; + saturation?: number; + aberrationIntensity?: number; + elasticity?: number; + cornerRadius?: number; + globalMousePos?: { + x: number; + y: number; + }; + mouseOffset?: { + x: number; + y: number; + }; + mouseContainer?: React.RefObject | null; + className?: string; + padding?: string; + style?: React.CSSProperties; + overLight?: boolean; + mode?: "standard" | "polar" | "prominent" | "shader"; + onClick?: () => void; +} +export default function LiquidGlass({ children, displacementScale, blurAmount, saturation, aberrationIntensity, elasticity, cornerRadius, globalMousePos: externalGlobalMousePos, mouseOffset: externalMouseOffset, mouseContainer, className, padding, overLight, style, mode, onClick, }: LiquidGlassProps): import("react/jsx-runtime").JSX.Element; +export {}; +//# sourceMappingURL=index.d.ts.map \ No newline at end of file diff --git a/plugins/liquid-glass/index.d.ts.map b/plugins/liquid-glass/index.d.ts.map new file mode 100644 index 0000000..5fad08f --- /dev/null +++ b/plugins/liquid-glass/index.d.ts.map @@ -0,0 +1 @@ +{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.tsx"],"names":[],"mappings":"AAwPA,UAAU,gBAAgB;IACxB,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAA;IACzB,iBAAiB,CAAC,EAAE,MAAM,CAAA;IAC1B,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,mBAAmB,CAAC,EAAE,MAAM,CAAA;IAC5B,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,cAAc,CAAC,EAAE;QAAE,CAAC,EAAE,MAAM,CAAC;QAAC,CAAC,EAAE,MAAM,CAAA;KAAE,CAAA;IACzC,WAAW,CAAC,EAAE;QAAE,CAAC,EAAE,MAAM,CAAC;QAAC,CAAC,EAAE,MAAM,CAAA;KAAE,CAAA;IACtC,cAAc,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,IAAI,CAAA;IAC3D,SAAS,CAAC,EAAE,MAAM,CAAA;IAClB,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,KAAK,CAAC,EAAE,KAAK,CAAC,aAAa,CAAA;IAC3B,SAAS,CAAC,EAAE,OAAO,CAAA;IACnB,IAAI,CAAC,EAAE,UAAU,GAAG,OAAO,GAAG,WAAW,GAAG,QAAQ,CAAA;IACpD,OAAO,CAAC,EAAE,MAAM,IAAI,CAAA;CACrB;AAED,MAAM,CAAC,OAAO,UAAU,WAAW,CAAC,EAClC,QAAQ,EACR,iBAAsB,EACtB,UAAmB,EACnB,UAAgB,EAChB,mBAAuB,EACvB,UAAiB,EACjB,YAAkB,EAClB,cAAc,EAAE,sBAAsB,EACtC,WAAW,EAAE,mBAAmB,EAChC,cAAqB,EACrB,SAAc,EACd,OAAqB,EACrB,SAAiB,EACjB,KAAU,EACV,IAAiB,EACjB,OAAO,GACR,EAAE,gBAAgB,2CAuUlB"} \ No newline at end of file diff --git a/plugins/liquid-glass/index.esm.js b/plugins/liquid-glass/index.esm.js new file mode 100644 index 0000000..98375df --- /dev/null +++ b/plugins/liquid-glass/index.esm.js @@ -0,0 +1,586 @@ +// src/index.tsx +import { forwardRef, useCallback, useEffect, useId, useRef, useState } from "react"; + +// src/shader-utils.ts +function smoothStep(a, b, t) { + t = Math.max(0, Math.min(1, (t - a) / (b - a))); + return t * t * (3 - 2 * t); +} +function length(x, y) { + return Math.sqrt(x * x + y * y); +} +function roundedRectSDF(x, y, width, height, radius) { + const qx = Math.abs(x) - width + radius; + const qy = Math.abs(y) - height + radius; + return Math.min(Math.max(qx, qy), 0) + length(Math.max(qx, 0), Math.max(qy, 0)) - radius; +} +function texture(x, y) { + return { x, y }; +} +var fragmentShaders = { + liquidGlass: (uv) => { + const ix = uv.x - 0.5; + const iy = uv.y - 0.5; + const distanceToEdge = roundedRectSDF(ix, iy, 0.3, 0.2, 0.6); + const displacement = smoothStep(0.8, 0, distanceToEdge - 0.15); + const scaled = smoothStep(0, 1, displacement); + return texture(ix * scaled + 0.5, iy * scaled + 0.5); + } +}; +var ShaderDisplacementGenerator = class { + constructor(options) { + this.options = options; + this.canvasDPI = 1; + this.canvas = document.createElement("canvas"); + this.canvas.width = options.width * this.canvasDPI; + this.canvas.height = options.height * this.canvasDPI; + this.canvas.style.display = "none"; + const context = this.canvas.getContext("2d"); + if (!context) { + throw new Error("Could not get 2D context"); + } + this.context = context; + } + updateShader(mousePosition) { + const w = this.options.width * this.canvasDPI; + const h = this.options.height * this.canvasDPI; + let maxScale = 0; + const rawValues = []; + for (let y = 0; y < h; y++) { + for (let x = 0; x < w; x++) { + const uv = { x: x / w, y: y / h }; + const pos = this.options.fragment(uv, mousePosition); + const dx = pos.x * w - x; + const dy = pos.y * h - y; + maxScale = Math.max(maxScale, Math.abs(dx), Math.abs(dy)); + rawValues.push(dx, dy); + } + } + if (maxScale > 0) { + maxScale = Math.max(maxScale, 1); + } else { + maxScale = 1; + } + const imageData = this.context.createImageData(w, h); + const data = imageData.data; + let rawIndex = 0; + for (let y = 0; y < h; y++) { + for (let x = 0; x < w; x++) { + const dx = rawValues[rawIndex++]; + const dy = rawValues[rawIndex++]; + const edgeDistance = Math.min(x, y, w - x - 1, h - y - 1); + const edgeFactor = Math.min(1, edgeDistance / 2); + const smoothedDx = dx * edgeFactor; + const smoothedDy = dy * edgeFactor; + const r = smoothedDx / maxScale + 0.5; + const g = smoothedDy / maxScale + 0.5; + const pixelIndex = (y * w + x) * 4; + data[pixelIndex] = Math.max(0, Math.min(255, r * 255)); + data[pixelIndex + 1] = Math.max(0, Math.min(255, g * 255)); + data[pixelIndex + 2] = Math.max(0, Math.min(255, g * 255)); + data[pixelIndex + 3] = 255; + } + } + this.context.putImageData(imageData, 0, 0); + return this.canvas.toDataURL(); + } + destroy() { + this.canvas.remove(); + } + getScale() { + return this.canvasDPI; + } +}; + +// src/utils.ts +var displacementMap = ""; +var polarDisplacementMap = ""; +var prominentDisplacementMap = ""; + +// src/index.tsx +import { Fragment, jsx, jsxs } from "react/jsx-runtime"; +var generateShaderDisplacementMap = (width, height) => { + const generator = new ShaderDisplacementGenerator({ + width, + height, + fragment: fragmentShaders.liquidGlass + }); + const dataUrl = generator.updateShader(); + generator.destroy(); + return dataUrl; +}; +var getMap = (mode, shaderMapUrl) => { + switch (mode) { + case "standard": + return displacementMap; + case "polar": + return polarDisplacementMap; + case "prominent": + return prominentDisplacementMap; + case "shader": + return shaderMapUrl || displacementMap; + default: + throw new Error(`Invalid mode: ${mode}`); + } +}; +var GlassFilter = ({ + id, + displacementScale, + aberrationIntensity, + width, + height, + mode, + shaderMapUrl +}) => /* @__PURE__ */ jsx("svg", { style: { position: "absolute", width, height }, "aria-hidden": "true", children: /* @__PURE__ */ jsxs("defs", { children: [ + /* @__PURE__ */ jsxs("radialGradient", { id: `${id}-edge-mask`, cx: "50%", cy: "50%", r: "50%", children: [ + /* @__PURE__ */ jsx("stop", { offset: "0%", stopColor: "black", stopOpacity: "0" }), + /* @__PURE__ */ jsx("stop", { offset: `${Math.max(30, 80 - aberrationIntensity * 2)}%`, stopColor: "black", stopOpacity: "0" }), + /* @__PURE__ */ jsx("stop", { offset: "100%", stopColor: "white", stopOpacity: "1" }) + ] }), + /* @__PURE__ */ jsxs("filter", { id, x: "-35%", y: "-35%", width: "170%", height: "170%", colorInterpolationFilters: "sRGB", children: [ + /* @__PURE__ */ jsx("feImage", { id: "feimage", x: "0", y: "0", width: "100%", height: "100%", result: "DISPLACEMENT_MAP", href: getMap(mode, shaderMapUrl), preserveAspectRatio: "xMidYMid slice" }), + /* @__PURE__ */ jsx( + "feColorMatrix", + { + in: "DISPLACEMENT_MAP", + type: "matrix", + values: "0.3 0.3 0.3 0 0\n 0.3 0.3 0.3 0 0\n 0.3 0.3 0.3 0 0\n 0 0 0 1 0", + result: "EDGE_INTENSITY" + } + ), + /* @__PURE__ */ jsx("feComponentTransfer", { in: "EDGE_INTENSITY", result: "EDGE_MASK", children: /* @__PURE__ */ jsx("feFuncA", { type: "discrete", tableValues: `0 ${aberrationIntensity * 0.05} 1` }) }), + /* @__PURE__ */ jsx("feOffset", { in: "SourceGraphic", dx: "0", dy: "0", result: "CENTER_ORIGINAL" }), + /* @__PURE__ */ jsx("feDisplacementMap", { in: "SourceGraphic", in2: "DISPLACEMENT_MAP", scale: displacementScale * (mode === "shader" ? 1 : -1), xChannelSelector: "R", yChannelSelector: "B", result: "RED_DISPLACED" }), + /* @__PURE__ */ jsx( + "feColorMatrix", + { + in: "RED_DISPLACED", + type: "matrix", + values: "1 0 0 0 0\n 0 0 0 0 0\n 0 0 0 0 0\n 0 0 0 1 0", + result: "RED_CHANNEL" + } + ), + /* @__PURE__ */ jsx("feDisplacementMap", { in: "SourceGraphic", in2: "DISPLACEMENT_MAP", scale: displacementScale * ((mode === "shader" ? 1 : -1) - aberrationIntensity * 0.05), xChannelSelector: "R", yChannelSelector: "B", result: "GREEN_DISPLACED" }), + /* @__PURE__ */ jsx( + "feColorMatrix", + { + in: "GREEN_DISPLACED", + type: "matrix", + values: "0 0 0 0 0\n 0 1 0 0 0\n 0 0 0 0 0\n 0 0 0 1 0", + result: "GREEN_CHANNEL" + } + ), + /* @__PURE__ */ jsx("feDisplacementMap", { in: "SourceGraphic", in2: "DISPLACEMENT_MAP", scale: displacementScale * ((mode === "shader" ? 1 : -1) - aberrationIntensity * 0.1), xChannelSelector: "R", yChannelSelector: "B", result: "BLUE_DISPLACED" }), + /* @__PURE__ */ jsx( + "feColorMatrix", + { + in: "BLUE_DISPLACED", + type: "matrix", + values: "0 0 0 0 0\n 0 0 0 0 0\n 0 0 1 0 0\n 0 0 0 1 0", + result: "BLUE_CHANNEL" + } + ), + /* @__PURE__ */ jsx("feBlend", { in: "GREEN_CHANNEL", in2: "BLUE_CHANNEL", mode: "screen", result: "GB_COMBINED" }), + /* @__PURE__ */ jsx("feBlend", { in: "RED_CHANNEL", in2: "GB_COMBINED", mode: "screen", result: "RGB_COMBINED" }), + /* @__PURE__ */ jsx("feGaussianBlur", { in: "RGB_COMBINED", stdDeviation: Math.max(0.1, 0.5 - aberrationIntensity * 0.1), result: "ABERRATED_BLURRED" }), + /* @__PURE__ */ jsx("feComposite", { in: "ABERRATED_BLURRED", in2: "EDGE_MASK", operator: "in", result: "EDGE_ABERRATION" }), + /* @__PURE__ */ jsx("feComponentTransfer", { in: "EDGE_MASK", result: "INVERTED_MASK", children: /* @__PURE__ */ jsx("feFuncA", { type: "table", tableValues: "1 0" }) }), + /* @__PURE__ */ jsx("feComposite", { in: "CENTER_ORIGINAL", in2: "INVERTED_MASK", operator: "in", result: "CENTER_CLEAN" }), + /* @__PURE__ */ jsx("feComposite", { in: "EDGE_ABERRATION", in2: "CENTER_CLEAN", operator: "over" }) + ] }) +] }) }); +var GlassContainer = forwardRef( + ({ + children, + className = "", + style, + displacementScale = 25, + blurAmount = 12, + saturation = 180, + aberrationIntensity = 2, + onMouseEnter, + onMouseLeave, + onMouseDown, + onMouseUp, + active = false, + overLight = false, + cornerRadius = 999, + padding = "24px 32px", + glassSize = { width: 270, height: 69 }, + onClick, + mode = "standard" + }, ref) => { + const filterId = useId(); + const [shaderMapUrl, setShaderMapUrl] = useState(""); + const isFirefox = navigator.userAgent.toLowerCase().includes("firefox"); + useEffect(() => { + if (mode === "shader") { + const url = generateShaderDisplacementMap(glassSize.width, glassSize.height); + setShaderMapUrl(url); + } + }, [mode, glassSize.width, glassSize.height]); + const backdropStyle = { + filter: isFirefox ? null : `url(#${filterId})`, + backdropFilter: `blur(${(overLight ? 12 : 4) + blurAmount * 32}px) saturate(${saturation}%)` + }; + return /* @__PURE__ */ jsxs("div", { ref, className: `relative ${className} ${active ? "active" : ""} ${Boolean(onClick) ? "cursor-pointer" : ""}`, style, onClick, children: [ + /* @__PURE__ */ jsx(GlassFilter, { mode, id: filterId, displacementScale, aberrationIntensity, width: glassSize.width, height: glassSize.height, shaderMapUrl }), + /* @__PURE__ */ jsxs( + "div", + { + className: "glass", + style: { + borderRadius: `${cornerRadius}px`, + position: "relative", + display: "inline-flex", + alignItems: "center", + gap: "24px", + padding, + overflow: "hidden", + transition: "all 0.2s ease-in-out", + boxShadow: overLight ? "0px 16px 70px rgba(0, 0, 0, 0.75)" : "0px 12px 40px rgba(0, 0, 0, 0.25)" + }, + onMouseEnter, + onMouseLeave, + onMouseDown, + onMouseUp, + children: [ + /* @__PURE__ */ jsx( + "span", + { + className: "glass__warp", + style: { + ...backdropStyle, + position: "absolute", + inset: "0" + } + } + ), + /* @__PURE__ */ jsx( + "div", + { + className: "transition-all duration-150 ease-in-out text-white", + style: { + position: "relative", + zIndex: 1, + font: "500 20px/1 system-ui", + textShadow: overLight ? "0px 2px 12px rgba(0, 0, 0, 0)" : "0px 2px 12px rgba(0, 0, 0, 0.4)" + }, + children + } + ) + ] + } + ) + ] }); + } +); +GlassContainer.displayName = "GlassContainer"; +function LiquidGlass({ + children, + displacementScale = 70, + blurAmount = 0.0625, + saturation = 140, + aberrationIntensity = 2, + elasticity = 0.15, + cornerRadius = 999, + globalMousePos: externalGlobalMousePos, + mouseOffset: externalMouseOffset, + mouseContainer = null, + className = "", + padding = "24px 32px", + overLight = false, + style = {}, + mode = "standard", + onClick +}) { + const glassRef = useRef(null); + const [isHovered, setIsHovered] = useState(false); + const [isActive, setIsActive] = useState(false); + const [glassSize, setGlassSize] = useState({ width: 270, height: 69 }); + const [internalGlobalMousePos, setInternalGlobalMousePos] = useState({ x: 0, y: 0 }); + const [internalMouseOffset, setInternalMouseOffset] = useState({ x: 0, y: 0 }); + const globalMousePos = externalGlobalMousePos || internalGlobalMousePos; + const mouseOffset = externalMouseOffset || internalMouseOffset; + const handleMouseMove = useCallback( + (e) => { + const container = mouseContainer?.current || glassRef.current; + if (!container) { + return; + } + const rect = container.getBoundingClientRect(); + const centerX = rect.left + rect.width / 2; + const centerY = rect.top + rect.height / 2; + setInternalMouseOffset({ + x: (e.clientX - centerX) / rect.width * 100, + y: (e.clientY - centerY) / rect.height * 100 + }); + setInternalGlobalMousePos({ + x: e.clientX, + y: e.clientY + }); + }, + [mouseContainer] + ); + useEffect(() => { + if (externalGlobalMousePos && externalMouseOffset) { + return; + } + const container = mouseContainer?.current || glassRef.current; + if (!container) { + return; + } + container.addEventListener("mousemove", handleMouseMove); + return () => { + container.removeEventListener("mousemove", handleMouseMove); + }; + }, [handleMouseMove, mouseContainer, externalGlobalMousePos, externalMouseOffset]); + const calculateDirectionalScale = useCallback(() => { + if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) { + return "scale(1)"; + } + const rect = glassRef.current.getBoundingClientRect(); + const pillCenterX = rect.left + rect.width / 2; + const pillCenterY = rect.top + rect.height / 2; + const pillWidth = glassSize.width; + const pillHeight = glassSize.height; + const deltaX = globalMousePos.x - pillCenterX; + const deltaY = globalMousePos.y - pillCenterY; + const edgeDistanceX = Math.max(0, Math.abs(deltaX) - pillWidth / 2); + const edgeDistanceY = Math.max(0, Math.abs(deltaY) - pillHeight / 2); + const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY); + const activationZone = 200; + if (edgeDistance > activationZone) { + return "scale(1)"; + } + const fadeInFactor = 1 - edgeDistance / activationZone; + const centerDistance = Math.sqrt(deltaX * deltaX + deltaY * deltaY); + if (centerDistance === 0) { + return "scale(1)"; + } + const normalizedX = deltaX / centerDistance; + const normalizedY = deltaY / centerDistance; + const stretchIntensity = Math.min(centerDistance / 300, 1) * elasticity * fadeInFactor; + const scaleX = 1 + Math.abs(normalizedX) * stretchIntensity * 0.3 - Math.abs(normalizedY) * stretchIntensity * 0.15; + const scaleY = 1 + Math.abs(normalizedY) * stretchIntensity * 0.3 - Math.abs(normalizedX) * stretchIntensity * 0.15; + return `scaleX(${Math.max(0.8, scaleX)}) scaleY(${Math.max(0.8, scaleY)})`; + }, [globalMousePos, elasticity, glassSize]); + const calculateFadeInFactor = useCallback(() => { + if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) { + return 0; + } + const rect = glassRef.current.getBoundingClientRect(); + const pillCenterX = rect.left + rect.width / 2; + const pillCenterY = rect.top + rect.height / 2; + const pillWidth = glassSize.width; + const pillHeight = glassSize.height; + const edgeDistanceX = Math.max(0, Math.abs(globalMousePos.x - pillCenterX) - pillWidth / 2); + const edgeDistanceY = Math.max(0, Math.abs(globalMousePos.y - pillCenterY) - pillHeight / 2); + const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY); + const activationZone = 200; + return edgeDistance > activationZone ? 0 : 1 - edgeDistance / activationZone; + }, [globalMousePos, glassSize]); + const calculateElasticTranslation = useCallback(() => { + if (!glassRef.current) { + return { x: 0, y: 0 }; + } + const fadeInFactor = calculateFadeInFactor(); + const rect = glassRef.current.getBoundingClientRect(); + const pillCenterX = rect.left + rect.width / 2; + const pillCenterY = rect.top + rect.height / 2; + return { + x: (globalMousePos.x - pillCenterX) * elasticity * 0.1 * fadeInFactor, + y: (globalMousePos.y - pillCenterY) * elasticity * 0.1 * fadeInFactor + }; + }, [globalMousePos, elasticity, calculateFadeInFactor]); + useEffect(() => { + const updateGlassSize = () => { + if (glassRef.current) { + const rect = glassRef.current.getBoundingClientRect(); + setGlassSize({ width: rect.width, height: rect.height }); + } + }; + updateGlassSize(); + window.addEventListener("resize", updateGlassSize); + return () => window.removeEventListener("resize", updateGlassSize); + }, []); + const transformStyle = `translate(calc(-50% + ${calculateElasticTranslation().x}px), calc(-50% + ${calculateElasticTranslation().y}px)) ${isActive && Boolean(onClick) ? "scale(0.96)" : calculateDirectionalScale()}`; + const baseStyle = { + ...style, + transform: transformStyle, + transition: "all ease-out 0.2s" + }; + const positionStyles = { + position: baseStyle.position || "relative", + top: baseStyle.top || "50%", + left: baseStyle.left || "50%" + }; + return /* @__PURE__ */ jsxs(Fragment, { children: [ + /* @__PURE__ */ jsx( + "div", + { + className: `bg-black transition-all duration-150 ease-in-out pointer-events-none ${overLight ? "opacity-20" : "opacity-0"}`, + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition + } + } + ), + /* @__PURE__ */ jsx( + "div", + { + className: `bg-black transition-all duration-150 ease-in-out pointer-events-none mix-blend-overlay ${overLight ? "opacity-100" : "opacity-0"}`, + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition + } + } + ), + /* @__PURE__ */ jsx( + GlassContainer, + { + ref: glassRef, + className, + style: baseStyle, + cornerRadius, + displacementScale: overLight ? displacementScale * 0.5 : displacementScale, + blurAmount, + saturation, + aberrationIntensity, + glassSize, + padding, + mouseOffset, + onMouseEnter: () => setIsHovered(true), + onMouseLeave: () => setIsHovered(false), + onMouseDown: () => setIsActive(true), + onMouseUp: () => setIsActive(false), + active: isActive, + overLight, + onClick, + mode, + children + } + ), + /* @__PURE__ */ jsx( + "span", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition, + pointerEvents: "none", + mixBlendMode: "screen", + opacity: 0.2, + padding: "1.5px", + WebkitMask: "linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0)", + WebkitMaskComposite: "xor", + maskComposite: "exclude", + boxShadow: "0 0 0 0.5px rgba(255, 255, 255, 0.5) inset, 0 1px 3px rgba(255, 255, 255, 0.25) inset, 0 1px 4px rgba(0, 0, 0, 0.35)", + background: `linear-gradient( + ${135 + mouseOffset.x * 1.2}deg, + rgba(255, 255, 255, 0.0) 0%, + rgba(255, 255, 255, ${0.12 + Math.abs(mouseOffset.x) * 8e-3}) ${Math.max(10, 33 + mouseOffset.y * 0.3)}%, + rgba(255, 255, 255, ${0.4 + Math.abs(mouseOffset.x) * 0.012}) ${Math.min(90, 66 + mouseOffset.y * 0.4)}%, + rgba(255, 255, 255, 0.0) 100% + )` + } + } + ), + /* @__PURE__ */ jsx( + "span", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition, + pointerEvents: "none", + mixBlendMode: "overlay", + padding: "1.5px", + WebkitMask: "linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0)", + WebkitMaskComposite: "xor", + maskComposite: "exclude", + boxShadow: "0 0 0 0.5px rgba(255, 255, 255, 0.5) inset, 0 1px 3px rgba(255, 255, 255, 0.25) inset, 0 1px 4px rgba(0, 0, 0, 0.35)", + background: `linear-gradient( + ${135 + mouseOffset.x * 1.2}deg, + rgba(255, 255, 255, 0.0) 0%, + rgba(255, 255, 255, ${0.32 + Math.abs(mouseOffset.x) * 8e-3}) ${Math.max(10, 33 + mouseOffset.y * 0.3)}%, + rgba(255, 255, 255, ${0.6 + Math.abs(mouseOffset.x) * 0.012}) ${Math.min(90, 66 + mouseOffset.y * 0.4)}%, + rgba(255, 255, 255, 0.0) 100% + )` + } + } + ), + Boolean(onClick) && /* @__PURE__ */ jsxs(Fragment, { children: [ + /* @__PURE__ */ jsx( + "div", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width + 1, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + pointerEvents: "none", + transition: "all 0.2s ease-out", + opacity: isHovered || isActive ? 0.5 : 0, + backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0) 50%)", + mixBlendMode: "overlay" + } + } + ), + /* @__PURE__ */ jsx( + "div", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width + 1, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + pointerEvents: "none", + transition: "all 0.2s ease-out", + opacity: isActive ? 0.5 : 0, + backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 80%)", + mixBlendMode: "overlay" + } + } + ), + /* @__PURE__ */ jsx( + "div", + { + style: { + ...baseStyle, + height: glassSize.height, + width: glassSize.width + 1, + borderRadius: `${cornerRadius}px`, + position: baseStyle.position, + top: baseStyle.top, + left: baseStyle.left, + pointerEvents: "none", + transition: "all 0.2s ease-out", + opacity: isHovered ? 0.4 : isActive ? 0.8 : 0, + backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%)", + mixBlendMode: "overlay" + } + } + ) + ] }) + ] }); +} +export { + LiquidGlass as default +}; diff --git a/plugins/liquid-glass/index.js b/plugins/liquid-glass/index.js new file mode 100644 index 0000000..5ff1dce --- /dev/null +++ b/plugins/liquid-glass/index.js @@ -0,0 +1,607 @@ +"use strict"; +var __defProp = Object.defineProperty; +var __getOwnPropDesc = Object.getOwnPropertyDescriptor; +var __getOwnPropNames = Object.getOwnPropertyNames; +var __hasOwnProp = Object.prototype.hasOwnProperty; +var __export = (target, all) => { + for (var name in all) + __defProp(target, name, { get: all[name], enumerable: true }); +}; +var __copyProps = (to, from, except, desc) => { + if (from && typeof from === "object" || typeof from === "function") { + for (let key of __getOwnPropNames(from)) + if (!__hasOwnProp.call(to, key) && key !== except) + __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable }); + } + return to; +}; +var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod); + +// src/index.tsx +var src_exports = {}; +__export(src_exports, { + default: () => LiquidGlass +}); +module.exports = __toCommonJS(src_exports); +var import_react = require("react"); + +// src/shader-utils.ts +function smoothStep(a, b, t) { + t = Math.max(0, Math.min(1, (t - a) / (b - a))); + return t * t * (3 - 2 * t); +} +function length(x, y) { + return Math.sqrt(x * x + y * y); +} +function roundedRectSDF(x, y, width, height, radius) { + const qx = Math.abs(x) - width + radius; + const qy = Math.abs(y) - height + radius; + return Math.min(Math.max(qx, qy), 0) + length(Math.max(qx, 0), Math.max(qy, 0)) - radius; +} +function texture(x, y) { + return { x, y }; +} +var fragmentShaders = { + liquidGlass: (uv) => { + const ix = uv.x - 0.5; + const iy = uv.y - 0.5; + const distanceToEdge = roundedRectSDF(ix, iy, 0.3, 0.2, 0.6); + const displacement = smoothStep(0.8, 0, distanceToEdge - 0.15); + const scaled = smoothStep(0, 1, displacement); + return texture(ix * scaled + 0.5, iy * scaled + 0.5); + } +}; +var ShaderDisplacementGenerator = class { + constructor(options) { + this.options = options; + this.canvasDPI = 1; + this.canvas = document.createElement("canvas"); + this.canvas.width = options.width * this.canvasDPI; + this.canvas.height = options.height * this.canvasDPI; + this.canvas.style.display = "none"; + const context = this.canvas.getContext("2d"); + if (!context) { + throw new Error("Could not get 2D context"); + } + this.context = context; + } + updateShader(mousePosition) { + const w = this.options.width * this.canvasDPI; + const h = this.options.height * this.canvasDPI; + let maxScale = 0; + const rawValues = []; + for (let y = 0; y < h; y++) { + for (let x = 0; x < w; x++) { + const uv = { x: x / w, y: y / h }; + const pos = this.options.fragment(uv, mousePosition); + const dx = pos.x * w - x; + const dy = pos.y * h - y; + maxScale = Math.max(maxScale, Math.abs(dx), Math.abs(dy)); + rawValues.push(dx, dy); + } + } + if (maxScale > 0) { + maxScale = Math.max(maxScale, 1); + } else { + maxScale = 1; + } + const imageData = this.context.createImageData(w, h); + const data = imageData.data; + let rawIndex = 0; + for (let y = 0; y < h; y++) { + for (let x = 0; x < w; x++) { + const dx = rawValues[rawIndex++]; + const dy = rawValues[rawIndex++]; + const edgeDistance = Math.min(x, y, w - x - 1, h - y - 1); + const edgeFactor = Math.min(1, edgeDistance / 2); + const smoothedDx = dx * edgeFactor; + const smoothedDy = dy * edgeFactor; + const r = smoothedDx / maxScale + 0.5; + const g = smoothedDy / maxScale + 0.5; + const pixelIndex = (y * w + x) * 4; + data[pixelIndex] = Math.max(0, Math.min(255, r * 255)); + data[pixelIndex + 1] = Math.max(0, Math.min(255, g * 255)); + data[pixelIndex + 2] = Math.max(0, Math.min(255, g * 255)); + data[pixelIndex + 3] = 255; + } + } + this.context.putImageData(imageData, 0, 0); + return this.canvas.toDataURL(); + } + destroy() { + this.canvas.remove(); + } + getScale() { + return this.canvasDPI; + } +}; + +// src/utils.ts +var displacementMap = ""; +var polarDisplacementMap = ""; +var prominentDisplacementMap = ""; + +// src/index.tsx +var import_jsx_runtime = require("react/jsx-runtime"); +var generateShaderDisplacementMap = (width, height) => { + const generator = new ShaderDisplacementGenerator({ + width, + height, + fragment: fragmentShaders.liquidGlass + }); + const dataUrl = generator.updateShader(); + generator.destroy(); + return dataUrl; +}; +var getMap = (mode, shaderMapUrl) => { + switch (mode) { + case "standard": + return displacementMap; + case "polar": + return polarDisplacementMap; + case "prominent": + return prominentDisplacementMap; + case "shader": + return shaderMapUrl || displacementMap; + default: + throw new Error(`Invalid mode: ${mode}`); + } +}; +var GlassFilter = ({ + id, + displacementScale, + aberrationIntensity, + width, + height, + mode, + shaderMapUrl +}) => /* @__PURE__ */ (0, import_jsx_runtime.jsx)("svg", { style: { position: "absolute", width, height }, "aria-hidden": "true", children: /* @__PURE__ */ (0, import_jsx_runtime.jsxs)("defs", { children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsxs)("radialGradient", { id: `${id}-edge-mask`, cx: "50%", cy: "50%", r: "50%", children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("stop", { offset: "0%", stopColor: "black", stopOpacity: "0" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("stop", { offset: `${Math.max(30, 80 - aberrationIntensity * 2)}%`, stopColor: "black", stopOpacity: "0" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("stop", { offset: "100%", stopColor: "white", stopOpacity: "1" }) + ] }), + /* @__PURE__ */ (0, import_jsx_runtime.jsxs)("filter", { id, x: "-35%", y: "-35%", width: "170%", height: "170%", colorInterpolationFilters: "sRGB", children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feImage", { id: "feimage", x: "0", y: "0", width: "100%", height: "100%", result: "DISPLACEMENT_MAP", href: getMap(mode, shaderMapUrl), preserveAspectRatio: "xMidYMid slice" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "feColorMatrix", + { + in: "DISPLACEMENT_MAP", + type: "matrix", + values: "0.3 0.3 0.3 0 0\n 0.3 0.3 0.3 0 0\n 0.3 0.3 0.3 0 0\n 0 0 0 1 0", + result: "EDGE_INTENSITY" + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feComponentTransfer", { in: "EDGE_INTENSITY", result: "EDGE_MASK", children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feFuncA", { type: "discrete", tableValues: `0 ${aberrationIntensity * 0.05} 1` }) }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feOffset", { in: "SourceGraphic", dx: "0", dy: "0", result: "CENTER_ORIGINAL" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feDisplacementMap", { in: "SourceGraphic", in2: "DISPLACEMENT_MAP", scale: displacementScale * (mode === "shader" ? 1 : -1), xChannelSelector: "R", yChannelSelector: "B", result: "RED_DISPLACED" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "feColorMatrix", + { + in: "RED_DISPLACED", + type: "matrix", + values: "1 0 0 0 0\n 0 0 0 0 0\n 0 0 0 0 0\n 0 0 0 1 0", + result: "RED_CHANNEL" + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feDisplacementMap", { in: "SourceGraphic", in2: "DISPLACEMENT_MAP", scale: displacementScale * ((mode === "shader" ? 1 : -1) - aberrationIntensity * 0.05), xChannelSelector: "R", yChannelSelector: "B", result: "GREEN_DISPLACED" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "feColorMatrix", + { + in: "GREEN_DISPLACED", + type: "matrix", + values: "0 0 0 0 0\n 0 1 0 0 0\n 0 0 0 0 0\n 0 0 0 1 0", + result: "GREEN_CHANNEL" + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feDisplacementMap", { in: "SourceGraphic", in2: "DISPLACEMENT_MAP", scale: displacementScale * ((mode === "shader" ? 1 : -1) - aberrationIntensity * 0.1), xChannelSelector: "R", yChannelSelector: "B", result: "BLUE_DISPLACED" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "feColorMatrix", + { + in: "BLUE_DISPLACED", + type: "matrix", + values: "0 0 0 0 0\n 0 0 0 0 0\n 0 0 1 0 0\n 0 0 0 1 0", + result: "BLUE_CHANNEL" + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feBlend", { in: "GREEN_CHANNEL", in2: "BLUE_CHANNEL", mode: "screen", result: "GB_COMBINED" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feBlend", { in: "RED_CHANNEL", in2: "GB_COMBINED", mode: "screen", result: "RGB_COMBINED" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feGaussianBlur", { in: "RGB_COMBINED", stdDeviation: Math.max(0.1, 0.5 - aberrationIntensity * 0.1), result: "ABERRATED_BLURRED" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feComposite", { in: "ABERRATED_BLURRED", in2: "EDGE_MASK", operator: "in", result: "EDGE_ABERRATION" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feComponentTransfer", { in: "EDGE_MASK", result: "INVERTED_MASK", children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feFuncA", { type: "table", tableValues: "1 0" }) }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feComposite", { in: "CENTER_ORIGINAL", in2: "INVERTED_MASK", operator: "in", result: "CENTER_CLEAN" }), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)("feComposite", { in: "EDGE_ABERRATION", in2: "CENTER_CLEAN", operator: "over" }) + ] }) +] }) }); +var GlassContainer = (0, import_react.forwardRef)( + ({ + children, + className = "", + style, + displacementScale = 25, + blurAmount = 12, + saturation = 180, + aberrationIntensity = 2, + onMouseEnter, + onMouseLeave, + onMouseDown, + onMouseUp, + active = false, + overLight = false, + cornerRadius = 999, + padding = "24px 32px", + glassSize = { width: 270, height: 69 }, + onClick, + mode = "standard" + }, ref) => { + const filterId = (0, import_react.useId)(); + const [shaderMapUrl, setShaderMapUrl] = (0, import_react.useState)(""); + const isFirefox = navigator.userAgent.toLowerCase().includes("firefox"); + (0, import_react.useEffect)(() => { + if (mode === "shader") { + const url = generateShaderDisplacementMap(glassSize.width, glassSize.height); + setShaderMapUrl(url); + } + }, [mode, glassSize.width, glassSize.height]); + const backdropStyle = { + filter: isFirefox ? null : `url(#${filterId})`, + backdropFilter: `blur(${(overLight ? 12 : 4) + blurAmount * 32}px) saturate(${saturation}%)` + }; + return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)("div", { ref, className: `relative ${className} ${active ? "active" : ""} ${Boolean(onClick) ? "cursor-pointer" : ""}`, style, onClick, children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsx)(GlassFilter, { mode, id: filterId, displacementScale, aberrationIntensity, width: glassSize.width, height: glassSize.height, shaderMapUrl }), + /* @__PURE__ */ (0, import_jsx_runtime.jsxs)( + "div", + { + className: "glass", + style: { + borderRadius: `${cornerRadius}px`, + position: "relative", + display: "inline-flex", + alignItems: "center", + gap: "24px", + padding, + overflow: "hidden", + transition: "all 0.2s ease-in-out", + boxShadow: overLight ? "0px 16px 70px rgba(0, 0, 0, 0.75)" : "0px 12px 40px rgba(0, 0, 0, 0.25)" + }, + onMouseEnter, + onMouseLeave, + onMouseDown, + onMouseUp, + children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "span", + { + className: "glass__warp", + style: { + ...backdropStyle, + position: "absolute", + inset: "0" + } + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "div", + { + className: "transition-all duration-150 ease-in-out text-white", + style: { + position: "relative", + zIndex: 1, + font: "500 20px/1 system-ui", + textShadow: overLight ? "0px 2px 12px rgba(0, 0, 0, 0)" : "0px 2px 12px rgba(0, 0, 0, 0.4)" + }, + children + } + ) + ] + } + ) + ] }); + } +); +GlassContainer.displayName = "GlassContainer"; +function LiquidGlass({ + children, + displacementScale = 70, + blurAmount = 0.0625, + saturation = 140, + aberrationIntensity = 2, + elasticity = 0.15, + cornerRadius = 999, + globalMousePos: externalGlobalMousePos, + mouseOffset: externalMouseOffset, + mouseContainer = null, + className = "", + padding = "24px 32px", + overLight = false, + style = {}, + mode = "standard", + onClick +}) { + const glassRef = (0, import_react.useRef)(null); + const [isHovered, setIsHovered] = (0, import_react.useState)(false); + const [isActive, setIsActive] = (0, import_react.useState)(false); + const [glassSize, setGlassSize] = (0, import_react.useState)({ width: 270, height: 69 }); + const [internalGlobalMousePos, setInternalGlobalMousePos] = (0, import_react.useState)({ x: 0, y: 0 }); + const [internalMouseOffset, setInternalMouseOffset] = (0, import_react.useState)({ x: 0, y: 0 }); + const globalMousePos = externalGlobalMousePos || internalGlobalMousePos; + const mouseOffset = externalMouseOffset || internalMouseOffset; + const handleMouseMove = (0, import_react.useCallback)( + (e) => { + const container = mouseContainer?.current || glassRef.current; + if (!container) { + return; + } + const rect = container.getBoundingClientRect(); + const centerX = rect.left + rect.width / 2; + const centerY = rect.top + rect.height / 2; + setInternalMouseOffset({ + x: (e.clientX - centerX) / rect.width * 100, + y: (e.clientY - centerY) / rect.height * 100 + }); + setInternalGlobalMousePos({ + x: e.clientX, + y: e.clientY + }); + }, + [mouseContainer] + ); + (0, import_react.useEffect)(() => { + if (externalGlobalMousePos && externalMouseOffset) { + return; + } + const container = mouseContainer?.current || glassRef.current; + if (!container) { + return; + } + container.addEventListener("mousemove", handleMouseMove); + return () => { + container.removeEventListener("mousemove", handleMouseMove); + }; + }, [handleMouseMove, mouseContainer, externalGlobalMousePos, externalMouseOffset]); + const calculateDirectionalScale = (0, import_react.useCallback)(() => { + if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) { + return "scale(1)"; + } + const rect = glassRef.current.getBoundingClientRect(); + const pillCenterX = rect.left + rect.width / 2; + const pillCenterY = rect.top + rect.height / 2; + const pillWidth = glassSize.width; + const pillHeight = glassSize.height; + const deltaX = globalMousePos.x - pillCenterX; + const deltaY = globalMousePos.y - pillCenterY; + const edgeDistanceX = Math.max(0, Math.abs(deltaX) - pillWidth / 2); + const edgeDistanceY = Math.max(0, Math.abs(deltaY) - pillHeight / 2); + const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY); + const activationZone = 200; + if (edgeDistance > activationZone) { + return "scale(1)"; + } + const fadeInFactor = 1 - edgeDistance / activationZone; + const centerDistance = Math.sqrt(deltaX * deltaX + deltaY * deltaY); + if (centerDistance === 0) { + return "scale(1)"; + } + const normalizedX = deltaX / centerDistance; + const normalizedY = deltaY / centerDistance; + const stretchIntensity = Math.min(centerDistance / 300, 1) * elasticity * fadeInFactor; + const scaleX = 1 + Math.abs(normalizedX) * stretchIntensity * 0.3 - Math.abs(normalizedY) * stretchIntensity * 0.15; + const scaleY = 1 + Math.abs(normalizedY) * stretchIntensity * 0.3 - Math.abs(normalizedX) * stretchIntensity * 0.15; + return `scaleX(${Math.max(0.8, scaleX)}) scaleY(${Math.max(0.8, scaleY)})`; + }, [globalMousePos, elasticity, glassSize]); + const calculateFadeInFactor = (0, import_react.useCallback)(() => { + if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) { + return 0; + } + const rect = glassRef.current.getBoundingClientRect(); + const pillCenterX = rect.left + rect.width / 2; + const pillCenterY = rect.top + rect.height / 2; + const pillWidth = glassSize.width; + const pillHeight = glassSize.height; + const edgeDistanceX = Math.max(0, Math.abs(globalMousePos.x - pillCenterX) - pillWidth / 2); + const edgeDistanceY = Math.max(0, Math.abs(globalMousePos.y - pillCenterY) - pillHeight / 2); + const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY); + const activationZone = 200; + return edgeDistance > activationZone ? 0 : 1 - edgeDistance / activationZone; + }, [globalMousePos, glassSize]); + const calculateElasticTranslation = (0, import_react.useCallback)(() => { + if (!glassRef.current) { + return { x: 0, y: 0 }; + } + const fadeInFactor = calculateFadeInFactor(); + const rect = glassRef.current.getBoundingClientRect(); + const pillCenterX = rect.left + rect.width / 2; + const pillCenterY = rect.top + rect.height / 2; + return { + x: (globalMousePos.x - pillCenterX) * elasticity * 0.1 * fadeInFactor, + y: (globalMousePos.y - pillCenterY) * elasticity * 0.1 * fadeInFactor + }; + }, [globalMousePos, elasticity, calculateFadeInFactor]); + (0, import_react.useEffect)(() => { + const updateGlassSize = () => { + if (glassRef.current) { + const rect = glassRef.current.getBoundingClientRect(); + setGlassSize({ width: rect.width, height: rect.height }); + } + }; + updateGlassSize(); + window.addEventListener("resize", updateGlassSize); + return () => window.removeEventListener("resize", updateGlassSize); + }, []); + const transformStyle = `translate(calc(-50% + ${calculateElasticTranslation().x}px), calc(-50% + ${calculateElasticTranslation().y}px)) ${isActive && Boolean(onClick) ? "scale(0.96)" : calculateDirectionalScale()}`; + const baseStyle = { + ...style, + transform: transformStyle, + transition: "all ease-out 0.2s" + }; + const positionStyles = { + position: baseStyle.position || "relative", + top: baseStyle.top || "50%", + left: baseStyle.left || "50%" + }; + return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_jsx_runtime.Fragment, { children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "div", + { + className: `bg-black transition-all duration-150 ease-in-out pointer-events-none ${overLight ? "opacity-20" : "opacity-0"}`, + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition + } + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "div", + { + className: `bg-black transition-all duration-150 ease-in-out pointer-events-none mix-blend-overlay ${overLight ? "opacity-100" : "opacity-0"}`, + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition + } + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + GlassContainer, + { + ref: glassRef, + className, + style: baseStyle, + cornerRadius, + displacementScale: overLight ? displacementScale * 0.5 : displacementScale, + blurAmount, + saturation, + aberrationIntensity, + glassSize, + padding, + mouseOffset, + onMouseEnter: () => setIsHovered(true), + onMouseLeave: () => setIsHovered(false), + onMouseDown: () => setIsActive(true), + onMouseUp: () => setIsActive(false), + active: isActive, + overLight, + onClick, + mode, + children + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "span", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition, + pointerEvents: "none", + mixBlendMode: "screen", + opacity: 0.2, + padding: "1.5px", + WebkitMask: "linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0)", + WebkitMaskComposite: "xor", + maskComposite: "exclude", + boxShadow: "0 0 0 0.5px rgba(255, 255, 255, 0.5) inset, 0 1px 3px rgba(255, 255, 255, 0.25) inset, 0 1px 4px rgba(0, 0, 0, 0.35)", + background: `linear-gradient( + ${135 + mouseOffset.x * 1.2}deg, + rgba(255, 255, 255, 0.0) 0%, + rgba(255, 255, 255, ${0.12 + Math.abs(mouseOffset.x) * 8e-3}) ${Math.max(10, 33 + mouseOffset.y * 0.3)}%, + rgba(255, 255, 255, ${0.4 + Math.abs(mouseOffset.x) * 0.012}) ${Math.min(90, 66 + mouseOffset.y * 0.4)}%, + rgba(255, 255, 255, 0.0) 100% + )` + } + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "span", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + transition: baseStyle.transition, + pointerEvents: "none", + mixBlendMode: "overlay", + padding: "1.5px", + WebkitMask: "linear-gradient(#000 0 0) content-box, linear-gradient(#000 0 0)", + WebkitMaskComposite: "xor", + maskComposite: "exclude", + boxShadow: "0 0 0 0.5px rgba(255, 255, 255, 0.5) inset, 0 1px 3px rgba(255, 255, 255, 0.25) inset, 0 1px 4px rgba(0, 0, 0, 0.35)", + background: `linear-gradient( + ${135 + mouseOffset.x * 1.2}deg, + rgba(255, 255, 255, 0.0) 0%, + rgba(255, 255, 255, ${0.32 + Math.abs(mouseOffset.x) * 8e-3}) ${Math.max(10, 33 + mouseOffset.y * 0.3)}%, + rgba(255, 255, 255, ${0.6 + Math.abs(mouseOffset.x) * 0.012}) ${Math.min(90, 66 + mouseOffset.y * 0.4)}%, + rgba(255, 255, 255, 0.0) 100% + )` + } + } + ), + Boolean(onClick) && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_jsx_runtime.Fragment, { children: [ + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "div", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width + 1, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + pointerEvents: "none", + transition: "all 0.2s ease-out", + opacity: isHovered || isActive ? 0.5 : 0, + backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0) 50%)", + mixBlendMode: "overlay" + } + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "div", + { + style: { + ...positionStyles, + height: glassSize.height, + width: glassSize.width + 1, + borderRadius: `${cornerRadius}px`, + transform: baseStyle.transform, + pointerEvents: "none", + transition: "all 0.2s ease-out", + opacity: isActive ? 0.5 : 0, + backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 80%)", + mixBlendMode: "overlay" + } + } + ), + /* @__PURE__ */ (0, import_jsx_runtime.jsx)( + "div", + { + style: { + ...baseStyle, + height: glassSize.height, + width: glassSize.width + 1, + borderRadius: `${cornerRadius}px`, + position: baseStyle.position, + top: baseStyle.top, + left: baseStyle.left, + pointerEvents: "none", + transition: "all 0.2s ease-out", + opacity: isHovered ? 0.4 : isActive ? 0.8 : 0, + backgroundImage: "radial-gradient(circle at 50% 0%, rgba(255, 255, 255, 1) 0%, rgba(255, 255, 255, 0) 100%)", + mixBlendMode: "overlay" + } + } + ) + ] }) + ] }); +} diff --git a/plugins/liquid-glass/shader-utils.d.ts b/plugins/liquid-glass/shader-utils.d.ts new file mode 100644 index 0000000..3b39226 --- /dev/null +++ b/plugins/liquid-glass/shader-utils.d.ts @@ -0,0 +1,25 @@ +export interface Vec2 { + x: number; + y: number; +} +export interface ShaderOptions { + width: number; + height: number; + fragment: (uv: Vec2, mouse?: Vec2) => Vec2; + mousePosition?: Vec2; +} +export declare const fragmentShaders: { + liquidGlass: (uv: Vec2) => Vec2; +}; +export type FragmentShaderType = keyof typeof fragmentShaders; +export declare class ShaderDisplacementGenerator { + private options; + private canvas; + private context; + private canvasDPI; + constructor(options: ShaderOptions); + updateShader(mousePosition?: Vec2): string; + destroy(): void; + getScale(): number; +} +//# sourceMappingURL=shader-utils.d.ts.map \ No newline at end of file diff --git a/plugins/liquid-glass/shader-utils.d.ts.map b/plugins/liquid-glass/shader-utils.d.ts.map new file mode 100644 index 0000000..0a1bcc4 --- /dev/null +++ b/plugins/liquid-glass/shader-utils.d.ts.map @@ -0,0 +1 @@ +{"version":3,"file":"shader-utils.d.ts","sourceRoot":"","sources":["../src/shader-utils.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,IAAI;IACnB,CAAC,EAAE,MAAM,CAAA;IACT,CAAC,EAAE,MAAM,CAAA;CACV;AAED,MAAM,WAAW,aAAa;IAC5B,KAAK,EAAE,MAAM,CAAA;IACb,MAAM,EAAE,MAAM,CAAA;IACd,QAAQ,EAAE,CAAC,EAAE,EAAE,IAAI,EAAE,KAAK,CAAC,EAAE,IAAI,KAAK,IAAI,CAAA;IAC1C,aAAa,CAAC,EAAE,IAAI,CAAA;CACrB;AAsBD,eAAO,MAAM,eAAe;sBACR,IAAI,KAAG,IAAI;CAQ9B,CAAA;AAED,MAAM,MAAM,kBAAkB,GAAG,MAAM,OAAO,eAAe,CAAA;AAE7D,qBAAa,2BAA2B;IAK1B,OAAO,CAAC,OAAO;IAJ3B,OAAO,CAAC,MAAM,CAAmB;IACjC,OAAO,CAAC,OAAO,CAA0B;IACzC,OAAO,CAAC,SAAS,CAAI;gBAED,OAAO,EAAE,aAAa;IAa1C,YAAY,CAAC,aAAa,CAAC,EAAE,IAAI,GAAG,MAAM;IA6D1C,OAAO,IAAI,IAAI;IAIf,QAAQ,IAAI,MAAM;CAGnB"} \ No newline at end of file diff --git a/plugins/liquid-glass/utils.d.ts b/plugins/liquid-glass/utils.d.ts new file mode 100644 index 0000000..a24d355 --- /dev/null +++ b/plugins/liquid-glass/utils.d.ts @@ -0,0 +1,4 @@ +export declare const displacementMap = ""; +export declare const polarDisplacementMap = ""; +export declare const prominentDisplacementMap = ""; +//# sourceMappingURL=utils.d.ts.map \ No newline at end of file diff --git a/plugins/liquid-glass/utils.d.ts.map b/plugins/liquid-glass/utils.d.ts.map new file mode 100644 index 0000000..b9fbaf5 --- /dev/null +++ b/plugins/liquid-glass/utils.d.ts.map @@ -0,0 +1 @@ +{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../src/utils.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,eAAe,40LAC+yL,CAAA;AAE30L,eAAO,MAAM,oBAAoB,4pLAC0nL,CAAA;AAE3pL,eAAO,MAAM,wBAAwB,uj5BACih5B,CAAA"} \ No newline at end of file diff --git a/tailwind.config.js b/tailwind.config.js new file mode 100644 index 0000000..0db4379 --- /dev/null +++ b/tailwind.config.js @@ -0,0 +1,74 @@ +/** @type {import('tailwindcss').Config} */ +module.exports = { + darkMode: ["class"], + content: [ + './pages/**/*.{ts,tsx}', + './components/**/*.{ts,tsx}', + './app/**/*.{ts,tsx}', + './src/**/*.{ts,tsx}', + ], + theme: { + extend: { + colors: { + border: "hsl(var(--border))", + input: "hsl(var(--input))", + ring: "hsl(var(--ring))", + background: "hsl(var(--background))", + foreground: "hsl(var(--foreground))", + primary: { + DEFAULT: "hsl(var(--primary))", + foreground: "hsl(var(--primary-foreground))", + }, + secondary: { + DEFAULT: "hsl(var(--secondary))", + foreground: "hsl(var(--secondary-foreground))", + }, + destructive: { + DEFAULT: "hsl(var(--destructive))", + foreground: "hsl(var(--destructive-foreground))", + }, + muted: { + DEFAULT: "hsl(var(--muted))", + foreground: "hsl(var(--muted-foreground))", + }, + accent: { + DEFAULT: "hsl(var(--accent))", + foreground: "hsl(var(--accent-foreground))", + }, + popover: { + DEFAULT: "hsl(var(--popover))", + foreground: "hsl(var(--popover-foreground))", + }, + card: { + DEFAULT: "hsl(var(--card))", + foreground: "hsl(var(--card-foreground))", + }, + }, + borderRadius: { + lg: "var(--radius)", + md: "calc(var(--radius) - 2px)", + sm: "calc(var(--radius) - 4px)", + }, + keyframes: { + "accordion-down": { + from: { height: 0 }, + to: { height: "var(--radix-accordion-content-height)" }, + }, + "accordion-up": { + from: { height: "var(--radix-accordion-content-height)" }, + to: { height: 0 }, + }, + }, + animation: { + "accordion-down": "accordion-down 0.2s ease-out", + "accordion-up": "accordion-up 0.2s ease-out", + }, + transitionDelay: { + '100': '100ms', + '200': '200ms', + '300': '300ms', + } + }, + }, + plugins: [require("tailwindcss-animate")], +} \ No newline at end of file