From 9e20c1322cdc79525ac8041e31fccda822b2f892 Mon Sep 17 00:00:00 2001 From: Gianni Ferullo Date: Mon, 22 Apr 2024 08:05:57 -0400 Subject: [PATCH 1/3] react-components: support external state for grid allowing for easier data mutations --- .../react-components/src/components/grid.tsx | 15 ++- .../stories/grid-editable.stories.tsx | 117 ++++++++++++++++++ 2 files changed, 129 insertions(+), 3 deletions(-) create mode 100644 packages/react-components/stories/grid-editable.stories.tsx diff --git a/packages/react-components/src/components/grid.tsx b/packages/react-components/src/components/grid.tsx index eba65d06..403d8ff4 100644 --- a/packages/react-components/src/components/grid.tsx +++ b/packages/react-components/src/components/grid.tsx @@ -28,6 +28,7 @@ type Props = { noLink?: boolean noResultsMessage?: string | JSX.Element initialGifs?: IGif[] + externalGifs?: IGif[] useTransform?: boolean columnOffsets?: number[] backgroundColor?: string @@ -73,7 +74,7 @@ class Grid extends PureComponent { unmounted: boolean = false paginator = gifPaginator(this.props.fetchGifs, this.state.gifs) static getDerivedStateFromProps: GetDerivedStateFromProps = ( - { columns, gutter, width }: Props, + { columns, gutter, width, externalGifs }: Props, prevState: State ) => { const gutterOffset = gutter * (columns - 1) @@ -81,6 +82,9 @@ class Grid extends PureComponent { if (prevState.gifWidth !== gifWidth) { return { gifWidth } } + if (externalGifs && externalGifs !== prevState.gifs) { + return { gifs: externalGifs } + } return null } @@ -100,8 +104,13 @@ class Grid extends PureComponent { onFetch = debounce(Grid.fetchDebounce, async () => { if (this.unmounted) return - const { isFetching, isLoaderVisible, gifs } = this.state - const prefetchCount = gifs.length + const { isFetching, isLoaderVisible } = this.state + const { externalGifs, fetchGifs } = this.props + const prefetchCount = (externalGifs || this.state.gifs).length + if (externalGifs) { + // reinitialize the paginator every fetch with the new external gifs + this.paginator = gifPaginator(fetchGifs, externalGifs) + } if (!isFetching && isLoaderVisible) { this.setState({ isFetching: true, isError: false }) let gifs diff --git a/packages/react-components/stories/grid-editable.stories.tsx b/packages/react-components/stories/grid-editable.stories.tsx new file mode 100644 index 00000000..f79ef891 --- /dev/null +++ b/packages/react-components/stories/grid-editable.stories.tsx @@ -0,0 +1,117 @@ +import styled from '@emotion/styled' +import { GiphyFetch } from '@giphy/js-fetch-api' +import { Meta, StoryObj } from '@storybook/react' +import fetchMock from 'fetch-mock' +import React, { useEffect, useState } from 'react' +import { throttle } from 'throttle-debounce' +import { GifOverlayProps, Grid as GridComponent } from '../src' +import inTestsRunner from './in-tests-runner' +import mockGifsResult from './mock-data/gifs.json' +import { IGif } from '@giphy/js-types' +import { giphyPurple } from '@giphy/colors' + +const apiKey = 'sXpGFDGZs0Dv1mmNFvYaGUvYwKX0PWIh' +const gf = new GiphyFetch(apiKey) + +const OverlayContainer = styled.div` + position: absolute; + left: 0; + top: 0; + right: 0; + bottom: 0; + display: flex; + color: white; + justify-content: center; + align-items: center; +` +const ButtonContainer = styled.div` + display: flex; + flex-direction: column; + gap: 2px; + align-items: center; + h4 { + text-align: center; + } +` +const Button = styled.div` + cursor: pointer; + background: ${giphyPurple}; + padding: 6px; + border-radius: 4px; + max-width: fit-content; +` + +type GridProps = Partial> + +const Grid = ({ loader, ...other }: GridProps) => { + const [width, setWidth] = useState(innerWidth - 24) + const [externalGifs, setExternalGifs] = useState() + const Overlay = ({ gif, isHovered }: GifOverlayProps) => ( + + {isHovered && ( + +

{gif.title}

+ + +
+ )} +
+ ) + const onResize = throttle(500, () => setWidth(innerWidth - 24)) + useEffect(() => { + window.addEventListener('resize', onResize, false) + return () => window.removeEventListener('resize', onResize, false) + // eslint-disable-next-line react-hooks/exhaustive-deps + }, []) + + const fetchGifs = async (offset: number) => { + if (inTestsRunner()) { + fetchMock.restore().getOnce(`begin:https://api.giphy.com/v1/gifs/search`, { body: mockGifsResult }) + } + const result = await gf.trending({ offset, limit: 10 }) + fetchMock.restore() + setExternalGifs(result.data) + return result + } + return ( + setExternalGifs(gifs)} + {...other} + /> + ) +} + +const meta: Meta = { + component: Grid, + title: 'React Components/Grid/Editable', +} + +export default meta + +type Story = StoryObj + +export const EditableGrid: Story = {} From ccf0ecbe7a9d0416ccfa7985f3cabfce912b0e0f Mon Sep 17 00:00:00 2001 From: Gianni Ferullo Date: Mon, 22 Apr 2024 15:34:01 -0400 Subject: [PATCH 2/3] react-components: gif make fetchpriority lowercase to avoid react warning --- packages/react-components/src/components/gif.tsx | 2 +- .../stories/grid-editable.stories.tsx | 14 ++++++++------ 2 files changed, 9 insertions(+), 7 deletions(-) diff --git a/packages/react-components/src/components/gif.tsx b/packages/react-components/src/components/gif.tsx index 6f2aa13d..3dcc42ea 100644 --- a/packages/react-components/src/components/gif.tsx +++ b/packages/react-components/src/components/gif.tsx @@ -277,7 +277,7 @@ const Gif = ({ suppressHydrationWarning // @ts-ignore - fetchPriority is not recognized by React typescript // eslint-disable-next-line react/no-unknown-property - fetchPriority={fetchPriority} + fetchpriority={fetchPriority} className={[Gif.imgClassName, loadedClassname].join(' ')} src={shouldShowMedia ? rendition.url : placeholder} style={{ background }} diff --git a/packages/react-components/stories/grid-editable.stories.tsx b/packages/react-components/stories/grid-editable.stories.tsx index f79ef891..f66c8977 100644 --- a/packages/react-components/stories/grid-editable.stories.tsx +++ b/packages/react-components/stories/grid-editable.stories.tsx @@ -1,5 +1,7 @@ import styled from '@emotion/styled' +import { giphyPurple } from '@giphy/colors' import { GiphyFetch } from '@giphy/js-fetch-api' +import { IGif } from '@giphy/js-types' import { Meta, StoryObj } from '@storybook/react' import fetchMock from 'fetch-mock' import React, { useEffect, useState } from 'react' @@ -7,8 +9,6 @@ import { throttle } from 'throttle-debounce' import { GifOverlayProps, Grid as GridComponent } from '../src' import inTestsRunner from './in-tests-runner' import mockGifsResult from './mock-data/gifs.json' -import { IGif } from '@giphy/js-types' -import { giphyPurple } from '@giphy/colors' const apiKey = 'sXpGFDGZs0Dv1mmNFvYaGUvYwKX0PWIh' const gf = new GiphyFetch(apiKey) @@ -61,10 +61,12 @@ const Grid = ({ loader, ...other }: GridProps) => {