From 5b7ccf0b671e2999b62befc729a3e517a0433728 Mon Sep 17 00:00:00 2001 From: Bertrand Yuan Date: Mon, 15 Dec 2025 23:48:10 +0800 Subject: initial commit -- the front-end prototype The initial code is base on Anirudh's work. More to see at: https://github.com/techwithanirudh/shadcn-blog Therefore, the code in this commit is under MIT license. --- src/components/icons/animated/check.tsx | 111 ++++++++++++++++++++++++++ src/components/icons/animated/upload.tsx | 102 ++++++++++++++++++++++++ src/components/icons/icons.tsx | 131 +++++++++++++++++++++++++++++++ 3 files changed, 344 insertions(+) create mode 100644 src/components/icons/animated/check.tsx create mode 100644 src/components/icons/animated/upload.tsx create mode 100644 src/components/icons/icons.tsx (limited to 'src/components/icons') diff --git a/src/components/icons/animated/check.tsx b/src/components/icons/animated/check.tsx new file mode 100644 index 0000000..24a246c --- /dev/null +++ b/src/components/icons/animated/check.tsx @@ -0,0 +1,111 @@ +'use client'; + +import { cn } from '@/lib/utils'; +import type { Variants } from 'motion/react'; +import { motion, useAnimation } from 'motion/react'; +import type { HTMLAttributes } from 'react'; +import { forwardRef, useCallback, useImperativeHandle, useRef } from 'react'; + +export interface CheckIconHandle { + startAnimation: () => void; + stopAnimation: () => void; +} + +interface CheckIconProps extends HTMLAttributes { + size?: number; +} + +const pathVariants: Variants = { + normal: { + opacity: 1, + pathLength: 1, + scale: 1, + transition: { + duration: 0.3, + opacity: { duration: 0.1 }, + }, + }, + animate: { + opacity: [0, 1], + pathLength: [0, 1], + scale: [0.5, 1], + transition: { + duration: 0.4, + opacity: { duration: 0.1 }, + }, + }, +}; + +const CheckIcon = forwardRef( + ({ onMouseEnter, onMouseLeave, className, size = 28, ...props }, ref) => { + const controls = useAnimation(); + const isControlledRef = useRef(false); + + useImperativeHandle(ref, () => { + isControlledRef.current = true; + + return { + startAnimation: () => controls.start('animate'), + stopAnimation: () => controls.start('normal'), + }; + }); + + const handleMouseEnter = useCallback( + (e: React.MouseEvent) => { + if (!isControlledRef.current) { + controls.start('animate'); + } else { + onMouseEnter?.(e); + } + }, + [controls, onMouseEnter], + ); + + const handleMouseLeave = useCallback( + (e: React.MouseEvent) => { + if (!isControlledRef.current) { + controls.start('normal'); + } else { + onMouseLeave?.(e); + } + }, + [controls, onMouseLeave], + ); + + return ( +
+ {/* biome-ignore lint/a11y/noSvgWithoutTitle: */} + + + +
+ ); + }, +); + +CheckIcon.displayName = 'CheckIcon'; + +export { CheckIcon }; diff --git a/src/components/icons/animated/upload.tsx b/src/components/icons/animated/upload.tsx new file mode 100644 index 0000000..19ac51e --- /dev/null +++ b/src/components/icons/animated/upload.tsx @@ -0,0 +1,102 @@ +'use client'; + +import { cn } from '@/lib/utils'; +import type { Variants } from 'motion/react'; +import { motion, useAnimation } from 'motion/react'; +import type { HTMLAttributes } from 'react'; +import { forwardRef, useCallback, useImperativeHandle, useRef } from 'react'; + +export interface UploadIconHandle { + startAnimation: () => void; + stopAnimation: () => void; +} + +interface UploadIconProps extends HTMLAttributes { + size?: number; +} + +const arrowVariants: Variants = { + normal: { y: 0 }, + animate: { + y: -2, + transition: { + type: 'spring', + stiffness: 200, + damping: 10, + mass: 1, + }, + }, +}; + +const UploadIcon = forwardRef( + ({ onMouseEnter, onMouseLeave, className, size = 28, ...props }, ref) => { + const controls = useAnimation(); + const isControlledRef = useRef(false); + + useImperativeHandle(ref, () => { + isControlledRef.current = true; + + return { + startAnimation: () => controls.start('animate'), + stopAnimation: () => controls.start('normal'), + }; + }); + + const handleMouseEnter = useCallback( + (e: React.MouseEvent) => { + if (!isControlledRef.current) { + controls.start('animate'); + } else { + onMouseEnter?.(e); + } + }, + [controls, onMouseEnter], + ); + + const handleMouseLeave = useCallback( + (e: React.MouseEvent) => { + if (!isControlledRef.current) { + controls.start('normal'); + } else { + onMouseLeave?.(e); + } + }, + [controls, onMouseLeave], + ); + + return ( +
+ {/* biome-ignore lint/a11y/noSvgWithoutTitle: */} + + + + + + + +
+ ); + }, +); + +UploadIcon.displayName = 'UploadIcon'; + +export { UploadIcon }; diff --git a/src/components/icons/icons.tsx b/src/components/icons/icons.tsx new file mode 100644 index 0000000..e861252 --- /dev/null +++ b/src/components/icons/icons.tsx @@ -0,0 +1,131 @@ +import type { Icon as LucideIcon, LucideProps } from 'lucide-react'; +import { + AlertTriangle, + ArrowRight, + ArrowUpRight, + Check, + CheckCircle, + ChevronDown, + ChevronLeft, + ChevronRight, + ClipboardCheck, + Code, + CreditCard, + File, + FileText, + HelpCircle, + Home, + Image, + Info, + Laptop, + Loader2, + LogIn, + LogOut, + Mail, + Menu, + Moon, + MoreVertical, + Newspaper, + Pizza, + Plus, + Rss, + SendHorizonal, + Settings, + ShareIcon, + SunMedium, + Tag, + Tags, + Trash, + User, + X, +} from 'lucide-react'; + +export type Icon = typeof LucideIcon; + +export const Icons = { + logo: Code, + close: X, + menu: Menu, + code: Code, + copied: ClipboardCheck, + success: CheckCircle, + spinner: Loader2, + chevronLeft: ChevronLeft, + chevronRight: ChevronRight, + trash: Trash, + tags: Tags, + tag: Tag, + share: ShareIcon, + posts: Newspaper, + post: FileText, + page: File, + media: Image, + settings: Settings, + billing: CreditCard, + ellipsis: MoreVertical, + add: Plus, + logIn: LogIn, + logOut: LogOut, + warning: AlertTriangle, + user: User, + arrowRight: ArrowRight, + help: HelpCircle, + pizza: Pizza, + sun: SunMedium, + moon: Moon, + laptop: Laptop, + home: Home, + info: Info, + arrowUpRight: ArrowUpRight, + chevronDown: ChevronDown, + mail: Mail, + send: SendHorizonal, + gitHub: ({ ...props }: LucideProps) => ( + + ), + google: ({ ...props }: LucideProps) => ( + + ), + check: Check, + rss: Rss, +}; -- cgit v1.2.3