All components
Linear Modal
modalsUi-Layouts component.
responsive · 680px
Install
Same command in any shadcn project — React (Vite/CRA), Next.js, Remix, Astro, and more:
$
npx shadcn@latest add https://your-domain/r/linear-modal.jsonUsage
"use client";
import {
Dialog,
DialogClose,
DialogContainer,
DialogContent,
DialogDescription,
DialogImage,
DialogTitle,
DialogTrigger,
} from "@/registry/ui-layouts/linear-modal";
import { Plus } from "lucide-react";
export default function Demo() {
return (
<div className="flex h-full items-center justify-center">
<Dialog
transition={{
type: "spring",
bounce: 0.05,
duration: 0.5,
}}
>
<DialogTrigger
style={{ borderRadius: "12px" }}
className="flex w-64 flex-col overflow-hidden border dark:bg-black bg-white cursor-pointer"
>
<DialogImage
src="https://images.unsplash.com/photo-1517849845537-4d257902454a?w=800"
alt="Mountain landscape"
className="h-44 w-full object-cover"
/>
<div className="flex grow flex-row items-end justify-between p-3">
<div>
<DialogTitle className="text-base font-semibold text-primary">
Mountain Vista
</DialogTitle>
<DialogDescription
disableLayoutAnimation
variants={{
initial: { opacity: 0 },
animate: { opacity: 1 },
exit: { opacity: 0 },
}}
>
<span className="text-xs text-muted-foreground">
Click to expand
</span>
</DialogDescription>
</div>
<button className="p-2 bg-neutral-200 dark:bg-neutral-800 rounded-lg">
<Plus className="w-4 h-4" />
</button>
</div>
</DialogTrigger>
<DialogContainer className="grid place-items-center h-full">
<DialogContent
style={{ borderRadius: "24px" }}
className="h-fit mx-auto border"
>
<div className="flex flex-col relative dark:bg-neutral-900 bg-neutral-100 w-[480px] max-w-[90vw] overflow-hidden">
<DialogImage
src="https://images.unsplash.com/photo-1517849845537-4d257902454a?w=800"
alt="Mountain landscape"
className="h-64 w-full object-cover"
/>
<div className="p-6">
<DialogTitle className="text-3xl font-bold dark:text-white text-zinc-900">
Mountain Vista
</DialogTitle>
<DialogDescription
disableLayoutAnimation
variants={{
initial: { opacity: 0, scale: 0.9, y: -20 },
animate: { opacity: 1, scale: 1, y: 0 },
exit: { opacity: 0, scale: 0.9, y: -20 },
}}
>
<p className="mt-2 text-sm text-zinc-600 dark:text-zinc-400 leading-relaxed">
A breathtaking alpine panorama captured at golden hour. The
layered ridgelines fade into misty blue distance, offering a
sense of boundless scale and serenity.
</p>
<div className="flex gap-2 mt-4">
<button className="px-4 py-2 bg-black dark:bg-white text-white dark:text-black rounded-lg text-sm font-medium">
Download
</button>
<button className="px-4 py-2 border rounded-lg text-sm font-medium">
Share
</button>
</div>
</DialogDescription>
</div>
<DialogClose className="text-zinc-800 dark:text-zinc-100 bg-neutral-200 dark:bg-neutral-800 p-2 rounded-lg" />
</div>
</DialogContent>
</DialogContainer>
</Dialog>
</div>
);
}Component source
'use client';
import { cn } from '@/lib/utils';
import { XIcon } from 'lucide-react';
import { AnimatePresence, MotionConfig, type Transition, type Variant, motion } from 'motion/react';
import React, { useCallback, useContext, useEffect, useId, useMemo, useRef, useState } from 'react';
import { createPortal } from 'react-dom';
interface DialogContextType {
isOpen: boolean;
setIsOpen: React.Dispatch<React.SetStateAction<boolean>>;
uniqueId: string;
triggerRef: React.RefObject<HTMLDivElement>;
}
const DialogContext = React.createContext<DialogContextType | null>(null);
function useDialog() {
const context = useContext(DialogContext);
if (!context) {
throw new Error('useDialog must be used within a DialogProvider');
}
return context;
}
type DialogProviderProps = {
children: React.ReactNode;
transition?: Transition;
};
function DialogProvider({ children, transition }: DialogProviderProps) {
const [isOpen, setIsOpen] = useState(false);
const uniqueId = useId();
const triggerRef = useRef<HTMLDivElement>(null);
const contextValue = useMemo(
() => ({ isOpen, setIsOpen, uniqueId, triggerRef }),
[isOpen, uniqueId]
);
return (
<DialogContext.Provider
//@ts-expect-error
value={contextValue}
>
<MotionConfig transition={transition}>{children}</MotionConfig>
</DialogContext.Provider>
);
}
type DialogProps = {
children: React.ReactNode;
transition?: Transition;
};
function Dialog({ children, transition }: DialogProps) {
return (
<DialogProvider>
<MotionConfig transition={transition}>{children}</MotionConfig>
</DialogProvider>
);
}
type DialogTriggerProps = {
children: React.ReactNode;
className?: string;
style?: React.CSSProperties;
triggerRef?: React.RefObject<HTMLDivElement>;
};
function DialogTrigger({ children, className, style, triggerRef }: DialogTriggerProps) {
const { setIsOpen, isOpen, uniqueId } = useDialog();
const handleClick = useCallback(() => {
setIsOpen(!isOpen);
}, [isOpen, setIsOpen]);
const handleKeyDown = useCallback(
(event: React.KeyboardEvent) => {
if (event.key === 'Enter' || event.key === ' ') {
event.preventDefault();
setIsOpen(!isOpen);
}
},
[isOpen, setIsOpen]
);
return (
<motion.div
ref={triggerRef}
layoutId={`dialog-${uniqueId}`}
className={cn('relative cursor-pointer', className)}
onClick={handleClick}
onKeyDown={handleKeyDown}
style={style}
role='button'
aria-haspopup='dialog'
aria-expanded={isOpen}
aria-controls={`dialog-content-${uniqueId}`}
>
{children}
</motion.div>
);
}
type DialogContent = {
children: React.ReactNode;
className?: string;
style?: React.CSSProperties;
};
function DialogContent({ children, className, style }: DialogContent) {
const { setIsOpen, isOpen, uniqueId, triggerRef } = useDialog();
const containerRef = useRef<HTMLDivElement>(null);
const [firstFocusableElement, setFirstFocusableElement] = useState<HTMLElement | null>(null);
const [lastFocusableElement, setLastFocusableElement] = useState<HTMLElement | null>(null);
useEffect(() => {
const handleKeyDown = (event: KeyboardEvent) => {
if (event.key === 'Escape') {
setIsOpen(false);
}
if (event.key === 'Tab') {
if (!firstFocusableElement || !lastFocusableElement) return;
if (event.shiftKey) {
if (document.activeElement === firstFocusableElement) {
event.preventDefault();
lastFocusableElement.focus();
}
} else {
if (document.activeElement === lastFocusableElement) {
event.preventDefault();
firstFocusableElement.focus();
}
}
}
};
document.addEventListener('keydown', handleKeyDown);
return () => {
document.removeEventListener('keydown', handleKeyDown);
};
}, [setIsOpen, firstFocusableElement, lastFocusableElement]);
useEffect(() => {
if (isOpen) {
document.body.style.overflow = 'hidden';
const focusableElements = containerRef.current?.querySelectorAll(
'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'
);
if (focusableElements && focusableElements.length > 0) {
setFirstFocusableElement(focusableElements[0] as HTMLElement);
setLastFocusableElement(focusableElements[focusableElements.length - 1] as HTMLElement);
// Delay focus slightly to allow animation to start
requestAnimationFrame(() => {
(focusableElements[0] as HTMLElement).focus();
});
}
if (containerRef.current) {
containerRef.current.scrollTop = 0;
}
} else {
document.body.style.overflow = '';
triggerRef.current?.focus();
}
}, [isOpen, triggerRef]);
return (
<>
<motion.div
ref={containerRef}
layoutId={`dialog-${uniqueId}`}
className={cn('overflow-hidden', className)}
style={{
...style,
willChange: 'transform, opacity', // GPU acceleration
}}
role='dialog'
aria-modal='true'
aria-labelledby={`dialog-title-${uniqueId}`}
aria-describedby={`dialog-description-${uniqueId}`}
initial={{ scale: 0.95, opacity: 0 }}
animate={{ scale: 1, opacity: 1 }}
exit={{ scale: 0.95, opacity: 0 }}
transition={{
type: 'spring',
damping: 25,
stiffness: 300,
mass: 0.8,
}}
>
{children}
</motion.div>
</>
);
}
type DialogContainerProps = {
children: React.ReactNode;
className?: string;
overlayClassName?: string;
style?: React.CSSProperties;
};
function DialogContainer({ children, className, overlayClassName }: DialogContainerProps) {
const { isOpen, setIsOpen, uniqueId } = useDialog();
const [mounted, setMounted] = useState(false);
useEffect(() => {
const drawerWrapper = document.querySelectorAll('[drawer-wrapper]');
if (isOpen) {
document.body.classList.add('overflow-hidden');
drawerWrapper.forEach((wrapper) => wrapper?.classList.add('open'));
} else {
document.body.classList.remove('overflow-hidden');
drawerWrapper.forEach((wrapper) => wrapper?.classList.remove('open'));
}
const handleKeyDown = (event: KeyboardEvent) => {
if (event.key === 'Escape') {
setIsOpen(false);
}
};
document.addEventListener('keydown', handleKeyDown);
return () => {
document.removeEventListener('keydown', handleKeyDown);
};
}, [isOpen]);
useEffect(() => {
setMounted(true);
return () => {
setMounted(false);
};
}, []);
if (!mounted) return null;
return createPortal(
<AnimatePresence initial={false} mode='wait'>
{isOpen && (
<>
<motion.div
key={`backdrop-${uniqueId}`}
data-lenis-prevent
className={cn(
'fixed inset-0 h-full z-50 w-full backdrop-blur-xl dark:bg-[radial-gradient(125%_125%_at_50%_10%,#050505_40%,#243aff_100%)] bg-[radial-gradient(125%_125%_at_50%_10%,#ffffff_40%,#243aff_100%)]',
overlayClassName
)}
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
transition={{
duration: 0.2,
ease: [0.4, 0.0, 0.4, 1],
}}
onClick={() => setIsOpen(false)}
></motion.div>
<motion.div
className={cn(`fixed inset-0 z-50 w-fit mx-auto`, className)}
style={{ willChange: 'transform' }} // GPU acceleration for transforms
>
{children}
</motion.div>
</>
)}
</AnimatePresence>,
document.body
);
}
type DialogTitleProps = {
children: React.ReactNode;
className?: string;
style?: React.CSSProperties;
};
function DialogTitle({ children, className, style }: DialogTitleProps) {
const { uniqueId } = useDialog();
return (
<motion.h1
layoutId={`dialog-title-container-${uniqueId}`}
className={className}
style={style}
layout
>
{children}
</motion.h1>
);
}
type DialogSubtitleProps = {
children: React.ReactNode;
className?: string;
style?: React.CSSProperties;
};
function DialogSubtitle({ children, className, style }: DialogSubtitleProps) {
const { uniqueId } = useDialog();
return (
<motion.div
layoutId={`dialog-subtitle-container-${uniqueId}`}
className={className}
style={style}
>
{children}
</motion.div>
);
}
type DialogDescriptionProps = {
children: React.ReactNode;
className?: string;
disableLayoutAnimation?: boolean;
variants?: {
initial: Variant;
animate: Variant;
exit: Variant;
};
};
function DialogDescription({
children,
className,
variants,
disableLayoutAnimation,
}: DialogDescriptionProps) {
const { uniqueId } = useDialog();
return (
<motion.div
key={`dialog-description-${uniqueId}`}
layoutId={disableLayoutAnimation ? undefined : `dialog-description-content-${uniqueId}`}
variants={variants}
className={className}
initial='initial'
animate='animate'
exit='exit'
id={`dialog-description-${uniqueId}`}
>
{children}
</motion.div>
);
}
type DialogImageProps = {
src: string;
alt: string;
className?: string;
style?: React.CSSProperties;
};
function DialogImage({ src, alt, className, style }: DialogImageProps) {
const { uniqueId } = useDialog();
return (
<motion.img
src={src}
alt={alt}
className={cn(className)}
layoutId={`dialog-img-${uniqueId}`}
style={style}
/>
);
}
type DialogCloseProps = {
children?: React.ReactNode;
className?: string;
variants?: {
initial: Variant;
animate: Variant;
exit: Variant;
};
};
function DialogClose({ children, className, variants }: DialogCloseProps) {
const { setIsOpen, uniqueId } = useDialog();
const handleClose = useCallback(() => {
setIsOpen(false);
}, [setIsOpen]);
return (
<motion.button
onClick={handleClose}
type='button'
aria-label='Close dialog'
key={`dialog-close-${uniqueId}`}
className={cn('absolute right-6 top-6 text-white', className)}
initial='initial'
animate='animate'
exit='exit'
variants={variants}
>
{children || <XIcon size={24} />}
</motion.button>
);
}
export {
Dialog,
DialogClose,
DialogContainer,
DialogContent,
DialogDescription,
DialogImage,
DialogSubtitle,
DialogTitle,
DialogTrigger,
};Dependencies
motionlucide-react
Source: Ui-Layouts