mirror of
https://github.com/navidrome/navidrome.git
synced 2025-06-05 09:53:18 +03:00
Add bulk action to make playlists private/public
Better responsiveness
This commit is contained in:
parent
778f474d26
commit
975986ab16
@ -150,7 +150,9 @@
|
|||||||
"actions": {
|
"actions": {
|
||||||
"selectPlaylist": "Selecione a playlist:",
|
"selectPlaylist": "Selecione a playlist:",
|
||||||
"addNewPlaylist": "Criar \"%{name}\"",
|
"addNewPlaylist": "Criar \"%{name}\"",
|
||||||
"export": "Exportar"
|
"export": "Exportar",
|
||||||
|
"makePublic": "Pública",
|
||||||
|
"makePrivate": "Pessoal"
|
||||||
},
|
},
|
||||||
"message": {
|
"message": {
|
||||||
"duplicate_song": "Adicionar músicas duplicadas",
|
"duplicate_song": "Adicionar músicas duplicadas",
|
||||||
|
@ -150,7 +150,9 @@
|
|||||||
"actions": {
|
"actions": {
|
||||||
"selectPlaylist": "Select a playlist:",
|
"selectPlaylist": "Select a playlist:",
|
||||||
"addNewPlaylist": "Create \"%{name}\"",
|
"addNewPlaylist": "Create \"%{name}\"",
|
||||||
"export": "Export"
|
"export": "Export",
|
||||||
|
"makePublic": "Make Public",
|
||||||
|
"makePrivate": "Make Private"
|
||||||
},
|
},
|
||||||
"message": {
|
"message": {
|
||||||
"duplicate_song": "Add duplicated songs",
|
"duplicate_song": "Add duplicated songs",
|
||||||
|
17
ui/src/playlist/ChangePublicStatusButton.js
Normal file
17
ui/src/playlist/ChangePublicStatusButton.js
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import * as React from 'react'
|
||||||
|
import { LockOpen, Lock } from '@material-ui/icons'
|
||||||
|
import { BulkUpdateButton, useTranslate } from 'react-admin'
|
||||||
|
|
||||||
|
const ChangePublicStatusButton = (props) => {
|
||||||
|
const translate = useTranslate()
|
||||||
|
const playlists = { public: props?.public }
|
||||||
|
const label = props?.public
|
||||||
|
? translate('resources.playlist.actions.makePublic')
|
||||||
|
: translate('resources.playlist.actions.makePrivate')
|
||||||
|
const icon = props?.public ? <LockOpen /> : <Lock />
|
||||||
|
return (
|
||||||
|
<BulkUpdateButton {...props} data={playlists} label={label} icon={icon} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ChangePublicStatusButton
|
@ -13,7 +13,7 @@ import {
|
|||||||
ReferenceInput,
|
ReferenceInput,
|
||||||
SelectInput,
|
SelectInput,
|
||||||
} from 'react-admin'
|
} from 'react-admin'
|
||||||
import { isSmartPlaylist, isWritable, Title } from '../common'
|
import { isWritable, Title } from '../common'
|
||||||
|
|
||||||
const SyncFragment = ({ formData, variant, ...rest }) => {
|
const SyncFragment = ({ formData, variant, ...rest }) => {
|
||||||
return (
|
return (
|
||||||
@ -52,10 +52,7 @@ const PlaylistEditForm = (props) => {
|
|||||||
) : (
|
) : (
|
||||||
<TextField source="ownerName" />
|
<TextField source="ownerName" />
|
||||||
)}
|
)}
|
||||||
<BooleanInput
|
<BooleanInput source="public" disabled={!isWritable(record.ownerId)} />
|
||||||
source="public"
|
|
||||||
disabled={!isWritable(record.ownerId) || isSmartPlaylist(record)}
|
|
||||||
/>
|
|
||||||
<FormDataConsumer>
|
<FormDataConsumer>
|
||||||
{(formDataProps) => <SyncFragment {...formDataProps} />}
|
{(formDataProps) => <SyncFragment {...formDataProps} />}
|
||||||
</FormDataConsumer>
|
</FormDataConsumer>
|
||||||
|
@ -12,6 +12,8 @@ import {
|
|||||||
useUpdate,
|
useUpdate,
|
||||||
useNotify,
|
useNotify,
|
||||||
useRecordContext,
|
useRecordContext,
|
||||||
|
BulkDeleteButton,
|
||||||
|
usePermissions,
|
||||||
} from 'react-admin'
|
} from 'react-admin'
|
||||||
import Switch from '@material-ui/core/Switch'
|
import Switch from '@material-ui/core/Switch'
|
||||||
import { useMediaQuery } from '@material-ui/core'
|
import { useMediaQuery } from '@material-ui/core'
|
||||||
@ -22,24 +24,29 @@ import {
|
|||||||
isWritable,
|
isWritable,
|
||||||
useSelectedFields,
|
useSelectedFields,
|
||||||
useResourceRefresh,
|
useResourceRefresh,
|
||||||
isSmartPlaylist,
|
|
||||||
} from '../common'
|
} from '../common'
|
||||||
import PlaylistListActions from './PlaylistListActions'
|
import PlaylistListActions from './PlaylistListActions'
|
||||||
|
import ChangePublicStatusButton from './ChangePublicStatusButton'
|
||||||
|
|
||||||
const PlaylistFilter = (props) => (
|
const PlaylistFilter = (props) => {
|
||||||
<Filter {...props} variant={'outlined'}>
|
const { permissions } = usePermissions()
|
||||||
<SearchInput source="q" alwaysOn />
|
return (
|
||||||
<ReferenceInput
|
<Filter {...props} variant={'outlined'}>
|
||||||
source="owner_id"
|
<SearchInput source="q" alwaysOn />
|
||||||
reference="user"
|
{permissions === 'admin' && (
|
||||||
perPage={0}
|
<ReferenceInput
|
||||||
sort={{ field: 'name', order: 'ASC' }}
|
source="owner_id"
|
||||||
alwaysOn
|
reference="user"
|
||||||
>
|
perPage={0}
|
||||||
<SelectInput optionText="name" />
|
sort={{ field: 'name', order: 'ASC' }}
|
||||||
</ReferenceInput>
|
alwaysOn
|
||||||
</Filter>
|
>
|
||||||
)
|
<SelectInput optionText="name" />
|
||||||
|
</ReferenceInput>
|
||||||
|
)}
|
||||||
|
</Filter>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
const TogglePublicInput = ({ resource, source }) => {
|
const TogglePublicInput = ({ resource, source }) => {
|
||||||
const record = useRecordContext()
|
const record = useRecordContext()
|
||||||
@ -69,11 +76,19 @@ const TogglePublicInput = ({ resource, source }) => {
|
|||||||
<Switch
|
<Switch
|
||||||
checked={record[source]}
|
checked={record[source]}
|
||||||
onClick={handleClick}
|
onClick={handleClick}
|
||||||
disabled={!isWritable(record.ownerId) || isSmartPlaylist(record)}
|
disabled={!isWritable(record.ownerId)}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const PlaylistListBulkActions = (props) => (
|
||||||
|
<>
|
||||||
|
<ChangePublicStatusButton public={true} {...props} />
|
||||||
|
<ChangePublicStatusButton public={false} {...props} />
|
||||||
|
<BulkDeleteButton {...props} />
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
|
||||||
const PlaylistList = (props) => {
|
const PlaylistList = (props) => {
|
||||||
const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
|
const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
|
||||||
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('md'))
|
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('md'))
|
||||||
@ -81,9 +96,9 @@ const PlaylistList = (props) => {
|
|||||||
|
|
||||||
const toggleableFields = useMemo(
|
const toggleableFields = useMemo(
|
||||||
() => ({
|
() => ({
|
||||||
ownerName: <TextField source="ownerName" />,
|
ownerName: isDesktop && <TextField source="ownerName" />,
|
||||||
songCount: isDesktop && <NumberField source="songCount" />,
|
songCount: !isXsmall && <NumberField source="songCount" />,
|
||||||
duration: isDesktop && <DurationField source="duration" />,
|
duration: <DurationField source="duration" />,
|
||||||
updatedAt: isDesktop && (
|
updatedAt: isDesktop && (
|
||||||
<DateField source="updatedAt" sortByOrder={'DESC'} />
|
<DateField source="updatedAt" sortByOrder={'DESC'} />
|
||||||
),
|
),
|
||||||
@ -105,6 +120,7 @@ const PlaylistList = (props) => {
|
|||||||
exporter={false}
|
exporter={false}
|
||||||
filters={<PlaylistFilter />}
|
filters={<PlaylistFilter />}
|
||||||
actions={<PlaylistListActions />}
|
actions={<PlaylistListActions />}
|
||||||
|
bulkActionButtons={!isXsmall && <PlaylistListBulkActions />}
|
||||||
>
|
>
|
||||||
<Datagrid rowClick="show" isRowSelectable={(r) => isWritable(r?.ownerId)}>
|
<Datagrid rowClick="show" isRowSelectable={(r) => isWritable(r?.ownerId)}>
|
||||||
<TextField source="name" />
|
<TextField source="name" />
|
||||||
|
Loading…
x
Reference in New Issue
Block a user