mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Add option to allow share to be downloaded
This commit is contained in:
parent
a22eef39f7
commit
a7d3e6e1f1
13 changed files with 70 additions and 7 deletions
|
@ -59,6 +59,7 @@ type configOptions struct {
|
||||||
EnableStarRating bool
|
EnableStarRating bool
|
||||||
EnableUserEditing bool
|
EnableUserEditing bool
|
||||||
EnableSharing bool
|
EnableSharing bool
|
||||||
|
DefaultDownloadableShare bool
|
||||||
DefaultTheme string
|
DefaultTheme string
|
||||||
DefaultLanguage string
|
DefaultLanguage string
|
||||||
DefaultUIVolume int
|
DefaultUIVolume int
|
||||||
|
@ -306,6 +307,7 @@ func init() {
|
||||||
viper.SetDefault("devautologinusername", "")
|
viper.SetDefault("devautologinusername", "")
|
||||||
viper.SetDefault("devactivitypanel", true)
|
viper.SetDefault("devactivitypanel", true)
|
||||||
viper.SetDefault("enablesharing", false)
|
viper.SetDefault("enablesharing", false)
|
||||||
|
viper.SetDefault("defaultdownloadableshare", false)
|
||||||
viper.SetDefault("devenablebufferedscrobble", true)
|
viper.SetDefault("devenablebufferedscrobble", true)
|
||||||
viper.SetDefault("devsidebarplaylists", true)
|
viper.SetDefault("devsidebarplaylists", true)
|
||||||
viper.SetDefault("devshowartistpage", true)
|
viper.SetDefault("devshowartistpage", true)
|
||||||
|
|
|
@ -125,7 +125,7 @@ func (r *shareRepositoryWrapper) Save(entity interface{}) (string, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *shareRepositoryWrapper) Update(id string, entity interface{}, _ ...string) error {
|
func (r *shareRepositoryWrapper) Update(id string, entity interface{}, _ ...string) error {
|
||||||
cols := []string{"description"}
|
cols := []string{"description", "downloadable"}
|
||||||
|
|
||||||
// TODO Better handling of Share expiration
|
// TODO Better handling of Share expiration
|
||||||
if !entity.(*model.Share).ExpiresAt.IsZero() {
|
if !entity.(*model.Share).ExpiresAt.IsZero() {
|
||||||
|
|
|
@ -45,7 +45,7 @@ var _ = Describe("Share", func() {
|
||||||
entity := &model.Share{}
|
entity := &model.Share{}
|
||||||
err := repo.Update("id", entity)
|
err := repo.Update("id", entity)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(mockedRepo.(*tests.MockShareRepo).Cols).To(ConsistOf("description"))
|
Expect(mockedRepo.(*tests.MockShareRepo).Cols).To(ConsistOf("description", "downloadable"))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
23
db/migration/20230310222612_add_download_to_share.go
Normal file
23
db/migration/20230310222612_add_download_to_share.go
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
package migrations
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
|
||||||
|
"github.com/pressly/goose"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
goose.AddMigration(upAddDownloadToShare, downAddDownloadToShare)
|
||||||
|
}
|
||||||
|
|
||||||
|
func upAddDownloadToShare(tx *sql.Tx) error {
|
||||||
|
_, err := tx.Exec(`
|
||||||
|
alter table share
|
||||||
|
add downloadable bool not null default false;
|
||||||
|
`)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func downAddDownloadToShare(tx *sql.Tx) error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -12,6 +12,7 @@ type Share struct {
|
||||||
UserID string `structs:"user_id" json:"userId,omitempty" orm:"column(user_id)"`
|
UserID string `structs:"user_id" json:"userId,omitempty" orm:"column(user_id)"`
|
||||||
Username string `structs:"-" json:"username,omitempty" orm:"-"`
|
Username string `structs:"-" json:"username,omitempty" orm:"-"`
|
||||||
Description string `structs:"description" json:"description,omitempty"`
|
Description string `structs:"description" json:"description,omitempty"`
|
||||||
|
Downloadable bool `structs:"downloadable" json:"downloadable"`
|
||||||
ExpiresAt time.Time `structs:"expires_at" json:"expiresAt,omitempty"`
|
ExpiresAt time.Time `structs:"expires_at" json:"expiresAt,omitempty"`
|
||||||
LastVisitedAt time.Time `structs:"last_visited_at" json:"lastVisitedAt,omitempty"`
|
LastVisitedAt time.Time `structs:"last_visited_at" json:"lastVisitedAt,omitempty"`
|
||||||
ResourceIDs string `structs:"resource_ids" json:"resourceIds,omitempty" orm:"column(resource_ids)"`
|
ResourceIDs string `structs:"resource_ids" json:"resourceIds,omitempty" orm:"column(resource_ids)"`
|
||||||
|
|
|
@ -183,6 +183,7 @@
|
||||||
"username": "Compartilhado por",
|
"username": "Compartilhado por",
|
||||||
"url": "Link",
|
"url": "Link",
|
||||||
"description": "Descrição",
|
"description": "Descrição",
|
||||||
|
"downloadable": "Permitir Baixar?",
|
||||||
"contents": "Conteúdo",
|
"contents": "Conteúdo",
|
||||||
"expiresAt": "Dt. Expiração",
|
"expiresAt": "Dt. Expiração",
|
||||||
"lastVisitedAt": "Última visita",
|
"lastVisitedAt": "Última visita",
|
||||||
|
|
|
@ -58,6 +58,7 @@ func serveIndex(ds model.DataStore, fs fs.FS, shareInfo *model.Share) http.Handl
|
||||||
"devActivityPanel": conf.Server.DevActivityPanel,
|
"devActivityPanel": conf.Server.DevActivityPanel,
|
||||||
"enableUserEditing": conf.Server.EnableUserEditing,
|
"enableUserEditing": conf.Server.EnableUserEditing,
|
||||||
"enableSharing": conf.Server.EnableSharing,
|
"enableSharing": conf.Server.EnableSharing,
|
||||||
|
"defaultDownloadableShare": conf.Server.DefaultDownloadableShare,
|
||||||
"devSidebarPlaylists": conf.Server.DevSidebarPlaylists,
|
"devSidebarPlaylists": conf.Server.DevSidebarPlaylists,
|
||||||
"lastFMEnabled": conf.Server.LastFM.Enabled,
|
"lastFMEnabled": conf.Server.LastFM.Enabled,
|
||||||
"lastFMApiKey": conf.Server.LastFM.ApiKey,
|
"lastFMApiKey": conf.Server.LastFM.ApiKey,
|
||||||
|
|
|
@ -245,6 +245,17 @@ var _ = Describe("serveIndex", func() {
|
||||||
Expect(config).To(HaveKeyWithValue("enableSharing", false))
|
Expect(config).To(HaveKeyWithValue("enableSharing", false))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("sets the defaultDownloadableShare", func() {
|
||||||
|
conf.Server.DefaultDownloadableShare = true
|
||||||
|
r := httptest.NewRequest("GET", "/index.html", nil)
|
||||||
|
w := httptest.NewRecorder()
|
||||||
|
|
||||||
|
serveIndex(ds, fs, nil)(w, r)
|
||||||
|
|
||||||
|
config := extractAppConfig(w.Body.String())
|
||||||
|
Expect(config).To(HaveKeyWithValue("defaultDownloadableShare", true))
|
||||||
|
})
|
||||||
|
|
||||||
It("sets the defaultDownsamplingFormat", func() {
|
It("sets the defaultDownsamplingFormat", func() {
|
||||||
r := httptest.NewRequest("GET", "/index.html", nil)
|
r := httptest.NewRequest("GET", "/index.html", nil)
|
||||||
w := httptest.NewRecorder()
|
w := httptest.NewRecorder()
|
||||||
|
|
|
@ -22,6 +22,7 @@ const defaultConfig = {
|
||||||
defaultUIVolume: 100,
|
defaultUIVolume: 100,
|
||||||
enableUserEditing: true,
|
enableUserEditing: true,
|
||||||
enableSharing: true,
|
enableSharing: true,
|
||||||
|
defaultDownloadableShare: true,
|
||||||
devSidebarPlaylists: true,
|
devSidebarPlaylists: true,
|
||||||
lastFMEnabled: true,
|
lastFMEnabled: true,
|
||||||
lastFMApiKey: '9b94a5515ea66b2da3ec03c12300327e',
|
lastFMApiKey: '9b94a5515ea66b2da3ec03c12300327e',
|
||||||
|
|
|
@ -8,6 +8,7 @@ import {
|
||||||
import {
|
import {
|
||||||
SimpleForm,
|
SimpleForm,
|
||||||
TextInput,
|
TextInput,
|
||||||
|
BooleanInput,
|
||||||
useCreate,
|
useCreate,
|
||||||
useNotify,
|
useNotify,
|
||||||
useTranslate,
|
useTranslate,
|
||||||
|
@ -17,6 +18,7 @@ import { shareUrl } from '../utils'
|
||||||
import { useTranscodingOptions } from './useTranscodingOptions'
|
import { useTranscodingOptions } from './useTranscodingOptions'
|
||||||
import { useDispatch, useSelector } from 'react-redux'
|
import { useDispatch, useSelector } from 'react-redux'
|
||||||
import { closeShareMenu } from '../actions'
|
import { closeShareMenu } from '../actions'
|
||||||
|
import config from '../config'
|
||||||
|
|
||||||
export const ShareDialog = () => {
|
export const ShareDialog = () => {
|
||||||
const {
|
const {
|
||||||
|
@ -30,6 +32,9 @@ export const ShareDialog = () => {
|
||||||
const notify = useNotify()
|
const notify = useNotify()
|
||||||
const translate = useTranslate()
|
const translate = useTranslate()
|
||||||
const [description, setDescription] = useState('')
|
const [description, setDescription] = useState('')
|
||||||
|
const [downloadable, setDownloadable] = useState(
|
||||||
|
config.defaultDownloadableShare
|
||||||
|
)
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setDescription('')
|
setDescription('')
|
||||||
}, [ids])
|
}, [ids])
|
||||||
|
@ -41,6 +46,7 @@ export const ShareDialog = () => {
|
||||||
resourceType: resource,
|
resourceType: resource,
|
||||||
resourceIds: ids?.join(','),
|
resourceIds: ids?.join(','),
|
||||||
description,
|
description,
|
||||||
|
downloadable,
|
||||||
...(!originalFormat && { format }),
|
...(!originalFormat && { format }),
|
||||||
...(!originalFormat && { maxBitRate }),
|
...(!originalFormat && { maxBitRate }),
|
||||||
},
|
},
|
||||||
|
@ -105,12 +111,21 @@ export const ShareDialog = () => {
|
||||||
<DialogContent>
|
<DialogContent>
|
||||||
<SimpleForm toolbar={null} variant={'outlined'}>
|
<SimpleForm toolbar={null} variant={'outlined'}>
|
||||||
<TextInput
|
<TextInput
|
||||||
source="description"
|
resource={'share'}
|
||||||
|
source={'description'}
|
||||||
fullWidth
|
fullWidth
|
||||||
onChange={(event) => {
|
onChange={(event) => {
|
||||||
setDescription(event.target.value)
|
setDescription(event.target.value)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
<BooleanInput
|
||||||
|
resource={'share'}
|
||||||
|
source={'downloadable'}
|
||||||
|
defaultValue={downloadable}
|
||||||
|
onChange={(value) => {
|
||||||
|
setDownloadable(value)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
<TranscodingOptionsInput
|
<TranscodingOptionsInput
|
||||||
fullWidth
|
fullWidth
|
||||||
label={translate('message.shareOriginalFormat')}
|
label={translate('message.shareOriginalFormat')}
|
||||||
|
|
|
@ -184,6 +184,7 @@
|
||||||
"username": "Shared By",
|
"username": "Shared By",
|
||||||
"url": "URL",
|
"url": "URL",
|
||||||
"description": "Description",
|
"description": "Description",
|
||||||
|
"downloadable": "Allow Downloads?",
|
||||||
"contents": "Contents",
|
"contents": "Contents",
|
||||||
"expiresAt": "Expires",
|
"expiresAt": "Expires",
|
||||||
"lastVisitedAt": "Last Visited",
|
"lastVisitedAt": "Last Visited",
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import {
|
import {
|
||||||
DateTimeInput,
|
DateTimeInput,
|
||||||
|
BooleanInput,
|
||||||
Edit,
|
Edit,
|
||||||
NumberField,
|
NumberField,
|
||||||
SimpleForm,
|
SimpleForm,
|
||||||
|
@ -19,6 +20,7 @@ export const ShareEdit = (props) => {
|
||||||
{url}
|
{url}
|
||||||
</Link>
|
</Link>
|
||||||
<TextInput source="description" />
|
<TextInput source="description" />
|
||||||
|
<BooleanInput source="downloadable" />
|
||||||
<DateTimeInput source="expiresAt" />
|
<DateTimeInput source="expiresAt" />
|
||||||
<TextInput source="contents" disabled />
|
<TextInput source="contents" disabled />
|
||||||
<TextInput source="format" disabled />
|
<TextInput source="format" disabled />
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import {
|
import {
|
||||||
Datagrid,
|
Datagrid,
|
||||||
FunctionField,
|
FunctionField,
|
||||||
|
BooleanField,
|
||||||
List,
|
List,
|
||||||
NumberField,
|
NumberField,
|
||||||
SimpleList,
|
SimpleList,
|
||||||
|
@ -24,6 +25,7 @@ export const FormatInfo = ({ record, size }) => {
|
||||||
|
|
||||||
const ShareList = (props) => {
|
const ShareList = (props) => {
|
||||||
const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
|
const isXsmall = useMediaQuery((theme) => theme.breakpoints.down('xs'))
|
||||||
|
const isDesktop = useMediaQuery((theme) => theme.breakpoints.up('lg'))
|
||||||
const translate = useTranslate()
|
const translate = useTranslate()
|
||||||
const notify = useNotify()
|
const notify = useNotify()
|
||||||
|
|
||||||
|
@ -101,10 +103,13 @@ const ShareList = (props) => {
|
||||||
/>
|
/>
|
||||||
<TextField source="username" />
|
<TextField source="username" />
|
||||||
<TextField source="description" />
|
<TextField source="description" />
|
||||||
<TextField source="contents" />
|
{isDesktop && <TextField source="contents" />}
|
||||||
<FormatInfo source="format" />
|
{isDesktop && <FormatInfo source="format" />}
|
||||||
|
<BooleanField source="downloadable" />
|
||||||
<NumberField source="visitCount" />
|
<NumberField source="visitCount" />
|
||||||
|
{isDesktop && (
|
||||||
<DateField source="lastVisitedAt" showTime sortByOrder={'DESC'} />
|
<DateField source="lastVisitedAt" showTime sortByOrder={'DESC'} />
|
||||||
|
)}
|
||||||
<DateField source="expiresAt" showTime />
|
<DateField source="expiresAt" showTime />
|
||||||
</Datagrid>
|
</Datagrid>
|
||||||
)}
|
)}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue