-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add pixel background animation + "join discord" button
- Loading branch information
1 parent
9738442
commit de9638e
Showing
12 changed files
with
355 additions
and
11 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,150 @@ | ||
import { DISCORD_INVITE_URL } from '@/constants/discord' | ||
import { prefix } from '@/utils/prefix' | ||
import Image from 'next/image' | ||
import { motion } from 'framer-motion' | ||
import { FC, useEffect, useState } from 'react' | ||
import prefersReducedMotion from '@/utils/prefersReducedMotion' | ||
|
||
const PARTICLES_COUNT = Array.from({ length: 20 }, (_, i) => ({ | ||
index: i, | ||
x: Math.random() * 500 - 250, | ||
y: Math.random() * 300 - 150, | ||
delay: Math.random() * 2, | ||
duration: 2 + Math.random() * 2, | ||
})) | ||
|
||
const Particle: FC<{ | ||
x: number | ||
y: number | ||
delay: number | ||
duration: number | ||
isHovered: boolean | ||
}> = ({ x, y, delay, duration, isHovered }) => { | ||
const particleVariants = { | ||
hidden: { | ||
opacity: 0, | ||
x: x, | ||
y: y, | ||
duration: 10, | ||
delay: delay, | ||
}, | ||
visible: { | ||
opacity: [0, 0.8, 0], | ||
x: 0, | ||
y: 0, | ||
transition: { | ||
duration: 2, | ||
delay: Math.random() * 2, | ||
repeat: Infinity, | ||
repeatType: 'loop' as const, | ||
}, | ||
}, | ||
hover: { | ||
opacity: [0, 1, 0], | ||
x: 1, | ||
y: 1, | ||
transition: { | ||
duration: duration, | ||
delay: Math.random() * 2, | ||
repeat: Infinity, | ||
repeatType: 'loop' as const, | ||
}, | ||
}, | ||
} | ||
|
||
return ( | ||
<motion.div | ||
className="absolute w-2 h-2 bg-discordPurple rounded-sm" | ||
initial="hidden" | ||
variants={particleVariants} | ||
// animate={isHovered ? 'hover' : 'visible'} | ||
animate="visible" // Im sure there is a way to make the duration switch work, but it just doesn't seem to work for me + does not look good | ||
/> | ||
) | ||
} | ||
|
||
const Particles: FC<{ isHovered: boolean }> = ({ isHovered }) => { | ||
if (prefersReducedMotion()) return null | ||
return ( | ||
<> | ||
{PARTICLES_COUNT.map(({ index, x, y, delay, duration }) => ( | ||
<Particle | ||
key={index} | ||
x={x} | ||
y={y} | ||
delay={delay} | ||
duration={duration} | ||
isHovered={isHovered} | ||
/> | ||
))} | ||
</> | ||
) | ||
} | ||
|
||
// TODO: If we ever run this website on server, we should cache this value! | ||
const MemberCount: FC = () => { | ||
const [memberCount, setMemberCount] = useState<number | null>(null) | ||
const [onlineCount, setOnlineCount] = useState<number | null>(null) | ||
|
||
useEffect(() => { | ||
const fetchMemberCount = async () => { | ||
const INVITE_CODE = DISCORD_INVITE_URL.split('/').pop() | ||
try { | ||
const response = await fetch( | ||
`https://discord.com/api/v9/invites/${INVITE_CODE}?with_counts=true&with_expiration=true` | ||
) | ||
const data = await response.json() | ||
setMemberCount(data.approximate_member_count) | ||
setOnlineCount(data.approximate_presence_count) | ||
} catch (error) { | ||
console.error('Error fetching member count:', error) | ||
} | ||
} | ||
|
||
fetchMemberCount() | ||
}, []) | ||
|
||
return ( | ||
<div | ||
className="text-white text-center" | ||
title={`There are ${onlineCount} members online`}> | ||
<p>Join our Discord!</p> | ||
{memberCount !== null && ( | ||
<p className="text-xs text-left text-gray-300">{memberCount} members</p> | ||
)} | ||
</div> | ||
) | ||
} | ||
|
||
const JoinDiscord: FC = () => { | ||
const [isHovered, setIsHovered] = useState(false) | ||
|
||
return ( | ||
<div className="relative flex items-center justify-center"> | ||
<Particles isHovered={isHovered} /> | ||
<motion.a | ||
href={DISCORD_INVITE_URL} | ||
target="_blank" | ||
className="px-4 py-2 text-white bg-discordPurple rounded-lg z-10 shadow-lg flex items-center gap-2" | ||
whileHover={{ scale: 1.1 }} | ||
onHoverStart={() => setIsHovered(true)} | ||
onHoverEnd={() => setIsHovered(false)}> | ||
{/* | ||
I would use this icon... but its the ugliest thing I've ever seen | ||
<Discord width={48} height={48} /> | ||
So lets use a random svg | ||
*/} | ||
<Image | ||
src={`${prefix}/discord.svg`} | ||
alt="Discord logo" | ||
width={48} | ||
height={40} | ||
/> | ||
|
||
<MemberCount /> | ||
</motion.a> | ||
</div> | ||
) | ||
} | ||
|
||
export default JoinDiscord |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,75 @@ | ||
'use client' | ||
|
||
import React, { useEffect, useState, ReactNode } from 'react' | ||
import { | ||
AnimatedBackgroundProvider, | ||
useAnimatedBackground, | ||
} from '@/context/AnimatedBackgroundContext' | ||
|
||
const PixelGrid = () => { | ||
const { isActive } = useAnimatedBackground() | ||
const [numPixels, setNumPixels] = useState({ x: 0, y: 0 }) | ||
const [pixels, setPixels] = useState<number[]>([]) | ||
const [pixelSize, _] = useState<number>(64) // Smaller values are less performant | ||
|
||
const calculatePixels = () => { | ||
const x = Math.ceil(window.innerWidth / pixelSize) | ||
const y = Math.ceil(window.innerHeight / pixelSize) | ||
return { x, y } | ||
} | ||
|
||
const generatePixels = (x: number, y: number) => { | ||
return Array(x * y) | ||
.fill(0) | ||
.map(() => Math.random()) | ||
} | ||
|
||
useEffect(() => { | ||
const { x, y } = calculatePixels() | ||
setNumPixels({ x, y }) | ||
setPixels(generatePixels(x, y)) | ||
|
||
const handleResize = () => { | ||
const { x, y } = calculatePixels() | ||
setNumPixels({ x, y }) | ||
setPixels(generatePixels(x, y)) | ||
} | ||
|
||
window.addEventListener('resize', handleResize) | ||
|
||
return () => { | ||
window.removeEventListener('resize', handleResize) | ||
} | ||
}, []) | ||
|
||
return ( | ||
<div | ||
className="fixed pointer-events-none z-[-1] grid" | ||
style={{ gridTemplateColumns: `repeat(${numPixels.x}, ${pixelSize}px)` }}> | ||
{pixels.map((delay, index) => ( | ||
<div | ||
key={index} | ||
style={{ | ||
width: `${pixelSize}px`, | ||
height: `${pixelSize}px`, | ||
transitionDelay: `${delay * 750}ms`, | ||
}} | ||
className={`transition-colors ${ | ||
isActive ? 'bg-csred' : 'bg-background' | ||
}`} | ||
/> | ||
))} | ||
</div> | ||
) | ||
} | ||
|
||
const PixelBackground = ({ children }: { children: ReactNode }) => { | ||
return ( | ||
<AnimatedBackgroundProvider> | ||
<PixelGrid /> | ||
{children} | ||
</AnimatedBackgroundProvider> | ||
) | ||
} | ||
|
||
export default PixelBackground |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
export const DISCORD_INVITE_URL = 'https://discord.gg/fmp7p9Ca4y' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,41 @@ | ||
'use client' | ||
import { createContext, useContext, useState, ReactNode } from 'react' | ||
|
||
interface AnimatedBackgroundContextProps { | ||
isActive: boolean | ||
toggleBackground: () => void | ||
} | ||
|
||
const AnimatedBackgroundContext = createContext< | ||
AnimatedBackgroundContextProps | undefined | ||
>(undefined) | ||
|
||
export const useAnimatedBackground = (): AnimatedBackgroundContextProps => { | ||
const context = useContext(AnimatedBackgroundContext) | ||
if (!context) { | ||
throw new Error( | ||
'useAnimatedBackground must be used within an AnimatedBackgroundProvider' | ||
) | ||
} | ||
return context | ||
} | ||
|
||
interface AnimatedBackgroundProviderProps { | ||
children: ReactNode | ||
} | ||
|
||
export const AnimatedBackgroundProvider = ({ | ||
children, | ||
}: AnimatedBackgroundProviderProps) => { | ||
const [isActive, setIsActive] = useState(false) | ||
|
||
const toggleBackground = () => { | ||
setIsActive(!isActive) | ||
} | ||
|
||
return ( | ||
<AnimatedBackgroundContext.Provider value={{ isActive, toggleBackground }}> | ||
{children} | ||
</AnimatedBackgroundContext.Provider> | ||
) | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
Oops, something went wrong.