mirror of
https://github.com/langgenius/dify.git
synced 2026-02-09 15:10:13 -05:00
Co-authored-by: Crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com>
188 lines
6.5 KiB
TypeScript
188 lines
6.5 KiB
TypeScript
/* eslint-disable react-hooks-extra/no-direct-set-state-in-use-effect */
|
|
import type { FC } from 'react'
|
|
import type { Banner } from '@/models/app'
|
|
import { RiArrowRightLine } from '@remixicon/react'
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { useCarousel } from '@/app/components/base/carousel'
|
|
import { cn } from '@/utils/classnames'
|
|
import { IndicatorButton } from './indicator-button'
|
|
|
|
type BannerItemProps = {
|
|
banner: Banner
|
|
autoplayDelay: number
|
|
isPaused?: boolean
|
|
}
|
|
|
|
const RESPONSIVE_BREAKPOINT = 1200
|
|
const MAX_RESPONSIVE_WIDTH = 600
|
|
const INDICATOR_WIDTH = 20
|
|
const INDICATOR_GAP = 8
|
|
const MIN_VIEW_MORE_WIDTH = 480
|
|
|
|
export const BannerItem: FC<BannerItemProps> = ({ banner, autoplayDelay, isPaused = false }) => {
|
|
const { t } = useTranslation()
|
|
const { api, selectedIndex } = useCarousel()
|
|
const { category, title, description, 'img-src': imgSrc } = banner.content
|
|
|
|
const [resetKey, setResetKey] = useState(0)
|
|
const textAreaRef = useRef<HTMLDivElement>(null)
|
|
const [maxWidth, setMaxWidth] = useState<number | undefined>(undefined)
|
|
|
|
const slideInfo = useMemo(() => {
|
|
const slides = api?.slideNodes() ?? []
|
|
const totalSlides = slides.length
|
|
const nextIndex = totalSlides > 0 ? (selectedIndex + 1) % totalSlides : 0
|
|
return { slides, totalSlides, nextIndex }
|
|
}, [api, selectedIndex])
|
|
|
|
const indicatorsWidth = useMemo(() => {
|
|
const count = slideInfo.totalSlides
|
|
if (count === 0)
|
|
return 0
|
|
// Calculate: indicator buttons + gaps + extra spacing (3 * 20px for divider and padding)
|
|
return (count + 2) * INDICATOR_WIDTH + (count - 1) * INDICATOR_GAP
|
|
}, [slideInfo.totalSlides])
|
|
|
|
const viewMoreStyle = useMemo(() => {
|
|
if (!maxWidth)
|
|
return undefined
|
|
return {
|
|
maxWidth: `${maxWidth}px`,
|
|
minWidth: indicatorsWidth ? `${Math.min(maxWidth - indicatorsWidth, MIN_VIEW_MORE_WIDTH)}px` : undefined,
|
|
}
|
|
}, [maxWidth, indicatorsWidth])
|
|
|
|
const responsiveStyle = useMemo(
|
|
() => (maxWidth !== undefined ? { maxWidth: `${maxWidth}px` } : undefined),
|
|
[maxWidth],
|
|
)
|
|
|
|
const incrementResetKey = useCallback(() => setResetKey(prev => prev + 1), [])
|
|
|
|
useEffect(() => {
|
|
const updateMaxWidth = () => {
|
|
if (window.innerWidth < RESPONSIVE_BREAKPOINT && textAreaRef.current) {
|
|
const textAreaWidth = textAreaRef.current.offsetWidth
|
|
setMaxWidth(Math.min(textAreaWidth, MAX_RESPONSIVE_WIDTH))
|
|
}
|
|
else {
|
|
setMaxWidth(undefined)
|
|
}
|
|
}
|
|
|
|
updateMaxWidth()
|
|
|
|
const resizeObserver = new ResizeObserver(updateMaxWidth)
|
|
if (textAreaRef.current)
|
|
resizeObserver.observe(textAreaRef.current)
|
|
|
|
window.addEventListener('resize', updateMaxWidth)
|
|
|
|
return () => {
|
|
resizeObserver.disconnect()
|
|
window.removeEventListener('resize', updateMaxWidth)
|
|
}
|
|
}, [])
|
|
|
|
useEffect(() => {
|
|
incrementResetKey()
|
|
}, [selectedIndex, incrementResetKey])
|
|
|
|
const handleBannerClick = useCallback(() => {
|
|
incrementResetKey()
|
|
if (banner.link)
|
|
window.open(banner.link, '_blank', 'noopener,noreferrer')
|
|
}, [banner.link, incrementResetKey])
|
|
|
|
const handleIndicatorClick = useCallback((index: number) => {
|
|
incrementResetKey()
|
|
api?.scrollTo(index)
|
|
}, [api, incrementResetKey])
|
|
|
|
return (
|
|
<div
|
|
className="relative flex w-full min-w-[784px] cursor-pointer overflow-hidden rounded-2xl bg-components-panel-on-panel-item-bg pr-[288px] transition-shadow hover:shadow-md"
|
|
onClick={handleBannerClick}
|
|
>
|
|
{/* Left content area */}
|
|
<div className="min-w-0 flex-1">
|
|
<div className="flex h-full flex-col gap-3 py-6 pl-8 pr-0">
|
|
{/* Text section */}
|
|
<div className="flex min-h-24 flex-wrap items-end gap-1 py-1">
|
|
{/* Title area */}
|
|
<div
|
|
ref={textAreaRef}
|
|
className="flex min-w-[480px] max-w-[680px] flex-[1_0_0] flex-col pr-4"
|
|
style={responsiveStyle}
|
|
>
|
|
<p className="title-4xl-semi-bold line-clamp-1 text-dify-logo-blue">
|
|
{category}
|
|
</p>
|
|
<p className="title-4xl-semi-bold line-clamp-2 text-dify-logo-black">
|
|
{title}
|
|
</p>
|
|
</div>
|
|
{/* Description area */}
|
|
<div
|
|
className="min-w-60 max-w-[600px] flex-[1_0_0] self-end overflow-hidden py-1 pr-4"
|
|
style={responsiveStyle}
|
|
>
|
|
<p className="body-sm-regular line-clamp-4 overflow-hidden text-text-tertiary">
|
|
{description}
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Actions section */}
|
|
<div className="flex items-center gap-1">
|
|
{/* View more button */}
|
|
<div
|
|
className="flex min-w-[480px] max-w-[680px] flex-[1_0_0] items-center gap-[6px] py-1 pr-8"
|
|
style={viewMoreStyle}
|
|
>
|
|
<div className="flex h-4 w-4 items-center justify-center rounded-full bg-text-accent p-[2px]">
|
|
<RiArrowRightLine className="h-3 w-3 text-text-primary-on-surface" />
|
|
</div>
|
|
<span className="system-sm-semibold-uppercase text-text-accent">
|
|
{t('banner.viewMore', { ns: 'explore' })}
|
|
</span>
|
|
</div>
|
|
|
|
<div
|
|
className={cn('flex max-w-[600px] flex-[1_0_0] items-center gap-2 py-1 pr-10', maxWidth ? '' : 'min-w-60')}
|
|
style={responsiveStyle}
|
|
>
|
|
{/* Slide navigation indicators */}
|
|
<div className="flex items-center gap-2">
|
|
{slideInfo.slides.map((_: unknown, index: number) => (
|
|
<IndicatorButton
|
|
key={index}
|
|
index={index}
|
|
selectedIndex={selectedIndex}
|
|
isNextSlide={index === slideInfo.nextIndex}
|
|
autoplayDelay={autoplayDelay}
|
|
resetKey={resetKey}
|
|
isPaused={isPaused}
|
|
onClick={() => handleIndicatorClick(index)}
|
|
/>
|
|
))}
|
|
</div>
|
|
<div className="hidden h-[1px] flex-1 bg-divider-regular min-[1380px]:block" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Right image area */}
|
|
<div className="absolute right-0 top-0 flex h-full items-center p-2">
|
|
<img
|
|
src={imgSrc}
|
|
alt={title}
|
|
className="aspect-[4/3] h-full max-w-[296px] rounded-xl"
|
|
/>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|