mirror of
https://github.com/roleypoly/roleypoly.git
synced 2025-04-24 19:39:11 +00:00
* feat: add skeleton masthead and generic loading page * add generic loader to picker page * smooth out spinner, add no-motion state
78 lines
1.8 KiB
TypeScript
78 lines
1.8 KiB
TypeScript
import styled, { css, keyframes } from 'styled-components';
|
|
import { palette } from '../colors';
|
|
|
|
type SpinnerProps = {
|
|
size?: number;
|
|
reverse?: boolean;
|
|
color?: string;
|
|
speed?: number;
|
|
};
|
|
|
|
const spinnerKeyframes = keyframes`
|
|
0%, 100% {
|
|
border-width: 0.5px;
|
|
border-right-width: 3px;
|
|
border-left-width: 0;
|
|
}
|
|
25% {
|
|
border-width: 0.5px;
|
|
border-top-width: 3px;
|
|
border-bottom-width: 0;
|
|
}
|
|
50% {
|
|
border-width: 0.5px;
|
|
border-left-width: 3px;
|
|
border-right-width: 0;
|
|
}
|
|
75% {
|
|
border-width: 0.5px;
|
|
border-bottom-width: 3px;
|
|
border-top-width: 0;
|
|
}
|
|
`;
|
|
|
|
const SpinnerStyled = styled.div<Required<SpinnerProps>>`
|
|
@media (prefers-reduced-motion: no-preference) {
|
|
animation: ${spinnerKeyframes} ${(props) => props.speed}s linear infinite
|
|
${(props) => (props.reverse ? `reverse` : '')};
|
|
transform: rotateZ(0);
|
|
}
|
|
|
|
border: 0.5px solid ${(props) => props.color};
|
|
box-sizing: border-box;
|
|
width: ${(props) => props.size}px;
|
|
height: ${(props) => props.size}px;
|
|
border-radius: ${(props) => props.size * 0.7}px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
|
|
${(props) =>
|
|
props.reverse
|
|
? css`
|
|
border-right-width: 0;
|
|
border-left-width: 3px;
|
|
`
|
|
: css`
|
|
border-right-width: 3px;
|
|
border-left-width: 0;
|
|
`}
|
|
`;
|
|
|
|
export const Spinner = (
|
|
props: SpinnerProps & { innerColor?: string; outerColor?: string }
|
|
) => (
|
|
<SpinnerStyled
|
|
size={props.size || 50}
|
|
color={props.outerColor || palette.taupe400}
|
|
reverse={!!props.reverse}
|
|
speed={props.speed || 1}
|
|
>
|
|
<SpinnerStyled
|
|
size={(props.size || 50) * 0.75}
|
|
color={props.innerColor || palette.taupe100}
|
|
reverse={!props.reverse}
|
|
speed={(props.speed || 1) * 1.25}
|
|
/>
|
|
</SpinnerStyled>
|
|
);
|