fix: first load flicker on qr code

pull/3083/head
William Grant 10 months ago
parent 5e3742e281
commit b102a0675f

@ -1,9 +1,19 @@
import { MouseEvent, useEffect, useRef, useState } from 'react'; import { MouseEvent, useEffect, useRef, useState } from 'react';
import { QRCode } from 'react-qrcode-logo'; import { QRCode } from 'react-qrcode-logo';
import { CSSProperties } from 'styled-components'; import styled, { CSSProperties } from 'styled-components';
import { COLORS } from '../themes/constants/colors'; import { THEME_GLOBALS } from '../themes/globals';
import { saveQRCode } from '../util/saveQRCode'; import { saveQRCode } from '../util/saveQRCode';
import { Flex } from './basic/Flex'; import { AnimatedFlex } from './basic/Flex';
// AnimatedFlex because we fade in the QR code a flicker on first render
const StyledQRView = styled(AnimatedFlex)<{
size: number;
}>`
cursor: pointer;
border-radius: 10px;
overflow: hidden;
${props => props.size && `width: ${props.size}px; height: ${props.size}px;`}
`;
export type SessionQRCodeProps = { export type SessionQRCodeProps = {
id: string; id: string;
@ -25,8 +35,8 @@ export function SessionQRCode(props: SessionQRCodeProps) {
id, id,
value, value,
size, size,
backgroundColor = COLORS.WHITE, backgroundColor,
foregroundColor = COLORS.BLACK, foregroundColor,
hasLogo, hasLogo,
logoImage, logoImage,
logoSize, logoSize,
@ -45,32 +55,32 @@ export function SessionQRCode(props: SessionQRCodeProps) {
useEffect(() => { useEffect(() => {
// Don't pass the component props to the QR component directly instead update it's props in the next render cycle to prevent janky renders // Don't pass the component props to the QR component directly instead update it's props in the next render cycle to prevent janky renders
if (!loading && hasLogo && logo !== logoImage) { if (loading) {
return;
}
if (bgColor !== backgroundColor) {
setBgColor(backgroundColor); setBgColor(backgroundColor);
}
if (fgColor !== foregroundColor) {
setFgColor(foregroundColor); setFgColor(foregroundColor);
}
if (hasLogo && logo !== logoImage) {
setLogo(logoImage); setLogo(logoImage);
} }
}, [backgroundColor, foregroundColor, hasLogo, loading, logo, logoImage]); }, [backgroundColor, bgColor, fgColor, foregroundColor, hasLogo, loading, logo, logoImage]);
return ( return (
<Flex <StyledQRView
container={true} container={true}
justifyContent="center" justifyContent="center"
alignItems="center" alignItems="center"
width={`${size}px`} size={size}
height={`${size}px`}
id={id} id={id}
aria-label={ariaLabel || 'QR code'} aria-label={ariaLabel || 'QR code'}
title={window.i18n('clickToTrustContact')} title={window.i18n('clickToTrustContact')}
// onClick={(event: MouseEvent<HTMLDivElement>) => {
// event.preventDefault();
// const fileName = `${id}-${new Date().toISOString()}.png`;
// try {
// qrRef.current?.download('png', fileName);
// } catch (e) {
// window.log.error(`Error downloading QR code: ${fileName}\n${e}`);
// }
// }}
onClick={(event: MouseEvent<HTMLDivElement>) => { onClick={(event: MouseEvent<HTMLDivElement>) => {
event.preventDefault(); event.preventDefault();
void saveQRCode(id, { void saveQRCode(id, {
@ -80,6 +90,9 @@ export function SessionQRCode(props: SessionQRCodeProps) {
}); });
}} }}
data-testId={dataTestId || 'session-qr-code'} data-testId={dataTestId || 'session-qr-code'}
initial={{ opacity: 0 }}
animate={{ opacity: loading ? 0 : 1 }}
transition={{ duration: THEME_GLOBALS['--default-duration-seconds'] }}
style={style} style={style}
> >
<QRCode <QRCode
@ -96,13 +109,10 @@ export function SessionQRCode(props: SessionQRCodeProps) {
logoHeight={canvasLogoSize} logoHeight={canvasLogoSize}
removeQrCodeBehindLogo={true} removeQrCodeBehindLogo={true}
style={{ style={{
borderRadius: '10px',
cursor: 'pointer',
overflow: 'hidden',
width: size, width: size,
height: size, height: size,
}} }}
/> />
</Flex> </StyledQRView>
); );
} }

Loading…
Cancel
Save