diff --git a/lib/components/ScratchOffCard/index.tsx b/lib/components/ScratchOffCard/index.tsx index 7212bd6d..c8247c5a 100644 --- a/lib/components/ScratchOffCard/index.tsx +++ b/lib/components/ScratchOffCard/index.tsx @@ -1,6 +1,5 @@ import React, { useEffect, useRef, useState, memo } from 'react'; import styled from 'styled-components'; -import brushImg from './brush.jpg'; import { getAngleBetween, getDistanceBetween, @@ -9,6 +8,8 @@ import { } from './utils'; const DEFAULT_REVEAL_PERCENTAGE = 50; +const BRUSH_IMG = + ''; const StyledScratchOffCard = styled.div<{ width: number; height: number }>` position: relative; @@ -39,7 +40,7 @@ const StyledCoverImg = styled.img` visibility: hidden; `; -interface Props { +export interface ScratchOffCardProps { revealPercentage?: number; width: number; height: number; @@ -48,7 +49,7 @@ interface Props { handleReveal: () => void; } -const ScratchOffCard: React.FC = ({ +export const ScratchOffCard: React.FC = ({ revealPercentage = DEFAULT_REVEAL_PERCENTAGE, width, height, @@ -60,27 +61,30 @@ const ScratchOffCard: React.FC = ({ const canvasRef = useRef(null); const [isCoverImageReady, setIsCoverImageReady] = useState(false); - const handleCoverImgOnLoad = () => { - canvasRef.current - ?.getContext('2d') - .drawImage(coverImgRef.current, 0, 0, width, height); - setIsCoverImageReady(true); - }; + useEffect(() => { + coverImgRef.current!.src = coverImgSrc; + coverImgRef.current!.onload = () => { + setIsCoverImageReady(true); + }; + }, [coverImgSrc]); useEffect(() => { + if (!isCoverImageReady) { + return; + } + let isDrawing: boolean; let lastPoint: { x: number; y: number }; const canvas = canvasRef.current; - const ctx = canvasRef.current?.getContext('2d') as CanvasRenderingContext2D; - const image = coverImgRef.current as HTMLImageElement; + const ctx = canvas?.getContext('2d'); const brush = new Image(); - brush.src = brushImg; - ctx.drawImage(image, 0, 0, width, height); + brush.src = BRUSH_IMG; + ctx!.drawImage(coverImgRef.current!, 0, 0, width, height); const handleMouseDown = (e: MouseEvent | TouchEvent) => { isDrawing = true; - lastPoint = getMouse(e, canvasRef.current); + lastPoint = getMouse(e, canvas); }; const handleMouseMove = (e: MouseEvent | TouchEvent) => { @@ -90,7 +94,7 @@ const ScratchOffCard: React.FC = ({ e.preventDefault(); - const currentPoint = getMouse(e, canvasRef.current); + const currentPoint = getMouse(e, canvas); const dist = getDistanceBetween(lastPoint, currentPoint); const angle = getAngleBetween(lastPoint, currentPoint); let x; @@ -99,19 +103,16 @@ const ScratchOffCard: React.FC = ({ for (let i = 0; i < dist; i += 1) { x = lastPoint.x + Math.sin(angle) * i - 25; y = lastPoint.y + Math.cos(angle) * i - 25; - ctx.globalCompositeOperation = 'destination-out'; - ctx.drawImage(brush, x, y); + ctx!.globalCompositeOperation = 'destination-out'; + ctx!.drawImage(brush, x, y); } lastPoint = currentPoint; - const currentPercentage = getFilledInPixels(32, ctx, width, height); + const currentPercentage = getFilledInPixels(32, ctx!, width, height); - if ( - currentPercentage > revealPercentage && - canvasRef.current?.parentNode - ) { + if (currentPercentage > revealPercentage && canvas?.parentNode) { handleReveal(); - canvasRef.current?.parentNode.removeChild(canvasRef.current); + canvas?.parentNode.removeChild(canvas); } }; @@ -134,7 +135,7 @@ const ScratchOffCard: React.FC = ({ canvas?.removeEventListener('mouseup', handleMouseUp, false); canvas?.removeEventListener('touchend', handleMouseUp, false); }; - }, [handleReveal, revealPercentage, height, width]); + }, [handleReveal, revealPercentage, height, width, isCoverImageReady]); return ( @@ -142,15 +143,9 @@ const ScratchOffCard: React.FC = ({ {children} - + ); }; -export default memo(ScratchOffCard); +export default ScratchOffCard; diff --git a/lib/components/TransitionLeaderboardWrapper.tsx b/lib/components/TransitionLeaderboardWrapper.tsx index c8f17cf9..44db5312 100755 --- a/lib/components/TransitionLeaderboardWrapper.tsx +++ b/lib/components/TransitionLeaderboardWrapper.tsx @@ -1,12 +1,13 @@ -import React from 'react'; +import React, { memo } from 'react'; import styled from 'styled-components'; import useItemTransition, { ItemStyle } from '../hooks/useItemTransition'; import { User } from '../types'; -export interface Props { +export interface TransitionLeaderboardWrapperProps { user: User[]; rowCount: number; itemStyle: ItemStyle; + children: string | React.ReactNode; } const Wrapper = styled.div` @@ -19,7 +20,7 @@ const transitionStyle = { transition: 'all 0.5s ease 0.3s', }; -export const TransitionLeaderboardWrapper: React.FC = React.memo( +export const TransitionLeaderboardWrapper: React.FC = ({ user, itemStyle, rowCount, children }) => { const { itemTransitionStyle } = useItemTransition( itemStyle, @@ -45,7 +46,6 @@ export const TransitionLeaderboardWrapper: React.FC = React.memo( } return {renderChild()}; - }, -); + }; -export default TransitionLeaderboardWrapper; +export default memo(TransitionLeaderboardWrapper); diff --git a/lib/components/VirtualizedList.tsx b/lib/components/VirtualizedList.tsx index 89146fdc..a45fd78f 100644 --- a/lib/components/VirtualizedList.tsx +++ b/lib/components/VirtualizedList.tsx @@ -1,5 +1,5 @@ /* eslint-disable react/no-children-prop */ -import React, { useEffect, useRef, useCallback, ReactNode } from 'react'; +import React, { useEffect, useRef, useCallback, ReactNode, memo } from 'react'; import { VariableSizeList } from 'react-window'; import styled from 'styled-components'; @@ -57,7 +57,7 @@ export interface IVirtualizedListProps { * 在 url 上加上 streamerUserID= 直播主的id,就會將畫面移動到直播主的位置 * @param IVirtualizedListProps 使用方法參考 IVirtualizedListProps 說明 */ -export const VirtualizedList: React.FC = ({ +const VirtualizedList: React.FC = ({ dataset, children, itemHeight = 80, @@ -142,4 +142,4 @@ export const VirtualizedList: React.FC = ({ ); }; -export default VirtualizedList; +export default memo(VirtualizedList); diff --git a/lib/components/index.ts b/lib/components/index.ts index 643367f1..e08ca94a 100755 --- a/lib/components/index.ts +++ b/lib/components/index.ts @@ -1,7 +1,13 @@ -import { TransitionLeaderboardWrapper } from './TransitionLeaderboardWrapper'; -import { VirtualizedList } from './VirtualizedList'; +import TransitionLeaderboardWrapper from './TransitionLeaderboardWrapper'; +import VirtualizedList from './VirtualizedList'; +import ScratchOffCard from './ScratchOffCard'; export * from './TransitionLeaderboardWrapper'; export * from './VirtualizedList'; +export * from './ScratchOffCard'; -export default { TransitionLeaderboardWrapper, VirtualizedList }; +export default { + TransitionLeaderboardWrapper, + VirtualizedList, + ScratchOffCard, +}; diff --git a/lib/template/offlineNormal.tsx b/lib/template/offlineNormal.tsx index fbaef26b..c74ab27b 100644 --- a/lib/template/offlineNormal.tsx +++ b/lib/template/offlineNormal.tsx @@ -1,7 +1,7 @@ import React from 'react'; import styled from 'styled-components'; import usePageData from '../hooks/usePageData'; -import { TransitionLeaderboardWrapper } from '../components/TransitionLeaderboardWrapper'; +import TransitionLeaderboardWrapper from '../components/TransitionLeaderboardWrapper'; import { ItemStyle } from '../hooks/useItemTransition'; import { qs } from '../utils'; diff --git a/playground/ScrollToStreamer.tsx b/playground/ScrollToStreamer.tsx index 69d3dbc3..6da4fb2d 100755 --- a/playground/ScrollToStreamer.tsx +++ b/playground/ScrollToStreamer.tsx @@ -1,6 +1,6 @@ import React from 'react'; import styled from 'styled-components'; -import { VirtualizedList } from '../lib/components/VirtualizedList'; +import VirtualizedList from '../lib/components/VirtualizedList'; import fakeLeaderBoardData from './fakeLeaderBoardData.json'; const StyledSpace = styled.div`