mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
New config DefaultLanguage. Closes #1561
This commit is contained in:
parent
72cde6dfde
commit
cb3ba23fce
8 changed files with 75 additions and 24 deletions
|
@ -48,6 +48,7 @@ type configOptions struct {
|
||||||
EnableStarRating bool
|
EnableStarRating bool
|
||||||
EnableUserEditing bool
|
EnableUserEditing bool
|
||||||
DefaultTheme string
|
DefaultTheme string
|
||||||
|
DefaultLanguage string
|
||||||
EnableCoverAnimation bool
|
EnableCoverAnimation bool
|
||||||
GATrackingID string
|
GATrackingID string
|
||||||
EnableLogRedacting bool
|
EnableLogRedacting bool
|
||||||
|
@ -231,6 +232,7 @@ func init() {
|
||||||
viper.SetDefault("enablestarrating", true)
|
viper.SetDefault("enablestarrating", true)
|
||||||
viper.SetDefault("enableuserediting", true)
|
viper.SetDefault("enableuserediting", true)
|
||||||
viper.SetDefault("defaulttheme", "Dark")
|
viper.SetDefault("defaulttheme", "Dark")
|
||||||
|
viper.SetDefault("defaultlanguage", "")
|
||||||
viper.SetDefault("enablecoveranimation", true)
|
viper.SetDefault("enablecoveranimation", true)
|
||||||
viper.SetDefault("gatrackingid", "")
|
viper.SetDefault("gatrackingid", "")
|
||||||
viper.SetDefault("enablelogredacting", true)
|
viper.SetDefault("enablelogredacting", true)
|
||||||
|
|
|
@ -31,30 +31,35 @@ func New(ds model.DataStore, broker events.Broker, share core.Share) *Router {
|
||||||
func (n *Router) routes() http.Handler {
|
func (n *Router) routes() http.Handler {
|
||||||
r := chi.NewRouter()
|
r := chi.NewRouter()
|
||||||
|
|
||||||
r.Use(server.Authenticator(n.ds))
|
// Public
|
||||||
r.Use(server.JWTRefresher)
|
|
||||||
n.R(r, "/user", model.User{}, true)
|
|
||||||
n.R(r, "/song", model.MediaFile{}, false)
|
|
||||||
n.R(r, "/album", model.Album{}, false)
|
|
||||||
n.R(r, "/artist", model.Artist{}, false)
|
|
||||||
n.R(r, "/genre", model.Genre{}, false)
|
|
||||||
n.R(r, "/player", model.Player{}, true)
|
|
||||||
n.R(r, "/playlist", model.Playlist{}, true)
|
|
||||||
n.R(r, "/transcoding", model.Transcoding{}, conf.Server.EnableTranscodingConfig)
|
|
||||||
n.RX(r, "/share", n.share.NewRepository, true)
|
|
||||||
n.RX(r, "/translation", newTranslationRepository, false)
|
n.RX(r, "/translation", newTranslationRepository, false)
|
||||||
|
|
||||||
n.addPlaylistTrackRoute(r)
|
// Protected
|
||||||
|
r.Group(func(r chi.Router) {
|
||||||
|
r.Use(server.Authenticator(n.ds))
|
||||||
|
r.Use(server.JWTRefresher)
|
||||||
|
n.R(r, "/user", model.User{}, true)
|
||||||
|
n.R(r, "/song", model.MediaFile{}, false)
|
||||||
|
n.R(r, "/album", model.Album{}, false)
|
||||||
|
n.R(r, "/artist", model.Artist{}, false)
|
||||||
|
n.R(r, "/genre", model.Genre{}, false)
|
||||||
|
n.R(r, "/player", model.Player{}, true)
|
||||||
|
n.R(r, "/playlist", model.Playlist{}, true)
|
||||||
|
n.R(r, "/transcoding", model.Transcoding{}, conf.Server.EnableTranscodingConfig)
|
||||||
|
n.RX(r, "/share", n.share.NewRepository, true)
|
||||||
|
|
||||||
// Keepalive endpoint to be used to keep the session valid (ex: while playing songs)
|
n.addPlaylistTrackRoute(r)
|
||||||
r.Get("/keepalive/*", func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
_, _ = w.Write([]byte(`{"response":"ok", "id":"keepalive"}`))
|
// Keepalive endpoint to be used to keep the session valid (ex: while playing songs)
|
||||||
|
r.Get("/keepalive/*", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
_, _ = w.Write([]byte(`{"response":"ok", "id":"keepalive"}`))
|
||||||
|
})
|
||||||
|
|
||||||
|
if conf.Server.DevActivityPanel {
|
||||||
|
r.Handle("/events", n.broker)
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
if conf.Server.DevActivityPanel {
|
|
||||||
r.Handle("/events", n.broker)
|
|
||||||
}
|
|
||||||
|
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -38,6 +38,7 @@ func serveIndex(ds model.DataStore, fs fs.FS) http.HandlerFunc {
|
||||||
"enableFavourites": conf.Server.EnableFavourites,
|
"enableFavourites": conf.Server.EnableFavourites,
|
||||||
"enableStarRating": conf.Server.EnableStarRating,
|
"enableStarRating": conf.Server.EnableStarRating,
|
||||||
"defaultTheme": conf.Server.DefaultTheme,
|
"defaultTheme": conf.Server.DefaultTheme,
|
||||||
|
"defaultLanguage": conf.Server.DefaultLanguage,
|
||||||
"enableCoverAnimation": conf.Server.EnableCoverAnimation,
|
"enableCoverAnimation": conf.Server.EnableCoverAnimation,
|
||||||
"gaTrackingId": conf.Server.GATrackingID,
|
"gaTrackingId": conf.Server.GATrackingID,
|
||||||
"losslessFormats": strings.ToUpper(strings.Join(consts.LosslessFormats, ",")),
|
"losslessFormats": strings.ToUpper(strings.Join(consts.LosslessFormats, ",")),
|
||||||
|
|
|
@ -159,6 +159,17 @@ var _ = Describe("serveIndex", func() {
|
||||||
Expect(config).To(HaveKeyWithValue("defaultTheme", "Light"))
|
Expect(config).To(HaveKeyWithValue("defaultTheme", "Light"))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("sets the defaultLanguage", func() {
|
||||||
|
conf.Server.DefaultLanguage = "pt"
|
||||||
|
r := httptest.NewRequest("GET", "/index.html", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
serveIndex(ds, fs)(w, r)
|
||||||
|
|
||||||
|
config := extractAppConfig(w.Body.String())
|
||||||
|
Expect(config).To(HaveKeyWithValue("defaultLanguage", "pt"))
|
||||||
|
})
|
||||||
|
|
||||||
It("sets the enableCoverAnimation", func() {
|
It("sets the enableCoverAnimation", func() {
|
||||||
conf.Server.EnableCoverAnimation = true
|
conf.Server.EnableCoverAnimation = true
|
||||||
r := httptest.NewRequest("GET", "/index.html", nil)
|
r := httptest.NewRequest("GET", "/index.html", nil)
|
||||||
|
|
|
@ -18,6 +18,7 @@ const defaultConfig = {
|
||||||
devFastAccessCoverArt: false,
|
devFastAccessCoverArt: false,
|
||||||
enableStarRating: true,
|
enableStarRating: true,
|
||||||
defaultTheme: 'Dark',
|
defaultTheme: 'Dark',
|
||||||
|
defaultLanguage: '',
|
||||||
enableUserEditing: true,
|
enableUserEditing: true,
|
||||||
devEnableShare: true,
|
devEnableShare: true,
|
||||||
devSidebarPlaylists: true,
|
devSidebarPlaylists: true,
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
import i18nProvider from './provider'
|
import i18nProvider from './provider'
|
||||||
|
import { retrieveTranslation } from './provider'
|
||||||
import useGetLanguageChoices from './useGetLanguageChoices'
|
import useGetLanguageChoices from './useGetLanguageChoices'
|
||||||
|
|
||||||
export { i18nProvider, useGetLanguageChoices }
|
export { i18nProvider, retrieveTranslation, useGetLanguageChoices }
|
||||||
|
|
|
@ -18,7 +18,7 @@ const defaultLocale = function () {
|
||||||
return 'en'
|
return 'en'
|
||||||
}
|
}
|
||||||
|
|
||||||
function retrieveTranslation(locale) {
|
export function retrieveTranslation(locale) {
|
||||||
return dataProvider.getOne('translation', { id: locale }).then((res) => {
|
return dataProvider.getOne('translation', { id: locale }).then((res) => {
|
||||||
localStorage.setItem('translation', JSON.stringify(res.data))
|
localStorage.setItem('translation', JSON.stringify(res.data))
|
||||||
return prepareLanguage(JSON.parse(res.data.data))
|
return prepareLanguage(JSON.parse(res.data.data))
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import React, { useState, useCallback } from 'react'
|
import React, { useState, useCallback, useEffect } from 'react'
|
||||||
import PropTypes from 'prop-types'
|
import PropTypes from 'prop-types'
|
||||||
import { Field, Form } from 'react-final-form'
|
import { Field, Form } from 'react-final-form'
|
||||||
import { useDispatch } from 'react-redux'
|
import { useDispatch } from 'react-redux'
|
||||||
|
@ -8,13 +8,22 @@ import CardActions from '@material-ui/core/CardActions'
|
||||||
import CircularProgress from '@material-ui/core/CircularProgress'
|
import CircularProgress from '@material-ui/core/CircularProgress'
|
||||||
import TextField from '@material-ui/core/TextField'
|
import TextField from '@material-ui/core/TextField'
|
||||||
import { ThemeProvider, makeStyles } from '@material-ui/core/styles'
|
import { ThemeProvider, makeStyles } from '@material-ui/core/styles'
|
||||||
import { createMuiTheme, useLogin, useNotify, useTranslate } from 'react-admin'
|
import {
|
||||||
|
createMuiTheme,
|
||||||
|
useLogin,
|
||||||
|
useNotify,
|
||||||
|
useRefresh,
|
||||||
|
useSetLocale,
|
||||||
|
useTranslate,
|
||||||
|
useVersion,
|
||||||
|
} from 'react-admin'
|
||||||
import Logo from '../icons/android-icon-192x192.png'
|
import Logo from '../icons/android-icon-192x192.png'
|
||||||
|
|
||||||
import Notification from './Notification'
|
import Notification from './Notification'
|
||||||
import useCurrentTheme from '../themes/useCurrentTheme'
|
import useCurrentTheme from '../themes/useCurrentTheme'
|
||||||
import config from '../config'
|
import config from '../config'
|
||||||
import { clearQueue } from '../actions'
|
import { clearQueue } from '../actions'
|
||||||
|
import { retrieveTranslation } from '../i18n'
|
||||||
|
|
||||||
const useStyles = makeStyles(
|
const useStyles = makeStyles(
|
||||||
(theme) => ({
|
(theme) => ({
|
||||||
|
@ -322,9 +331,30 @@ Login.propTypes = {
|
||||||
// the right theme
|
// the right theme
|
||||||
const LoginWithTheme = (props) => {
|
const LoginWithTheme = (props) => {
|
||||||
const theme = useCurrentTheme()
|
const theme = useCurrentTheme()
|
||||||
|
const setLocale = useSetLocale()
|
||||||
|
const refresh = useRefresh()
|
||||||
|
const version = useVersion()
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (config.defaultLanguage !== '') {
|
||||||
|
retrieveTranslation(config.defaultLanguage)
|
||||||
|
.then(() => {
|
||||||
|
setLocale(config.defaultLanguage).then(() => {
|
||||||
|
localStorage.setItem('locale', config.defaultLanguage)
|
||||||
|
})
|
||||||
|
refresh(true)
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
throw new Error(
|
||||||
|
'Cannot load language "' + config.defaultLanguage + '": ' + e
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}, [refresh, setLocale])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ThemeProvider theme={createMuiTheme(theme)}>
|
<ThemeProvider theme={createMuiTheme(theme)}>
|
||||||
<Login {...props} />
|
<Login key={version} {...props} />
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue