feat: implement Suspense and Await for improved data loading in carousel components
This commit is contained in:
parent
3847fd1896
commit
ab3f748195
@ -1,15 +1,16 @@
|
|||||||
import useEmblaCarousel from 'embla-carousel-react'
|
import useEmblaCarousel from 'embla-carousel-react'
|
||||||
import { useCallback, useEffect, useState } from 'react'
|
import { Suspense, useCallback, useEffect, useState } from 'react'
|
||||||
import { useRouteLoaderData } from 'react-router'
|
import { Await, useRouteLoaderData } from 'react-router'
|
||||||
import { stripHtml } from 'string-strip-html'
|
import { stripHtml } from 'string-strip-html'
|
||||||
|
|
||||||
import { Button } from '~/components/ui/button'
|
|
||||||
import { CarouselButton } from '~/components/ui/button-slide'
|
import { CarouselButton } from '~/components/ui/button-slide'
|
||||||
import { useNewsContext } from '~/contexts/news'
|
import { useNewsContext } from '~/contexts/news'
|
||||||
import type { loader } from '~/routes/_news'
|
import type { loader } from '~/routes/_news'
|
||||||
import type { TNews } from '~/types/news'
|
import type { TNews } from '~/types/news'
|
||||||
import { getPremiumAttribute } from '~/utils/render'
|
import { getPremiumAttribute } from '~/utils/render'
|
||||||
|
|
||||||
|
import { Button } from './button'
|
||||||
|
|
||||||
export const CarouselHero = (properties: TNews) => {
|
export const CarouselHero = (properties: TNews) => {
|
||||||
const { setIsSuccessOpen } = useNewsContext()
|
const { setIsSuccessOpen } = useNewsContext()
|
||||||
const loaderData = useRouteLoaderData<typeof loader>('routes/_news')
|
const loaderData = useRouteLoaderData<typeof loader>('routes/_news')
|
||||||
@ -72,8 +73,14 @@ export const CarouselHero = (properties: TNews) => {
|
|||||||
ref={emblaReference}
|
ref={emblaReference}
|
||||||
>
|
>
|
||||||
<div className="embla__container hero flex sm:gap-x-8">
|
<div className="embla__container hero flex sm:gap-x-8">
|
||||||
{items.map(
|
<Suspense fallback={<div>Loading...</div>}>
|
||||||
({ featured_image, title, content, slug, is_premium }, index) => (
|
<Await resolve={items}>
|
||||||
|
{(value) =>
|
||||||
|
value.data.map(
|
||||||
|
(
|
||||||
|
{ featured_image, title, content, slug, is_premium },
|
||||||
|
index,
|
||||||
|
) => (
|
||||||
<div
|
<div
|
||||||
className="embla__slide hero w-full min-w-0 flex-none"
|
className="embla__slide hero w-full min-w-0 flex-none"
|
||||||
key={index}
|
key={index}
|
||||||
@ -108,7 +115,10 @@ export const CarouselHero = (properties: TNews) => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
),
|
),
|
||||||
)}
|
)
|
||||||
|
}
|
||||||
|
</Await>
|
||||||
|
</Suspense>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -1,15 +1,15 @@
|
|||||||
import useEmblaCarousel from 'embla-carousel-react'
|
import useEmblaCarousel from 'embla-carousel-react'
|
||||||
import { useCallback, useEffect, useState } from 'react'
|
import { Suspense, useCallback, useEffect, useState } from 'react'
|
||||||
import { useRouteLoaderData } from 'react-router'
|
import { Await, useRouteLoaderData } from 'react-router'
|
||||||
import { stripHtml } from 'string-strip-html'
|
import { stripHtml } from 'string-strip-html'
|
||||||
|
|
||||||
import { Button } from '~/components/ui/button'
|
|
||||||
import { CarouselButton } from '~/components/ui/button-slide'
|
import { CarouselButton } from '~/components/ui/button-slide'
|
||||||
import { useNewsContext } from '~/contexts/news'
|
import { useNewsContext } from '~/contexts/news'
|
||||||
import type { loader } from '~/routes/_news'
|
import type { loader } from '~/routes/_news'
|
||||||
import type { TNews } from '~/types/news'
|
import type { TNews } from '~/types/news'
|
||||||
import { getPremiumAttribute } from '~/utils/render'
|
import { getPremiumAttribute } from '~/utils/render'
|
||||||
|
|
||||||
|
import { Button } from './button'
|
||||||
import { Tags } from './tags'
|
import { Tags } from './tags'
|
||||||
|
|
||||||
export const CarouselSection = (properties: TNews) => {
|
export const CarouselSection = (properties: TNews) => {
|
||||||
@ -79,7 +79,10 @@ export const CarouselSection = (properties: TNews) => {
|
|||||||
ref={emblaReference}
|
ref={emblaReference}
|
||||||
>
|
>
|
||||||
<div className="embla__container col-span-3 flex max-h-[586px] sm:gap-x-8">
|
<div className="embla__container col-span-3 flex max-h-[586px] sm:gap-x-8">
|
||||||
{items.map(
|
<Suspense fallback={<div>Loading...</div>}>
|
||||||
|
<Await resolve={items}>
|
||||||
|
{(value) =>
|
||||||
|
value.data.map(
|
||||||
(
|
(
|
||||||
{ featured_image, title, content, tags, slug, is_premium },
|
{ featured_image, title, content, tags, slug, is_premium },
|
||||||
index,
|
index,
|
||||||
@ -122,7 +125,10 @@ export const CarouselSection = (properties: TNews) => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
),
|
),
|
||||||
)}
|
)
|
||||||
|
}
|
||||||
|
</Await>
|
||||||
|
</Suspense>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -12,17 +12,17 @@ export const NewsPage = () => {
|
|||||||
const spotlight: TNews = {
|
const spotlight: TNews = {
|
||||||
title: loaderData?.spotlightCategory?.name || '',
|
title: loaderData?.spotlightCategory?.name || '',
|
||||||
description: loaderData?.spotlightCategory?.description || '',
|
description: loaderData?.spotlightCategory?.description || '',
|
||||||
items: loaderData?.spotlightNews || [],
|
items: loaderData?.spotlightData || Promise.resolve({ data: [] }),
|
||||||
}
|
}
|
||||||
const berita: TNews = {
|
const berita: TNews = {
|
||||||
title: loaderData?.beritaCategory?.name || '',
|
title: loaderData?.beritaCategory?.name || '',
|
||||||
description: loaderData?.beritaCategory?.description || '',
|
description: loaderData?.beritaCategory?.description || '',
|
||||||
items: loaderData?.beritaNews || [],
|
items: loaderData?.beritaData || Promise.resolve({ data: [] }),
|
||||||
}
|
}
|
||||||
const kajian: TNews = {
|
const kajian: TNews = {
|
||||||
title: loaderData?.kajianCategory?.name || '',
|
title: loaderData?.kajianCategory?.name || '',
|
||||||
description: loaderData?.kajianCategory?.description || '',
|
description: loaderData?.kajianCategory?.description || '',
|
||||||
items: loaderData?.kajianNews || [],
|
items: loaderData?.kajianData || Promise.resolve({ data: [] }),
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@ -25,31 +25,22 @@ export const loader = async ({}: Route.LoaderArgs) => {
|
|||||||
(category) => category.code === kajianCode,
|
(category) => category.code === kajianCode,
|
||||||
)
|
)
|
||||||
|
|
||||||
let { data: spotlightNews } = await getNews({
|
const spotlightData = getNews({
|
||||||
categories: [spotlightCode],
|
categories: [spotlightCode],
|
||||||
})
|
})
|
||||||
spotlightNews = spotlightNews.filter(
|
const beritaData = getNews({
|
||||||
(news) => new Date(news.live_at) <= new Date(),
|
|
||||||
)
|
|
||||||
let { data: beritaNews } = await getNews({
|
|
||||||
categories: [beritaCode],
|
categories: [beritaCode],
|
||||||
})
|
})
|
||||||
beritaNews = beritaNews.filter((news) => new Date(news.live_at) <= new Date())
|
const kajianData = getNews({
|
||||||
let { data: kajianNews } = await getNews({
|
|
||||||
categories: [kajianCode],
|
categories: [kajianCode],
|
||||||
})
|
})
|
||||||
kajianNews = kajianNews.filter((news) => new Date(news.live_at) <= new Date())
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
spotlightCategory,
|
spotlightCategory,
|
||||||
spotlightCode,
|
|
||||||
beritaCategory,
|
beritaCategory,
|
||||||
beritaCode,
|
|
||||||
kajianCategory,
|
kajianCategory,
|
||||||
kajianCode,
|
spotlightData,
|
||||||
spotlightNews,
|
beritaData,
|
||||||
beritaNews,
|
kajianData,
|
||||||
kajianNews,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -3,5 +3,5 @@ import type { TNewsResponse } from '~/apis/common/get-news'
|
|||||||
export type TNews = {
|
export type TNews = {
|
||||||
title: string
|
title: string
|
||||||
description: string
|
description: string
|
||||||
items: TNewsResponse[]
|
items: Promise<{ data: TNewsResponse[] }>
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user