mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-13 18:57:18 +03:00
* feat: create vite project * feat: it's alive! * feat: `make dev` working! * feat: replace custom serviceWorker with vite plugin * test: replace Jest with Vitest * fix: run prettier * fix: skip eslint for now. * chore: remove ui.old folder * refactor: replace lodash.pick with simple destructuring * fix: eslint errors (wip) * fix: eslint errors (wip) * fix: display-name eslint errors (wip) * fix: no-console eslint errors (wip) * fix: react-refresh/only-export-components eslint errors (wip) * fix: react-refresh/only-export-components eslint errors (wip) * fix: react-refresh/only-export-components eslint errors (wip) * fix: react-refresh/only-export-components eslint errors (wip) * fix: build * fix: pwa manifest * refactor: pwa manifest * refactor: simplify PORT configuration * refactor: rename simple JS files * test: cover playlistUtils * fix: react-image-lightbox * feat(ui): add sourcemaps to help debug issues
364 lines
9.5 KiB
JavaScript
364 lines
9.5 KiB
JavaScript
import { useState, useEffect, useCallback } from 'react'
|
|
import {
|
|
Card,
|
|
CardContent,
|
|
CardMedia,
|
|
Collapse,
|
|
makeStyles,
|
|
Typography,
|
|
useMediaQuery,
|
|
withWidth,
|
|
} from '@material-ui/core'
|
|
import {
|
|
useRecordContext,
|
|
useTranslate,
|
|
ArrayField,
|
|
SingleFieldList,
|
|
ChipField,
|
|
Link,
|
|
} from 'react-admin'
|
|
import Lightbox from 'react-image-lightbox'
|
|
import 'react-image-lightbox/style.css'
|
|
import subsonic from '../subsonic'
|
|
import {
|
|
ArtistLinkField,
|
|
DurationField,
|
|
formatRange,
|
|
SizeField,
|
|
LoveButton,
|
|
RatingField,
|
|
useAlbumsPerPage,
|
|
CollapsibleComment,
|
|
} from '../common'
|
|
import config from '../config'
|
|
import { formatFullDate, intersperse } from '../utils'
|
|
import AlbumExternalLinks from './AlbumExternalLinks'
|
|
|
|
const useStyles = makeStyles(
|
|
(theme) => ({
|
|
root: {
|
|
[theme.breakpoints.down('xs')]: {
|
|
padding: '0.7em',
|
|
minWidth: '20em',
|
|
},
|
|
[theme.breakpoints.up('sm')]: {
|
|
padding: '1em',
|
|
minWidth: '32em',
|
|
},
|
|
},
|
|
cardContents: {
|
|
display: 'flex',
|
|
},
|
|
details: {
|
|
display: 'flex',
|
|
flexDirection: 'column',
|
|
},
|
|
content: {
|
|
flex: '2 0 auto',
|
|
},
|
|
coverParent: {
|
|
[theme.breakpoints.down('xs')]: {
|
|
height: '8em',
|
|
width: '8em',
|
|
minWidth: '8em',
|
|
},
|
|
[theme.breakpoints.up('sm')]: {
|
|
height: '10em',
|
|
width: '10em',
|
|
minWidth: '10em',
|
|
},
|
|
[theme.breakpoints.up('lg')]: {
|
|
height: '15em',
|
|
width: '15em',
|
|
minWidth: '15em',
|
|
},
|
|
},
|
|
cover: {
|
|
objectFit: 'contain',
|
|
cursor: 'pointer',
|
|
display: 'block',
|
|
width: '100%',
|
|
height: '100%',
|
|
},
|
|
loveButton: {
|
|
top: theme.spacing(-0.2),
|
|
left: theme.spacing(0.5),
|
|
},
|
|
notes: {
|
|
display: 'inline-block',
|
|
marginTop: '1em',
|
|
float: 'left',
|
|
wordBreak: 'break-word',
|
|
cursor: 'pointer',
|
|
},
|
|
recordName: {},
|
|
recordArtist: {},
|
|
recordMeta: {},
|
|
genreList: {
|
|
marginTop: theme.spacing(0.5),
|
|
},
|
|
externalLinks: {
|
|
marginTop: theme.spacing(1.5),
|
|
},
|
|
}),
|
|
{
|
|
name: 'NDAlbumDetails',
|
|
},
|
|
)
|
|
|
|
const useGetHandleGenreClick = (width) => {
|
|
const [perPage] = useAlbumsPerPage(width)
|
|
|
|
return (id) => {
|
|
return `/album?filter={"genre_id":"${id}"}&order=ASC&sort=name&perPage=${perPage}`
|
|
}
|
|
}
|
|
|
|
const GenreChipField = withWidth()(({ width, ...rest }) => {
|
|
const record = useRecordContext(rest)
|
|
const genreLink = useGetHandleGenreClick(width)
|
|
|
|
return (
|
|
<Link to={genreLink(record.id)} onClick={(e) => e.stopPropagation()}>
|
|
<ChipField
|
|
source="name"
|
|
// Workaround to force ChipField to be clickable
|
|
onClick={() => {}}
|
|
/>
|
|
</Link>
|
|
)
|
|
})
|
|
|
|
const GenreList = () => {
|
|
const classes = useStyles()
|
|
return (
|
|
<ArrayField className={classes.genreList} source={'genres'}>
|
|
<SingleFieldList linkType={false}>
|
|
<GenreChipField />
|
|
</SingleFieldList>
|
|
</ArrayField>
|
|
)
|
|
}
|
|
|
|
const Details = (props) => {
|
|
const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
|
|
const translate = useTranslate()
|
|
const record = useRecordContext(props)
|
|
let details = []
|
|
const addDetail = (obj) => {
|
|
const id = details.length
|
|
details.push(<span key={`detail-${record.id}-${id}`}>{obj}</span>)
|
|
}
|
|
|
|
const originalYearRange = formatRange(record, 'originalYear')
|
|
const originalDate = record.originalDate
|
|
? formatFullDate(record.originalDate)
|
|
: originalYearRange
|
|
const yearRange = formatRange(record, 'year')
|
|
const date = record.date ? formatFullDate(record.date) : yearRange
|
|
const releaseDate = record.releaseDate
|
|
? formatFullDate(record.releaseDate)
|
|
: date
|
|
|
|
const showReleaseDate = date !== releaseDate && releaseDate.length > 3
|
|
const showOriginalDate =
|
|
date !== originalDate &&
|
|
originalDate !== releaseDate &&
|
|
originalDate.length > 3
|
|
|
|
showOriginalDate &&
|
|
!isXsmall &&
|
|
addDetail(
|
|
<>
|
|
{[translate('resources.album.fields.originalDate'), originalDate].join(
|
|
' ',
|
|
)}
|
|
</>,
|
|
)
|
|
|
|
yearRange && addDetail(<>{['♫', !isXsmall ? date : yearRange].join(' ')}</>)
|
|
|
|
showReleaseDate &&
|
|
addDetail(
|
|
<>
|
|
{(!isXsmall
|
|
? [translate('resources.album.fields.releaseDate'), releaseDate]
|
|
: ['○', record.releaseDate.substring(0, 4)]
|
|
).join(' ')}
|
|
</>,
|
|
)
|
|
|
|
const showReleases = record.releases > 1
|
|
showReleases &&
|
|
addDetail(
|
|
<>
|
|
{!isXsmall
|
|
? [
|
|
record.releases,
|
|
translate('resources.album.fields.releases', {
|
|
smart_count: record.releases,
|
|
}),
|
|
].join(' ')
|
|
: ['(', record.releases, ')))'].join(' ')}
|
|
</>,
|
|
)
|
|
|
|
addDetail(
|
|
<>
|
|
{record.songCount +
|
|
' ' +
|
|
translate('resources.song.name', {
|
|
smart_count: record.songCount,
|
|
})}
|
|
</>,
|
|
)
|
|
!isXsmall && addDetail(<DurationField source={'duration'} />)
|
|
!isXsmall && addDetail(<SizeField source="size" />)
|
|
|
|
return <>{intersperse(details, ' · ')}</>
|
|
}
|
|
|
|
const AlbumDetails = (props) => {
|
|
const record = useRecordContext(props)
|
|
const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
|
|
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('lg'))
|
|
const classes = useStyles()
|
|
const [isLightboxOpen, setLightboxOpen] = useState(false)
|
|
const [expanded, setExpanded] = useState(false)
|
|
const [albumInfo, setAlbumInfo] = useState()
|
|
|
|
let notes =
|
|
albumInfo?.notes?.replace(new RegExp('<.*>', 'g'), '') || record.notes
|
|
|
|
if (notes !== undefined) {
|
|
notes += '..'
|
|
}
|
|
|
|
useEffect(() => {
|
|
subsonic
|
|
.getAlbumInfo(record.id)
|
|
.then((resp) => resp.json['subsonic-response'])
|
|
.then((data) => {
|
|
if (data.status === 'ok') {
|
|
setAlbumInfo(data.albumInfo)
|
|
}
|
|
})
|
|
.catch((e) => {
|
|
// eslint-disable-next-line no-console
|
|
console.error('error on album page', e)
|
|
})
|
|
}, [record])
|
|
|
|
const imageUrl = subsonic.getCoverArtUrl(record, 300)
|
|
const fullImageUrl = subsonic.getCoverArtUrl(record)
|
|
|
|
const handleOpenLightbox = useCallback(() => setLightboxOpen(true), [])
|
|
const handleCloseLightbox = useCallback(() => setLightboxOpen(false), [])
|
|
return (
|
|
<Card className={classes.root}>
|
|
<div className={classes.cardContents}>
|
|
<div className={classes.coverParent}>
|
|
<CardMedia
|
|
component={'img'}
|
|
src={imageUrl}
|
|
width="400"
|
|
height="400"
|
|
className={classes.cover}
|
|
onClick={handleOpenLightbox}
|
|
title={record.name}
|
|
/>
|
|
</div>
|
|
<div className={classes.details}>
|
|
<CardContent className={classes.content}>
|
|
<Typography
|
|
variant={isDesktop ? 'h5' : 'h6'}
|
|
className={classes.recordName}
|
|
>
|
|
{record.name}
|
|
<LoveButton
|
|
className={classes.loveButton}
|
|
record={record}
|
|
resource={'album'}
|
|
size={isDesktop ? 'default' : 'small'}
|
|
aria-label="love"
|
|
color="primary"
|
|
/>
|
|
</Typography>
|
|
<Typography component={'h6'} className={classes.recordArtist}>
|
|
<ArtistLinkField record={record} />
|
|
</Typography>
|
|
<Typography component={'div'} className={classes.recordMeta}>
|
|
<Details />
|
|
</Typography>
|
|
{config.enableStarRating && (
|
|
<div>
|
|
<RatingField
|
|
record={record}
|
|
resource={'album'}
|
|
size={isDesktop ? 'medium' : 'small'}
|
|
/>
|
|
</div>
|
|
)}
|
|
{isDesktop ? (
|
|
<GenreList />
|
|
) : (
|
|
<Typography component={'p'}>{record.genre}</Typography>
|
|
)}
|
|
{!isXsmall && (
|
|
<Typography component={'div'} className={classes.recordMeta}>
|
|
{config.enableExternalServices && (
|
|
<AlbumExternalLinks className={classes.externalLinks} />
|
|
)}
|
|
</Typography>
|
|
)}
|
|
{isDesktop && (
|
|
<Collapse
|
|
collapsedHeight={'2.75em'}
|
|
in={expanded}
|
|
timeout={'auto'}
|
|
className={classes.notes}
|
|
>
|
|
<Typography
|
|
variant={'body1'}
|
|
onClick={() => setExpanded(!expanded)}
|
|
>
|
|
<span dangerouslySetInnerHTML={{ __html: notes }} />
|
|
</Typography>
|
|
</Collapse>
|
|
)}
|
|
{isDesktop && record['comment'] && (
|
|
<CollapsibleComment record={record} />
|
|
)}
|
|
</CardContent>
|
|
</div>
|
|
</div>
|
|
{!isDesktop && record['comment'] && (
|
|
<CollapsibleComment record={record} />
|
|
)}
|
|
{!isDesktop && (
|
|
<div className={classes.notes}>
|
|
<Collapse collapsedHeight={'1.5em'} in={expanded} timeout={'auto'}>
|
|
<Typography
|
|
variant={'body1'}
|
|
onClick={() => setExpanded(!expanded)}
|
|
>
|
|
<span dangerouslySetInnerHTML={{ __html: notes }} />
|
|
</Typography>
|
|
</Collapse>
|
|
</div>
|
|
)}
|
|
{isLightboxOpen && (
|
|
<Lightbox
|
|
imagePadding={50}
|
|
animationDuration={200}
|
|
imageTitle={record.name}
|
|
mainSrc={fullImageUrl}
|
|
onCloseRequest={handleCloseLightbox}
|
|
/>
|
|
)}
|
|
</Card>
|
|
)
|
|
}
|
|
|
|
export default AlbumDetails
|