Created
August 7, 2022 16:26
-
-
Save syeo66/af150559bb75cde7e77382e935b6fcae to your computer and use it in GitHub Desktop.
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
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