You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
189 lines
4.7 KiB
TypeScript
189 lines
4.7 KiB
TypeScript
import React from 'react';
|
|
|
|
import { icons, SessionIconSize, SessionIconType } from '../icon';
|
|
import styled, { css, DefaultTheme, keyframes } from 'styled-components';
|
|
import { drop } from 'lodash';
|
|
|
|
export type SessionIconProps = {
|
|
iconType: SessionIconType;
|
|
iconSize: SessionIconSize | number;
|
|
iconColor?: string;
|
|
iconRotation?: number;
|
|
rotateDuration?: number;
|
|
glowDuration?: number;
|
|
borderRadius?: number;
|
|
glowStartDelay?: number;
|
|
theme: DefaultTheme;
|
|
};
|
|
|
|
const getIconDimensionFromIconSize = (iconSize: SessionIconSize | number) => {
|
|
if (typeof iconSize === 'number') {
|
|
return iconSize;
|
|
} else {
|
|
switch (iconSize) {
|
|
case SessionIconSize.Tiny:
|
|
return 12;
|
|
case SessionIconSize.Small:
|
|
return 15;
|
|
case SessionIconSize.Medium:
|
|
return 20;
|
|
case SessionIconSize.Large:
|
|
return 25;
|
|
case SessionIconSize.Huge:
|
|
return 30;
|
|
case SessionIconSize.Max:
|
|
return 80;
|
|
default:
|
|
return 20;
|
|
}
|
|
}
|
|
};
|
|
|
|
type StyledSvgProps = {
|
|
width: string | number;
|
|
height: string | number;
|
|
iconRotation: number;
|
|
rotateDuration?: number;
|
|
borderRadius?: number;
|
|
glowDuration?: number;
|
|
glowStartDelay?: number;
|
|
iconColor?: string;
|
|
};
|
|
|
|
const rotate = keyframes`
|
|
from {
|
|
transform: rotate(0deg);
|
|
}
|
|
to {
|
|
transform: rotate(360deg);
|
|
}
|
|
`;
|
|
|
|
/**
|
|
* Creates a glow animation made for multiple element sequentially
|
|
* @param color
|
|
* @param glowDuration
|
|
* @param glowStartDelay
|
|
* @returns
|
|
*/
|
|
const glow = (color: string, glowDuration: number, glowStartDelay: number) => {
|
|
let dropShadowType = `drop-shadow(0px 0px 6px ${color}) `;
|
|
//increase shadow intensity by 3
|
|
let dropShadow = `${dropShadowType.repeat(2)};`;
|
|
|
|
// TODO: Decrease dropshadow for last frame
|
|
// creating keyframe for sequential animations
|
|
let kf = '';
|
|
for (let i = 0; i <= glowDuration; i++) {
|
|
// const percent = (100 / glowDuration) * i;
|
|
const percent = (100 / glowDuration) * i;
|
|
if (i === glowStartDelay) {
|
|
kf += `${percent}% {
|
|
filter: ${dropShadow}
|
|
}`;
|
|
} else {
|
|
kf += `${percent}% {
|
|
filter: none;
|
|
}`;
|
|
}
|
|
}
|
|
return keyframes`${kf}`;
|
|
};
|
|
|
|
const animation = (props: any) => {
|
|
if (props.rotateDuration) {
|
|
return css`
|
|
${rotate} ${props.rotateDuration}s infinite linear;
|
|
`;
|
|
} else if (props.glowDuration !== undefined && props.glowStartDelay !== undefined) {
|
|
return css`
|
|
${glow(props.iconColor, props.glowDuration, props.glowStartDelay)} ${2}s ease-in infinite;
|
|
`;
|
|
} else {
|
|
return;
|
|
}
|
|
};
|
|
|
|
// ${glow(props.iconColor, props.glowDuration, props.glowStartDelay)} ${props.glowDuration}s ease-in ${ props.glowStartDelay }s infinite alternate;
|
|
|
|
//tslint:disable no-unnecessary-callback-wrapper
|
|
const Svg = styled.svg<StyledSvgProps>`
|
|
width: ${props => props.width};
|
|
animation: ${props => animation(props)};
|
|
transform: ${props => `rotate(${props.iconRotation}deg)`};
|
|
border-radius: ${props => props.borderRadius};
|
|
`;
|
|
//tslint:enable no-unnecessary-callback-wrapper
|
|
|
|
const SessionSvg = (props: {
|
|
viewBox: string;
|
|
path: string | Array<string>;
|
|
width: string | number;
|
|
height: string | number;
|
|
iconRotation: number;
|
|
iconColor?: string;
|
|
rotateDuration?: number;
|
|
glowDuration?: number;
|
|
glowStartDelay?: number;
|
|
borderRadius?: number;
|
|
theme: DefaultTheme;
|
|
}) => {
|
|
const colorSvg = props.iconColor || props?.theme?.colors.textColor;
|
|
const pathArray = props.path instanceof Array ? props.path : [props.path];
|
|
|
|
return (
|
|
<Svg {...props}>
|
|
{/* { props.glowDuration ?
|
|
<defs>
|
|
<filter>
|
|
<feDropShadow dx="0.2" dy="0.4" stdDeviation="0.2" />
|
|
</filter>
|
|
</defs>
|
|
:
|
|
null
|
|
} */}
|
|
{pathArray.map((path, index) => {
|
|
return <path key={index} fill={colorSvg} d={path} />;
|
|
})}
|
|
</Svg>
|
|
);
|
|
};
|
|
|
|
export const SessionIcon = (props: SessionIconProps) => {
|
|
const {
|
|
iconType,
|
|
iconColor,
|
|
theme,
|
|
rotateDuration,
|
|
glowDuration,
|
|
borderRadius,
|
|
glowStartDelay,
|
|
} = props;
|
|
let { iconSize, iconRotation } = props;
|
|
iconSize = iconSize || SessionIconSize.Medium;
|
|
iconRotation = iconRotation || 0;
|
|
|
|
const iconDimensions = getIconDimensionFromIconSize(iconSize);
|
|
const iconDef = icons[iconType];
|
|
const ratio = iconDef?.ratio || 1;
|
|
if (!theme) {
|
|
window.log.error('Missing theme props in SessionIcon');
|
|
}
|
|
|
|
return (
|
|
<SessionSvg
|
|
viewBox={iconDef.viewBox}
|
|
path={iconDef.path}
|
|
width={iconDimensions * ratio}
|
|
height={iconDimensions}
|
|
rotateDuration={rotateDuration}
|
|
glowDuration={glowDuration}
|
|
glowStartDelay={glowStartDelay}
|
|
borderRadius={borderRadius}
|
|
iconRotation={iconRotation}
|
|
iconColor={iconColor}
|
|
theme={theme}
|
|
/>
|
|
);
|
|
};
|