Skip to content

Instantly share code, notes, and snippets.

@syeo66
Created August 7, 2022 16:26
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save syeo66/af150559bb75cde7e77382e935b6fcae to your computer and use it in GitHub Desktop.
Save syeo66/af150559bb75cde7e77382e935b6fcae to your computer and use it in GitHub Desktop.
import { format } from 'date-fns'
import React, { memo, useEffect, useState } from 'react'
import styled from 'styled-components'
import { AnalogClockConfigType } from './ClockType'
const TIMER_PRECISION = 250
interface AnalogClockProps {
config: AnalogClockConfigType
}
const AnalogClock: React.FC<AnalogClockProps> = ({ config }) => {
const [time, setTime] = useState(() => new Date())
useEffect(() => {
const interval = setInterval(() => {
const now = new Date()
if (time.getSeconds() !== now.getSeconds()) {
setTime(now)
}
}, TIMER_PRECISION)
return () => clearInterval(interval)
}, [time])
const isDark = config.style === 'dark'
return (
<Clock>
<Background dark={isDark}>
<Hours time={time} dark={isDark} />
<Minutes time={time} dark={isDark} />
<Seconds time={time} dark={isDark} />
</Background>
<DateView>{format(time, config.dateFormat)}</DateView>
</Clock>
)
}
const Clock = styled.div`
display: flex;
flex-direction: column;
align-items: center;
`
interface BackgroundProps {
dark?: boolean
}
const Background = styled.div<BackgroundProps>`
background-color: ${({ dark }) => (dark ? '#000' : '#fff')};
border-radius: 50%;
border: ${({ dark }) => (dark ? '#eee' : '#000')} 1px solid;
height: 100px;
margin-bottom: 0.5rem;
position: relative;
width: 100px;
`
interface DateProps {
dark?: boolean
time: Date
}
const Hours = styled.div.attrs<DateProps>(({ time }) => ({
style: {
transform: `rotateZ(${((time.getHours() % 12) * 60 + time.getMinutes()) * 0.5}deg)`,
},
}))<DateProps>`
background-color: ${({ dark }) => (dark ? '#aaa' : '#000')};
border-radius: 2.5px;
height: 30px;
left: calc(50% - 2.5px);
position: absolute;
top: 25px;
transform-origin: center calc(100% - 5px);
width: 5px;
`
const Minutes = styled(Hours).attrs<DateProps>(({ time }) => ({
style: { transform: `rotateZ(${(time.getMinutes() * 60 + time.getSeconds()) * 0.1}deg)` },
}))<DateProps>`
height: 45px;
top: 10px;
`
const Seconds = styled(Hours).attrs<DateProps>(({ time }) => ({
style: { transform: `rotateZ(${time.getSeconds() * 6}deg)` },
}))<DateProps>`
background-color: red;
height: 50px;
left: calc(50% - 0.5px);
top: 5px;
width: 1px;
`
const DateView = styled.div`
font-size: 1rem;
text-shadow: 0 0 5px rgba(0, 0, 0, 0.5), 0 0 10px rgba(0, 0, 0, 0.4), 0 0 15px rgba(0, 0, 0, 0.3);
`
export default memo(AnalogClock)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment