mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Add Artist (discography) size, and show sizes in Download caption
This commit is contained in:
parent
1ffc8d619e
commit
68a9be5e86
7 changed files with 70 additions and 16 deletions
41
db/migration/20201012210022_add_artist_playlist_size.go
Normal file
41
db/migration/20201012210022_add_artist_playlist_size.go
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
package migration
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
|
||||||
|
"github.com/pressly/goose"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
goose.AddMigration(Up20201012210022, Down20201012210022)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Up20201012210022(tx *sql.Tx) error {
|
||||||
|
_, err := tx.Exec(`
|
||||||
|
alter table artist
|
||||||
|
add size integer default 0 not null;
|
||||||
|
create index if not exists artist_size
|
||||||
|
on artist(size);
|
||||||
|
|
||||||
|
alter table playlist
|
||||||
|
add size integer default 0 not null;
|
||||||
|
create index if not exists playlist_size
|
||||||
|
on playlist(size);
|
||||||
|
|
||||||
|
update playlist set size = ifnull((
|
||||||
|
select sum(size)
|
||||||
|
from media_file f
|
||||||
|
left join playlist_tracks pt on f.id = pt.media_file_id
|
||||||
|
where pt.playlist_id = playlist.id
|
||||||
|
), 0);`)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
notice(tx, "A full rescan will be performed to calculate artists (discographies) and playlists sizes.")
|
||||||
|
return forceFullRescan(tx)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Down20201012210022(tx *sql.Tx) error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -10,6 +10,7 @@ type Artist struct {
|
||||||
FullText string `json:"fullText"`
|
FullText string `json:"fullText"`
|
||||||
SortArtistName string `json:"sortArtistName"`
|
SortArtistName string `json:"sortArtistName"`
|
||||||
OrderArtistName string `json:"orderArtistName"`
|
OrderArtistName string `json:"orderArtistName"`
|
||||||
|
Size int64 `json:"size"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Artists []Artist
|
type Artists []Artist
|
||||||
|
|
|
@ -133,7 +133,7 @@ func (r *artistRepository) refresh(ids ...string) error {
|
||||||
var artists []refreshArtist
|
var artists []refreshArtist
|
||||||
sel := Select("f.album_artist_id as id", "f.album_artist as name", "count(*) as album_count", "a.id as current_id",
|
sel := Select("f.album_artist_id as id", "f.album_artist as name", "count(*) as album_count", "a.id as current_id",
|
||||||
"f.sort_album_artist_name as sort_artist_name", "f.order_album_artist_name as order_artist_name",
|
"f.sort_album_artist_name as sort_artist_name", "f.order_album_artist_name as order_artist_name",
|
||||||
"sum(f.song_count) as song_count").
|
"sum(f.song_count) as song_count", "sum(f.size) as size").
|
||||||
From("album f").
|
From("album f").
|
||||||
LeftJoin("artist a on f.album_artist_id = a.id").
|
LeftJoin("artist a on f.album_artist_id = a.id").
|
||||||
Where(Eq{"f.album_artist_id": ids}).
|
Where(Eq{"f.album_artist_id": ids}).
|
||||||
|
|
|
@ -13,10 +13,13 @@ import CloudDownloadOutlinedIcon from '@material-ui/icons/CloudDownloadOutlined'
|
||||||
import { RiPlayListAddFill, RiPlayList2Fill } from 'react-icons/ri'
|
import { RiPlayListAddFill, RiPlayList2Fill } from 'react-icons/ri'
|
||||||
import { playNext, addTracks, playTracks, shuffleTracks } from '../audioplayer'
|
import { playNext, addTracks, playTracks, shuffleTracks } from '../audioplayer'
|
||||||
import subsonic from '../subsonic'
|
import subsonic from '../subsonic'
|
||||||
|
import { formatBytes } from '../common/SizeField'
|
||||||
|
import { useMediaQuery } from '@material-ui/core'
|
||||||
|
|
||||||
const AlbumActions = ({ className, ids, data, record, ...rest }) => {
|
const AlbumActions = ({ className, ids, data, record, ...rest }) => {
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const translate = useTranslate()
|
const translate = useTranslate()
|
||||||
|
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('md'))
|
||||||
|
|
||||||
const handlePlay = React.useCallback(() => {
|
const handlePlay = React.useCallback(() => {
|
||||||
dispatch(playTracks(data, ids))
|
dispatch(playTracks(data, ids))
|
||||||
|
@ -66,7 +69,10 @@ const AlbumActions = ({ className, ids, data, record, ...rest }) => {
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
onClick={handleDownload}
|
onClick={handleDownload}
|
||||||
label={translate('resources.album.actions.download')}
|
label={
|
||||||
|
translate('resources.album.actions.download') +
|
||||||
|
(isDesktop ? ` (${formatBytes(record.size)})` : '')
|
||||||
|
}
|
||||||
>
|
>
|
||||||
<CloudDownloadOutlinedIcon />
|
<CloudDownloadOutlinedIcon />
|
||||||
</Button>
|
</Button>
|
||||||
|
|
|
@ -11,6 +11,7 @@ import { playNext, addTracks, playTracks, shuffleTracks } from '../audioplayer'
|
||||||
import { openAddToPlaylist } from '../dialogs/dialogState'
|
import { openAddToPlaylist } from '../dialogs/dialogState'
|
||||||
import subsonic from '../subsonic'
|
import subsonic from '../subsonic'
|
||||||
import StarButton from './StarButton'
|
import StarButton from './StarButton'
|
||||||
|
import { formatBytes } from './SizeField'
|
||||||
|
|
||||||
const useStyles = makeStyles({
|
const useStyles = makeStyles({
|
||||||
noWrap: {
|
noWrap: {
|
||||||
|
@ -40,32 +41,34 @@ const ContextMenu = ({
|
||||||
const options = {
|
const options = {
|
||||||
play: {
|
play: {
|
||||||
needData: true,
|
needData: true,
|
||||||
label: 'resources.album.actions.playAll',
|
label: translate('resources.album.actions.playAll'),
|
||||||
action: (data, ids) => dispatch(playTracks(data, ids)),
|
action: (data, ids) => dispatch(playTracks(data, ids)),
|
||||||
},
|
},
|
||||||
playNext: {
|
playNext: {
|
||||||
needData: true,
|
needData: true,
|
||||||
label: 'resources.album.actions.playNext',
|
label: translate('resources.album.actions.playNext'),
|
||||||
action: (data, ids) => dispatch(playNext(data, ids)),
|
action: (data, ids) => dispatch(playNext(data, ids)),
|
||||||
},
|
},
|
||||||
addToQueue: {
|
addToQueue: {
|
||||||
needData: true,
|
needData: true,
|
||||||
label: 'resources.album.actions.addToQueue',
|
label: translate('resources.album.actions.addToQueue'),
|
||||||
action: (data, ids) => dispatch(addTracks(data, ids)),
|
action: (data, ids) => dispatch(addTracks(data, ids)),
|
||||||
},
|
},
|
||||||
shuffle: {
|
shuffle: {
|
||||||
needData: true,
|
needData: true,
|
||||||
label: 'resources.album.actions.shuffle',
|
label: translate('resources.album.actions.shuffle'),
|
||||||
action: (data, ids) => dispatch(shuffleTracks(data, ids)),
|
action: (data, ids) => dispatch(shuffleTracks(data, ids)),
|
||||||
},
|
},
|
||||||
addToPlaylist: {
|
addToPlaylist: {
|
||||||
needData: true,
|
needData: true,
|
||||||
label: 'resources.album.actions.addToPlaylist',
|
label: translate('resources.album.actions.addToPlaylist'),
|
||||||
action: (data, ids) => dispatch(openAddToPlaylist({ selectedIds: ids })),
|
action: (data, ids) => dispatch(openAddToPlaylist({ selectedIds: ids })),
|
||||||
},
|
},
|
||||||
download: {
|
download: {
|
||||||
needData: false,
|
needData: false,
|
||||||
label: 'resources.album.actions.download',
|
label: `${translate('resources.album.actions.download')} (${formatBytes(
|
||||||
|
record.size
|
||||||
|
)})`,
|
||||||
action: () => subsonic.download(record.id),
|
action: () => subsonic.download(record.id),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -140,7 +143,7 @@ const ContextMenu = ({
|
||||||
>
|
>
|
||||||
{Object.keys(options).map((key) => (
|
{Object.keys(options).map((key) => (
|
||||||
<MenuItem value={key} key={key} onClick={handleItemClick}>
|
<MenuItem value={key} key={key} onClick={handleItemClick}>
|
||||||
{translate(options[key].label)}
|
{options[key].label}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
))}
|
))}
|
||||||
</Menu>
|
</Menu>
|
||||||
|
|
|
@ -5,7 +5,7 @@ const SizeField = ({ record = {}, source }) => {
|
||||||
return <span>{formatBytes(record[source])}</span>
|
return <span>{formatBytes(record[source])}</span>
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatBytes(bytes, decimals = 2) {
|
export const formatBytes = (bytes, decimals = 2) => {
|
||||||
if (bytes === 0) return '0 Bytes'
|
if (bytes === 0) return '0 Bytes'
|
||||||
|
|
||||||
const k = 1024
|
const k = 1024
|
||||||
|
|
|
@ -9,6 +9,7 @@ import { playNext, addTracks, setTrack } from '../audioplayer'
|
||||||
import { openAddToPlaylist } from '../dialogs/dialogState'
|
import { openAddToPlaylist } from '../dialogs/dialogState'
|
||||||
import subsonic from '../subsonic'
|
import subsonic from '../subsonic'
|
||||||
import StarButton from './StarButton'
|
import StarButton from './StarButton'
|
||||||
|
import { formatBytes } from './SizeField'
|
||||||
|
|
||||||
const useStyles = makeStyles({
|
const useStyles = makeStyles({
|
||||||
noWrap: {
|
noWrap: {
|
||||||
|
@ -32,19 +33,19 @@ const SongContextMenu = ({
|
||||||
const [anchorEl, setAnchorEl] = useState(null)
|
const [anchorEl, setAnchorEl] = useState(null)
|
||||||
const options = {
|
const options = {
|
||||||
playNow: {
|
playNow: {
|
||||||
label: 'resources.song.actions.playNow',
|
label: translate('resources.song.actions.playNow'),
|
||||||
action: (record) => dispatch(setTrack(record)),
|
action: (record) => dispatch(setTrack(record)),
|
||||||
},
|
},
|
||||||
playNext: {
|
playNext: {
|
||||||
label: 'resources.song.actions.playNext',
|
label: translate('resources.song.actions.playNext'),
|
||||||
action: (record) => dispatch(playNext({ [record.id]: record })),
|
action: (record) => dispatch(playNext({ [record.id]: record })),
|
||||||
},
|
},
|
||||||
addToQueue: {
|
addToQueue: {
|
||||||
label: 'resources.song.actions.addToQueue',
|
label: translate('resources.song.actions.addToQueue'),
|
||||||
action: (record) => dispatch(addTracks({ [record.id]: record })),
|
action: (record) => dispatch(addTracks({ [record.id]: record })),
|
||||||
},
|
},
|
||||||
addToPlaylist: {
|
addToPlaylist: {
|
||||||
label: 'resources.song.actions.addToPlaylist',
|
label: translate('resources.song.actions.addToPlaylist'),
|
||||||
action: (record) =>
|
action: (record) =>
|
||||||
dispatch(
|
dispatch(
|
||||||
openAddToPlaylist({
|
openAddToPlaylist({
|
||||||
|
@ -54,7 +55,9 @@ const SongContextMenu = ({
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
download: {
|
download: {
|
||||||
label: 'resources.song.actions.download',
|
label: `${translate('resources.song.actions.download')} (${formatBytes(
|
||||||
|
record.size
|
||||||
|
)})`,
|
||||||
action: (record) => subsonic.download(record.id),
|
action: (record) => subsonic.download(record.id),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -95,7 +98,7 @@ const SongContextMenu = ({
|
||||||
>
|
>
|
||||||
{Object.keys(options).map((key) => (
|
{Object.keys(options).map((key) => (
|
||||||
<MenuItem value={key} key={key} onClick={handleItemClick}>
|
<MenuItem value={key} key={key} onClick={handleItemClick}>
|
||||||
{translate(options[key].label)}
|
{options[key].label}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
))}
|
))}
|
||||||
</Menu>
|
</Menu>
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue