v3/packages/design-system/atoms/spinner/Spinner.tsx
Katalina e0fcfc310e
feat: add skeleton masthead and generic loading page (#182)
* feat: add skeleton masthead and generic loading page

* add generic loader to picker page

* smooth out spinner, add no-motion state
2021-03-15 19:51:56 -04:00

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>
);