mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
* fix(server): bring back legacy date mappings Signed-off-by: Deluan <deluan@navidrome.org> * reuse the mapDates logic in the legacyReleaseDate function Signed-off-by: Deluan <deluan@navidrome.org> * fix mappings Signed-off-by: Deluan <deluan@navidrome.org> * show original and release dates in album grid Signed-off-by: Deluan <deluan@navidrome.org> * fix tests based on new year mapping Signed-off-by: Deluan <deluan@navidrome.org> * fix(subsonic): prefer returning original_year over (recording) year when sorting albums Signed-off-by: Deluan <deluan@navidrome.org> * fix case when we don't have originalYear Signed-off-by: Deluan <deluan@navidrome.org> * show all dates in album's info, and remove the recording date from the album page Signed-off-by: Deluan <deluan@navidrome.org> * better? Signed-off-by: Deluan <deluan@navidrome.org> * add snapshot tests for Album Details Signed-off-by: Deluan <deluan@navidrome.org> * fix(subsonic): sort order for getAlbumList?type=byYear Signed-off-by: Deluan <deluan@navidrome.org> --------- Signed-off-by: Deluan <deluan@navidrome.org>
226 lines
6 KiB
JavaScript
226 lines
6 KiB
JavaScript
import React from 'react'
|
|
import {
|
|
GridList,
|
|
GridListTile,
|
|
Typography,
|
|
GridListTileBar,
|
|
useMediaQuery,
|
|
} from '@material-ui/core'
|
|
import { makeStyles } from '@material-ui/core/styles'
|
|
import withWidth from '@material-ui/core/withWidth'
|
|
import { Link } from 'react-router-dom'
|
|
import { linkToRecord, useListContext, Loading } from 'react-admin'
|
|
import { withContentRect } from 'react-measure'
|
|
import { useDrag } from 'react-dnd'
|
|
import subsonic from '../subsonic'
|
|
import { AlbumContextMenu, PlayButton, ArtistLinkField } from '../common'
|
|
import { DraggableTypes } from '../consts'
|
|
import clsx from 'clsx'
|
|
import { AlbumDatesField } from './AlbumDatesField.jsx'
|
|
|
|
const useStyles = makeStyles(
|
|
(theme) => ({
|
|
root: {
|
|
margin: '20px',
|
|
display: 'grid',
|
|
},
|
|
tileBar: {
|
|
transition: 'all 150ms ease-out',
|
|
opacity: 0,
|
|
textAlign: 'left',
|
|
marginBottom: '3px',
|
|
background:
|
|
'linear-gradient(to top, rgba(0,0,0,0.7) 0%,rgba(0,0,0,0.4) 70%,rgba(0,0,0,0) 100%)',
|
|
},
|
|
tileBarMobile: {
|
|
textAlign: 'left',
|
|
marginBottom: '3px',
|
|
background:
|
|
'linear-gradient(to top, rgba(0,0,0,0.7) 0%,rgba(0,0,0,0.4) 70%,rgba(0,0,0,0) 100%)',
|
|
},
|
|
albumArtistName: {
|
|
whiteSpace: 'nowrap',
|
|
overflow: 'hidden',
|
|
textOverflow: 'ellipsis',
|
|
textAlign: 'left',
|
|
fontSize: '1em',
|
|
},
|
|
albumName: {
|
|
fontSize: '14px',
|
|
color: theme.palette.type === 'dark' ? '#eee' : 'black',
|
|
overflow: 'hidden',
|
|
whiteSpace: 'nowrap',
|
|
textOverflow: 'ellipsis',
|
|
},
|
|
missingAlbum: {
|
|
opacity: 0.3,
|
|
},
|
|
albumVersion: {
|
|
fontSize: '12px',
|
|
color: theme.palette.type === 'dark' ? '#c5c5c5' : '#696969',
|
|
overflow: 'hidden',
|
|
whiteSpace: 'nowrap',
|
|
textOverflow: 'ellipsis',
|
|
},
|
|
albumSubtitle: {
|
|
fontSize: '12px',
|
|
color: theme.palette.type === 'dark' ? '#c5c5c5' : '#696969',
|
|
overflow: 'hidden',
|
|
whiteSpace: 'nowrap',
|
|
textOverflow: 'ellipsis',
|
|
},
|
|
link: {
|
|
position: 'relative',
|
|
display: 'block',
|
|
textDecoration: 'none',
|
|
'&:hover $tileBar': {
|
|
opacity: 1,
|
|
},
|
|
},
|
|
albumLink: {
|
|
position: 'relative',
|
|
display: 'block',
|
|
textDecoration: 'none',
|
|
},
|
|
albumContainer: {},
|
|
albumPlayButton: { color: 'white' },
|
|
}),
|
|
{ name: 'NDAlbumGridView' },
|
|
)
|
|
|
|
const useCoverStyles = makeStyles({
|
|
cover: {
|
|
display: 'inline-block',
|
|
width: '100%',
|
|
objectFit: 'contain',
|
|
height: (props) => props.height,
|
|
},
|
|
})
|
|
|
|
const getColsForWidth = (width) => {
|
|
if (width === 'xs') return 2
|
|
if (width === 'sm') return 3
|
|
if (width === 'md') return 4
|
|
if (width === 'lg') return 6
|
|
return 9
|
|
}
|
|
|
|
const Cover = withContentRect('bounds')(({
|
|
record,
|
|
measureRef,
|
|
contentRect,
|
|
}) => {
|
|
// Force height to be the same as the width determined by the GridList
|
|
// noinspection JSSuspiciousNameCombination
|
|
const classes = useCoverStyles({ height: contentRect.bounds.width })
|
|
const [, dragAlbumRef] = useDrag(
|
|
() => ({
|
|
type: DraggableTypes.ALBUM,
|
|
item: { albumIds: [record.id] },
|
|
options: { dropEffect: 'copy' },
|
|
}),
|
|
[record],
|
|
)
|
|
return (
|
|
<div ref={measureRef}>
|
|
<div ref={dragAlbumRef}>
|
|
<img
|
|
src={subsonic.getCoverArtUrl(record, 300, true)}
|
|
alt={record.name}
|
|
className={classes.cover}
|
|
/>
|
|
</div>
|
|
</div>
|
|
)
|
|
})
|
|
|
|
const AlbumGridTile = ({ showArtist, record, basePath, ...props }) => {
|
|
const classes = useStyles()
|
|
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('md'), {
|
|
noSsr: true,
|
|
})
|
|
if (!record) {
|
|
return null
|
|
}
|
|
const computedClasses = clsx(
|
|
classes.albumContainer,
|
|
record.missing && classes.missingAlbum,
|
|
)
|
|
return (
|
|
<div className={computedClasses}>
|
|
<Link
|
|
className={classes.link}
|
|
to={linkToRecord(basePath, record.id, 'show')}
|
|
>
|
|
<Cover record={record} />
|
|
<GridListTileBar
|
|
className={isDesktop ? classes.tileBar : classes.tileBarMobile}
|
|
subtitle={
|
|
!record.missing && (
|
|
<PlayButton
|
|
className={classes.albumPlayButton}
|
|
record={record}
|
|
size="small"
|
|
/>
|
|
)
|
|
}
|
|
actionIcon={<AlbumContextMenu record={record} color={'white'} />}
|
|
/>
|
|
</Link>
|
|
<Link
|
|
className={classes.albumLink}
|
|
to={linkToRecord(basePath, record.id, 'show')}
|
|
>
|
|
<span>
|
|
<Typography className={classes.albumName}>{record.name}</Typography>
|
|
{record.tags && record.tags['albumversion'] && (
|
|
<Typography className={classes.albumVersion}>
|
|
{record.tags['albumversion']}
|
|
</Typography>
|
|
)}
|
|
</span>
|
|
</Link>
|
|
{showArtist ? (
|
|
<ArtistLinkField record={record} className={classes.albumSubtitle} />
|
|
) : (
|
|
<AlbumDatesField record={record} className={classes.albumSubtitle} />
|
|
)}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
const LoadedAlbumGrid = ({ ids, data, basePath, width }) => {
|
|
const classes = useStyles()
|
|
const { filterValues } = useListContext()
|
|
const isArtistView = !!(filterValues && filterValues.artist_id)
|
|
return (
|
|
<div className={classes.root}>
|
|
<GridList
|
|
component={'div'}
|
|
cellHeight={'auto'}
|
|
cols={getColsForWidth(width)}
|
|
spacing={20}
|
|
>
|
|
{ids.map((id) => (
|
|
<GridListTile className={classes.gridListTile} key={id}>
|
|
<AlbumGridTile
|
|
record={data[id]}
|
|
basePath={basePath}
|
|
showArtist={!isArtistView}
|
|
/>
|
|
</GridListTile>
|
|
))}
|
|
</GridList>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
const AlbumGridView = ({ albumListType, loaded, loading, ...props }) => {
|
|
const hide =
|
|
(loading && albumListType === 'random') || !props.data || !props.ids
|
|
return hide ? <Loading /> : <LoadedAlbumGrid {...props} />
|
|
}
|
|
|
|
const AlbumGridViewWithWidth = withWidth()(AlbumGridView)
|
|
|
|
export default AlbumGridViewWithWidth
|