d7ec7355c9
* Rework frontend code interacting directly with DOM Rework frontend code that uses user-supplied data to render things like comments and notes. In places where using React's built-in sanitization is possible, the feature is used. In other places, where some markup might be necessary, DOMPurify is used to sanitize the HTML before rendering it. Solves: GHSA-rh3r-8pxm-hg4w * Remove test post DOM rework * fixup! Rework frontend code interacting directly with DOM
139 lines
3.9 KiB
React
139 lines
3.9 KiB
React
import { useState, useEffect } from 'react'
|
|
import { useMediaQuery, withWidth } from '@material-ui/core'
|
|
import {
|
|
useShowController,
|
|
ShowContextProvider,
|
|
useRecordContext,
|
|
useShowContext,
|
|
ReferenceManyField,
|
|
Pagination,
|
|
Title as RaTitle,
|
|
} from 'react-admin'
|
|
import subsonic from '../subsonic'
|
|
import AlbumGridView from '../album/AlbumGridView'
|
|
import MobileArtistDetails from './MobileArtistDetails'
|
|
import DesktopArtistDetails from './DesktopArtistDetails'
|
|
import { useAlbumsPerPage, useResourceRefresh, Title } from '../common/index.js'
|
|
import ArtistActions from './ArtistActions'
|
|
import { makeStyles } from '@material-ui/core'
|
|
|
|
const useStyles = makeStyles(
|
|
(theme) => ({
|
|
actions: {
|
|
width: '100%',
|
|
justifyContent: 'flex-start',
|
|
display: 'flex',
|
|
paddingTop: '0.25em',
|
|
paddingBottom: '0.25em',
|
|
paddingLeft: '1em',
|
|
paddingRight: '1em',
|
|
flexWrap: 'wrap',
|
|
overflowX: 'auto',
|
|
[theme.breakpoints.down('xs')]: {
|
|
paddingLeft: '0.5em',
|
|
paddingRight: '0.5em',
|
|
gap: '0.5em',
|
|
justifyContent: 'space-around',
|
|
},
|
|
},
|
|
actionsContainer: {
|
|
paddingLeft: '.75rem',
|
|
[theme.breakpoints.down('xs')]: {
|
|
padding: '.5rem',
|
|
},
|
|
},
|
|
}),
|
|
{
|
|
name: 'NDArtistShow',
|
|
},
|
|
)
|
|
|
|
const ArtistDetails = (props) => {
|
|
const record = useRecordContext(props)
|
|
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('sm'))
|
|
const [artistInfo, setArtistInfo] = useState()
|
|
|
|
const biography = artistInfo?.biography || record.biography
|
|
|
|
useEffect(() => {
|
|
subsonic
|
|
.getArtistInfo(record.id)
|
|
.then((resp) => resp.json['subsonic-response'])
|
|
.then((data) => {
|
|
if (data.status === 'ok') {
|
|
setArtistInfo(data.artistInfo)
|
|
}
|
|
})
|
|
.catch((e) => {
|
|
// eslint-disable-next-line no-console
|
|
console.error('error on artist page', e)
|
|
})
|
|
}, [record.id])
|
|
|
|
const Component = isDesktop ? DesktopArtistDetails : MobileArtistDetails
|
|
return <Component artistInfo={artistInfo} record={record} biography={biography} />
|
|
}
|
|
|
|
const ArtistShowLayout = (props) => {
|
|
const showContext = useShowContext(props)
|
|
const record = useRecordContext()
|
|
const { width } = props
|
|
const [, perPageOptions] = useAlbumsPerPage(width)
|
|
const classes = useStyles()
|
|
useResourceRefresh('artist', 'album')
|
|
|
|
const maxPerPage = 90
|
|
let perPage = 0
|
|
let pagination = null
|
|
|
|
// Use the main credit count instead of total count, as this is a precise measure
|
|
// of the number of albums where the artist is credited as an album artist OR
|
|
// artist
|
|
const count = record?.stats?.['maincredit']?.albumCount || 0
|
|
|
|
if (count > maxPerPage) {
|
|
perPage = Math.trunc(maxPerPage / perPageOptions[0]) * perPageOptions[0]
|
|
const rowsPerPageOptions = [1, 2, 3].map((option) =>
|
|
Math.trunc(option * (perPage / 3)),
|
|
)
|
|
pagination = <Pagination rowsPerPageOptions={rowsPerPageOptions} />
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{record && <RaTitle title={<Title subTitle={record.name} />} />}
|
|
{record && <ArtistDetails />}
|
|
{record && (
|
|
<div className={classes.actionsContainer}>
|
|
<ArtistActions record={record} className={classes.actions} />
|
|
</div>
|
|
)}
|
|
{record && (
|
|
<ReferenceManyField
|
|
{...showContext}
|
|
addLabel={false}
|
|
reference="album"
|
|
target="artist_id"
|
|
sort={{ field: 'max_year', order: 'ASC' }}
|
|
filter={{ artist_id: record?.id }}
|
|
perPage={perPage}
|
|
pagination={pagination}
|
|
>
|
|
<AlbumGridView {...props} />
|
|
</ReferenceManyField>
|
|
)}
|
|
</>
|
|
)
|
|
}
|
|
|
|
const ArtistShow = withWidth()((props) => {
|
|
const controllerProps = useShowController(props)
|
|
return (
|
|
<ShowContextProvider value={controllerProps}>
|
|
<ArtistShowLayout {...controllerProps} />
|
|
</ShowContextProvider>
|
|
)
|
|
})
|
|
|
|
export default ArtistShow
|