mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Initial support for song browsing from UI
This commit is contained in:
parent
fdf1ceeade
commit
9557f7ceed
17 changed files with 158 additions and 28 deletions
|
@ -21,7 +21,7 @@ type Entry struct {
|
||||||
Starred time.Time
|
Starred time.Time
|
||||||
Track int
|
Track int
|
||||||
Duration int
|
Duration int
|
||||||
Size string
|
Size int
|
||||||
Suffix string
|
Suffix string
|
||||||
BitRate int
|
BitRate int
|
||||||
ContentType string
|
ContentType string
|
||||||
|
|
|
@ -18,7 +18,7 @@ type MediaFile struct {
|
||||||
TrackNumber int
|
TrackNumber int
|
||||||
DiscNumber int
|
DiscNumber int
|
||||||
Year int
|
Year int
|
||||||
Size string
|
Size int
|
||||||
Suffix string
|
Suffix string
|
||||||
Duration int
|
Duration int
|
||||||
BitRate int
|
BitRate int
|
||||||
|
|
|
@ -11,26 +11,26 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type mediaFile struct {
|
type mediaFile struct {
|
||||||
ID string `orm:"pk;column(id)"`
|
ID string `json:"id" orm:"pk;column(id)"`
|
||||||
Path string `orm:"index"`
|
Path string `json:"path" orm:"index"`
|
||||||
Title string `orm:"index"`
|
Title string `json:"title" orm:"index"`
|
||||||
Album string ``
|
Album string `json:"album"`
|
||||||
Artist string ``
|
Artist string `json:"artist"`
|
||||||
ArtistID string `orm:"column(artist_id)"`
|
ArtistID string `json:"artistId" orm:"column(artist_id)"`
|
||||||
AlbumArtist string ``
|
AlbumArtist string `json:"albumArtist"`
|
||||||
AlbumID string `orm:"column(album_id);index"`
|
AlbumID string `json:"albumId" orm:"column(album_id);index"`
|
||||||
HasCoverArt bool ``
|
HasCoverArt bool `json:"-"`
|
||||||
TrackNumber int ``
|
TrackNumber int `json:"trackNumber"`
|
||||||
DiscNumber int ``
|
DiscNumber int `json:"discNumber"`
|
||||||
Year int ``
|
Year int `json:"year"`
|
||||||
Size string ``
|
Size int `json:"size"`
|
||||||
Suffix string ``
|
Suffix string `json:"suffix"`
|
||||||
Duration int ``
|
Duration int `json:"duration"`
|
||||||
BitRate int ``
|
BitRate int `json:"bitRate"`
|
||||||
Genre string `orm:"index"`
|
Genre string `json:"genre" orm:"index"`
|
||||||
Compilation bool ``
|
Compilation bool `json:"compilation"`
|
||||||
CreatedAt time.Time `orm:"null"`
|
CreatedAt time.Time `json:"createdAt" orm:"null"`
|
||||||
UpdatedAt time.Time `orm:"null"`
|
UpdatedAt time.Time `json:"updatedAt" orm:"null"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type mediaFileRepository struct {
|
type mediaFileRepository struct {
|
||||||
|
|
|
@ -8,7 +8,6 @@ import (
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"sort"
|
"sort"
|
||||||
"strconv"
|
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
@ -243,7 +242,7 @@ func (s *TagScanner) toMediaFile(md *Metadata) model.MediaFile {
|
||||||
mf.BitRate = md.BitRate()
|
mf.BitRate = md.BitRate()
|
||||||
mf.Path = md.FilePath()
|
mf.Path = md.FilePath()
|
||||||
mf.Suffix = md.Suffix()
|
mf.Suffix = md.Suffix()
|
||||||
mf.Size = strconv.Itoa(md.Size())
|
mf.Size = md.Size()
|
||||||
mf.HasCoverArt = md.HasPicture()
|
mf.HasCoverArt = md.HasPicture()
|
||||||
|
|
||||||
// TODO Get Creation time. https://github.com/djherbis/times ?
|
// TODO Get Creation time. https://github.com/djherbis/times ?
|
||||||
|
|
|
@ -53,6 +53,9 @@ func (app *Router) routes() http.Handler {
|
||||||
R(r, "/user", func(ctx context.Context) rest.Repository {
|
R(r, "/user", func(ctx context.Context) rest.Repository {
|
||||||
return app.ds.Resource(model.User{})
|
return app.ds.Resource(model.User{})
|
||||||
})
|
})
|
||||||
|
R(r, "/song", func(ctx context.Context) rest.Repository {
|
||||||
|
return app.ds.Resource(model.MediaFile{})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
|
@ -181,7 +181,7 @@ func ToChild(entry engine.Entry) responses.Child {
|
||||||
child.CoverArt = entry.CoverArt
|
child.CoverArt = entry.CoverArt
|
||||||
child.Track = entry.Track
|
child.Track = entry.Track
|
||||||
child.Duration = entry.Duration
|
child.Duration = entry.Duration
|
||||||
child.Size = entry.Size
|
child.Size = strconv.Itoa(entry.Size)
|
||||||
child.Suffix = entry.Suffix
|
child.Suffix = entry.Suffix
|
||||||
child.BitRate = entry.BitRate
|
child.BitRate = entry.BitRate
|
||||||
child.ContentType = entry.ContentType
|
child.ContentType = entry.ContentType
|
||||||
|
|
|
@ -2,6 +2,7 @@ package subsonic
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
"github.com/cloudsonic/sonic-server/engine"
|
"github.com/cloudsonic/sonic-server/engine"
|
||||||
"github.com/cloudsonic/sonic-server/log"
|
"github.com/cloudsonic/sonic-server/log"
|
||||||
|
@ -54,7 +55,7 @@ func (c *StreamController) Stream(w http.ResponseWriter, r *http.Request) (*resp
|
||||||
// contentLength = strconv.Itoa((mf.Duration + 1) * maxBitRate * 1000 / 8)
|
// contentLength = strconv.Itoa((mf.Duration + 1) * maxBitRate * 1000 / 8)
|
||||||
//}
|
//}
|
||||||
h := w.Header()
|
h := w.Header()
|
||||||
h.Set("Content-Length", mf.Size)
|
h.Set("Content-Length", strconv.Itoa(mf.Size))
|
||||||
h.Set("Content-Type", "audio/mpeg")
|
h.Set("Content-Type", "audio/mpeg")
|
||||||
h.Set("Expires", "0")
|
h.Set("Expires", "0")
|
||||||
h.Set("Cache-Control", "must-revalidate")
|
h.Set("Cache-Control", "must-revalidate")
|
||||||
|
|
|
@ -5,6 +5,7 @@ import dataProvider from './dataProvider'
|
||||||
import authProvider from './authProvider'
|
import authProvider from './authProvider'
|
||||||
import { Login } from './layout'
|
import { Login } from './layout'
|
||||||
import user from './user'
|
import user from './user'
|
||||||
|
import song from './song'
|
||||||
|
|
||||||
const App = () => (
|
const App = () => (
|
||||||
<Admin
|
<Admin
|
||||||
|
@ -12,6 +13,7 @@ const App = () => (
|
||||||
authProvider={authProvider}
|
authProvider={authProvider}
|
||||||
loginPage={Login}
|
loginPage={Login}
|
||||||
>
|
>
|
||||||
|
<Resource name="song" {...song} />
|
||||||
<Resource name="user" {...user} />
|
<Resource name="user" {...user} />
|
||||||
</Admin>
|
</Admin>
|
||||||
)
|
)
|
||||||
|
|
18
ui/src/common/BitrateField.js
Normal file
18
ui/src/common/BitrateField.js
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
import React from 'react'
|
||||||
|
import PropTypes from 'prop-types'
|
||||||
|
|
||||||
|
const BitrateField = ({ record = {}, source }) => {
|
||||||
|
return <span>{`${record[source]} kbps`}</span>
|
||||||
|
}
|
||||||
|
|
||||||
|
BitrateField.propTypes = {
|
||||||
|
label: PropTypes.string,
|
||||||
|
record: PropTypes.object,
|
||||||
|
source: PropTypes.string.isRequired
|
||||||
|
}
|
||||||
|
|
||||||
|
BitrateField.defaultProps = {
|
||||||
|
addLabel: true
|
||||||
|
}
|
||||||
|
|
||||||
|
export default BitrateField
|
25
ui/src/common/DurationField.js
Normal file
25
ui/src/common/DurationField.js
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
import React from 'react'
|
||||||
|
import PropTypes from 'prop-types'
|
||||||
|
|
||||||
|
const DurationField = ({ record = {}, source }) => {
|
||||||
|
return <span>{format(record[source])}</span>
|
||||||
|
}
|
||||||
|
|
||||||
|
const format = (d) => {
|
||||||
|
const date = new Date(null)
|
||||||
|
date.setSeconds(d)
|
||||||
|
const fmt = date.toISOString().substr(11, 8)
|
||||||
|
return fmt.replace(/^00:/, '')
|
||||||
|
}
|
||||||
|
|
||||||
|
DurationField.propTypes = {
|
||||||
|
label: PropTypes.string,
|
||||||
|
record: PropTypes.object,
|
||||||
|
source: PropTypes.string.isRequired
|
||||||
|
}
|
||||||
|
|
||||||
|
DurationField.defaultProps = {
|
||||||
|
addLabel: true
|
||||||
|
}
|
||||||
|
|
||||||
|
export default DurationField
|
7
ui/src/common/Title.js
Normal file
7
ui/src/common/Title.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import React from 'react'
|
||||||
|
|
||||||
|
const Title = ({ subTitle }) => {
|
||||||
|
return <span>CloudSonic {subTitle ? ` - ${subTitle}` : ''}</span>
|
||||||
|
}
|
||||||
|
|
||||||
|
export default Title
|
5
ui/src/common/index.js
Normal file
5
ui/src/common/index.js
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
import Title from './Title'
|
||||||
|
import DurationField from './DurationField'
|
||||||
|
import BitrateField from './BitrateField'
|
||||||
|
|
||||||
|
export { Title, DurationField, BitrateField }
|
59
ui/src/song/SongList.js
Normal file
59
ui/src/song/SongList.js
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
import React from 'react'
|
||||||
|
import {
|
||||||
|
BooleanField,
|
||||||
|
Datagrid,
|
||||||
|
DateField,
|
||||||
|
Filter,
|
||||||
|
List,
|
||||||
|
NumberField,
|
||||||
|
SearchInput,
|
||||||
|
Show,
|
||||||
|
SimpleShowLayout,
|
||||||
|
TextField
|
||||||
|
} from 'react-admin'
|
||||||
|
import { BitrateField, DurationField, Title } from '../common'
|
||||||
|
|
||||||
|
const SongFilter = (props) => (
|
||||||
|
<Filter {...props}>
|
||||||
|
<SearchInput source="title" alwaysOn />
|
||||||
|
</Filter>
|
||||||
|
)
|
||||||
|
|
||||||
|
const SongDetails = (props) => {
|
||||||
|
return (
|
||||||
|
<Show {...props} title=" ">
|
||||||
|
<SimpleShowLayout>
|
||||||
|
<TextField source="path" />
|
||||||
|
<TextField label="Album Artist" source="albumArtist" />
|
||||||
|
<TextField source="genre" />
|
||||||
|
<BooleanField source="compilation" />
|
||||||
|
<BitrateField source="bitRate" />
|
||||||
|
<DateField source="updatedAt" showTime />
|
||||||
|
</SimpleShowLayout>
|
||||||
|
</Show>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const SongList = (props) => (
|
||||||
|
<List
|
||||||
|
{...props}
|
||||||
|
title={<Title subTitle={'Songs'} />}
|
||||||
|
sort={{ field: 'title', order: 'ASC' }}
|
||||||
|
exporter={false}
|
||||||
|
bulkActionButtons={false}
|
||||||
|
filters={<SongFilter />}
|
||||||
|
perPage={15}
|
||||||
|
>
|
||||||
|
<Datagrid rowClick="expand" expand={<SongDetails />}>
|
||||||
|
<TextField source="title" />
|
||||||
|
<TextField source="album" />
|
||||||
|
<TextField source="artist" />
|
||||||
|
<NumberField label="Track #" source="trackNumber" />
|
||||||
|
<NumberField label="Disc #" source="discNumber" />
|
||||||
|
<TextField source="year" />
|
||||||
|
<DurationField label="Time" source="duration" />
|
||||||
|
</Datagrid>
|
||||||
|
</List>
|
||||||
|
)
|
||||||
|
|
||||||
|
export default SongList
|
7
ui/src/song/index.js
Normal file
7
ui/src/song/index.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import MusicNote from '@material-ui/icons/MusicNote'
|
||||||
|
import SongList from './SongList'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
list: SongList,
|
||||||
|
icon: MusicNote
|
||||||
|
}
|
|
@ -8,9 +8,10 @@ import {
|
||||||
email,
|
email,
|
||||||
SimpleForm
|
SimpleForm
|
||||||
} from 'react-admin'
|
} from 'react-admin'
|
||||||
|
import { Title } from '../common'
|
||||||
|
|
||||||
const UserCreate = (props) => (
|
const UserCreate = (props) => (
|
||||||
<Create {...props}>
|
<Create title={<Title subTitle={'Create User'} />} {...props}>
|
||||||
<SimpleForm redirect="list">
|
<SimpleForm redirect="list">
|
||||||
<TextInput source="userName" validate={[required()]} />
|
<TextInput source="userName" validate={[required()]} />
|
||||||
<TextInput source="name" validate={[required()]} />
|
<TextInput source="name" validate={[required()]} />
|
||||||
|
|
|
@ -9,9 +9,10 @@ import {
|
||||||
email,
|
email,
|
||||||
SimpleForm
|
SimpleForm
|
||||||
} from 'react-admin'
|
} from 'react-admin'
|
||||||
|
import { Title } from '../common'
|
||||||
|
|
||||||
const UserTitle = ({ record }) => {
|
const UserTitle = ({ record }) => {
|
||||||
return <span>User {record ? record.name : ''}</span>
|
return <Title subTitle={`User ${record ? record.name : ''}`} />
|
||||||
}
|
}
|
||||||
const UserEdit = (props) => (
|
const UserEdit = (props) => (
|
||||||
<Edit title={<UserTitle />} {...props}>
|
<Edit title={<UserTitle />} {...props}>
|
||||||
|
|
|
@ -10,6 +10,7 @@ import {
|
||||||
TextField
|
TextField
|
||||||
} from 'react-admin'
|
} from 'react-admin'
|
||||||
import { useMediaQuery } from '@material-ui/core'
|
import { useMediaQuery } from '@material-ui/core'
|
||||||
|
import { Title } from '../common'
|
||||||
|
|
||||||
const UserFilter = (props) => (
|
const UserFilter = (props) => (
|
||||||
<Filter {...props}>
|
<Filter {...props}>
|
||||||
|
@ -23,6 +24,7 @@ const UserList = (props) => {
|
||||||
return (
|
return (
|
||||||
<List
|
<List
|
||||||
{...props}
|
{...props}
|
||||||
|
title={<Title subTitle={'Users'} />}
|
||||||
sort={{ field: 'userName', order: 'ASC' }}
|
sort={{ field: 'userName', order: 'ASC' }}
|
||||||
exporter={false}
|
exporter={false}
|
||||||
filters={<UserFilter />}
|
filters={<UserFilter />}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue