mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 04:57:37 +03:00
Initial support for artist browsing from UI. Also add linking between resources
This commit is contained in:
parent
b23175e32b
commit
7dc3f49c1c
6 changed files with 60 additions and 7 deletions
|
@ -15,9 +15,9 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type artist struct {
|
type artist struct {
|
||||||
ID string `orm:"pk;column(id)"`
|
ID string `json:"id" orm:"pk;column(id)"`
|
||||||
Name string `orm:"index"`
|
Name string `json:"name" orm:"index"`
|
||||||
AlbumCount int `orm:"column(album_count)"`
|
AlbumCount int `json:"albumCount" orm:"column(album_count)"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type artistRepository struct {
|
type artistRepository struct {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import (
|
||||||
"net/url"
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/cloudsonic/sonic-server/conf"
|
||||||
"github.com/cloudsonic/sonic-server/model"
|
"github.com/cloudsonic/sonic-server/model"
|
||||||
"github.com/cloudsonic/sonic-server/server"
|
"github.com/cloudsonic/sonic-server/server"
|
||||||
"github.com/deluan/rest"
|
"github.com/deluan/rest"
|
||||||
|
@ -47,12 +48,14 @@ func (app *Router) routes() http.Handler {
|
||||||
r.Post("/login", Login(app.ds))
|
r.Post("/login", Login(app.ds))
|
||||||
|
|
||||||
r.Route("/api", func(r chi.Router) {
|
r.Route("/api", func(r chi.Router) {
|
||||||
// Add User resource
|
if !conf.Sonic.DevDisableAuthentication {
|
||||||
r.Use(jwtauth.Verifier(TokenAuth))
|
r.Use(jwtauth.Verifier(TokenAuth))
|
||||||
r.Use(Authenticator)
|
r.Use(Authenticator)
|
||||||
|
}
|
||||||
app.R(r, "/user", model.User{})
|
app.R(r, "/user", model.User{})
|
||||||
app.R(r, "/song", model.MediaFile{})
|
app.R(r, "/song", model.MediaFile{})
|
||||||
app.R(r, "/album", model.Album{})
|
app.R(r, "/album", model.Album{})
|
||||||
|
app.R(r, "/artist", model.Artist{})
|
||||||
})
|
})
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ import { Login, Layout } from './layout'
|
||||||
import user from './user'
|
import user from './user'
|
||||||
import song from './song'
|
import song from './song'
|
||||||
import album from './album'
|
import album from './album'
|
||||||
|
import artist from './artist'
|
||||||
|
|
||||||
const App = () => (
|
const App = () => (
|
||||||
<Admin
|
<Admin
|
||||||
|
@ -17,6 +18,7 @@ const App = () => (
|
||||||
>
|
>
|
||||||
<Resource name="song" {...song} options={{ subMenu: 'library' }} />
|
<Resource name="song" {...song} options={{ subMenu: 'library' }} />
|
||||||
<Resource name="album" {...album} options={{ subMenu: 'library' }} />
|
<Resource name="album" {...album} options={{ subMenu: 'library' }} />
|
||||||
|
<Resource name="artist" {...artist} options={{ subMenu: 'library' }} />
|
||||||
<Resource name="user" {...user} />
|
<Resource name="user" {...user} />
|
||||||
</Admin>
|
</Admin>
|
||||||
)
|
)
|
||||||
|
|
|
@ -34,6 +34,9 @@ const AlbumDetails = (props) => {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const albumRowClick = (id, basePath, record) =>
|
||||||
|
`/song?filter={"album_id":"${record.id}"}&order=ASC&sort=trackNumber`
|
||||||
|
|
||||||
const AlbumList = (props) => (
|
const AlbumList = (props) => (
|
||||||
<List
|
<List
|
||||||
{...props}
|
{...props}
|
||||||
|
@ -44,7 +47,7 @@ const AlbumList = (props) => (
|
||||||
filters={<AlbumFilter />}
|
filters={<AlbumFilter />}
|
||||||
perPage={15}
|
perPage={15}
|
||||||
>
|
>
|
||||||
<Datagrid expand={<AlbumDetails />}>
|
<Datagrid expand={<AlbumDetails />} rowClick={albumRowClick}>
|
||||||
<TextField source="name" />
|
<TextField source="name" />
|
||||||
<TextField source="artist" />
|
<TextField source="artist" />
|
||||||
<NumberField source="songCount" />
|
<NumberField source="songCount" />
|
||||||
|
|
38
ui/src/artist/ArtistList.js
Normal file
38
ui/src/artist/ArtistList.js
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
import React from 'react'
|
||||||
|
import {
|
||||||
|
Datagrid,
|
||||||
|
Filter,
|
||||||
|
List,
|
||||||
|
NumberField,
|
||||||
|
SearchInput,
|
||||||
|
TextField
|
||||||
|
} from 'react-admin'
|
||||||
|
import { Title } from '../common'
|
||||||
|
|
||||||
|
const ArtistFilter = (props) => (
|
||||||
|
<Filter {...props}>
|
||||||
|
<SearchInput source="name" alwaysOn />
|
||||||
|
</Filter>
|
||||||
|
)
|
||||||
|
|
||||||
|
const artistRowClick = (id, basePath, record) =>
|
||||||
|
`/album?filter={"artist_id":"${record.id}"}&order=ASC&sort=year`
|
||||||
|
|
||||||
|
const ArtistList = (props) => (
|
||||||
|
<List
|
||||||
|
{...props}
|
||||||
|
title={<Title subTitle={'Artists'} />}
|
||||||
|
sort={{ field: 'name', order: 'ASC' }}
|
||||||
|
exporter={false}
|
||||||
|
bulkActionButtons={false}
|
||||||
|
filters={<ArtistFilter />}
|
||||||
|
perPage={15}
|
||||||
|
>
|
||||||
|
<Datagrid rowClick={artistRowClick}>
|
||||||
|
<TextField source="name" />
|
||||||
|
<NumberField source="albumCount" />
|
||||||
|
</Datagrid>
|
||||||
|
</List>
|
||||||
|
)
|
||||||
|
|
||||||
|
export default ArtistList
|
7
ui/src/artist/index.js
Normal file
7
ui/src/artist/index.js
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
import MicIcon from '@material-ui/icons/Mic'
|
||||||
|
import ArtistList from './ArtistList'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
list: ArtistList,
|
||||||
|
icon: MicIcon
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue