mirror of
https://github.com/qier222/YesPlayMusic.git
synced 2025-02-28 11:04:28 +08:00
93 lines
3.0 KiB
TypeScript
93 lines
3.0 KiB
TypeScript
import { resizeImage } from '@/web/utils/common'
|
|
import { player } from '@/web/store'
|
|
import { useSnapshot } from 'valtio'
|
|
import useTracks from '@/web/api/hooks/useTracks'
|
|
import { css, cx } from '@emotion/css'
|
|
import { AnimatePresence, motion } from 'framer-motion'
|
|
import Image from './Image'
|
|
import Wave from './Wave'
|
|
|
|
const PlayingNext = ({ className }: { className?: string }) => {
|
|
const playerSnapshot = useSnapshot(player)
|
|
const { data: tracks } = useTracks({ ids: playerSnapshot.trackList })
|
|
|
|
return (
|
|
<>
|
|
<div
|
|
className={cx(
|
|
'absolute top-0 z-10 pb-6 text-14 font-bold text-neutral-700 dark:text-neutral-300'
|
|
)}
|
|
>
|
|
PLAYING NEXT
|
|
</div>
|
|
<div
|
|
className={cx(
|
|
'relative z-10 overflow-scroll',
|
|
className,
|
|
css``,
|
|
css`
|
|
padding-top: 42px;
|
|
mask-image: linear-gradient(to bottom, transparent 0, black 42px);
|
|
&::-webkit-scrollbar {
|
|
display: none;
|
|
}
|
|
`
|
|
)}
|
|
>
|
|
<motion.div className='grid gap-4'>
|
|
<AnimatePresence>
|
|
{tracks?.songs?.map((track, index) => (
|
|
<motion.div
|
|
className='flex items-center justify-between'
|
|
key={track.id}
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
exit={{ x: '100%', opacity: 0 }}
|
|
transition={{
|
|
duration: 0.24,
|
|
}}
|
|
layout
|
|
onClick={e => {
|
|
if (e.detail === 2) player.playTrack(track.id)
|
|
}}
|
|
>
|
|
{/* Cover */}
|
|
<Image
|
|
alt='Cover'
|
|
className='mr-4 aspect-square h-14 w-14 flex-shrink-0 rounded-12'
|
|
src={resizeImage(track.al.picUrl, 'sm')}
|
|
/>
|
|
|
|
{/* Track info */}
|
|
<div className='mr-3 flex-grow'>
|
|
<div className='line-clamp-1 text-18 font-medium text-neutral-700 dark:text-neutral-200'>
|
|
{track.name}
|
|
</div>
|
|
<div className='line-clamp-1 mt-1 text-16 font-bold text-neutral-200 dark:text-neutral-700'>
|
|
{track.ar.map(a => a.name).join(', ')}
|
|
</div>
|
|
</div>
|
|
|
|
{/* Wave icon */}
|
|
{playerSnapshot.trackIndex === index ? (
|
|
<Wave playing={playerSnapshot.state === 'playing'} />
|
|
) : (
|
|
<div className='text-18 font-medium text-neutral-700 dark:text-neutral-200'>
|
|
{String(index + 1).padStart(2, '0')}
|
|
</div>
|
|
)}
|
|
</motion.div>
|
|
))}
|
|
|
|
{(tracks?.songs?.length || 0) >= 4 && (
|
|
<div className='pointer-events-none sticky bottom-0 h-8 w-full bg-gradient-to-t from-black'></div>
|
|
)}
|
|
</AnimatePresence>
|
|
</motion.div>
|
|
</div>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default PlayingNext
|