Add option to allow share to be downloaded

This commit is contained in:
Deluan 2023-03-10 23:01:03 -05:00 committed by Deluan Quintão
parent a22eef39f7
commit a7d3e6e1f1
13 changed files with 70 additions and 7 deletions

View file

@ -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)

View file

@ -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() {

View file

@ -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"))
}) })
}) })
}) })

View 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
}

View file

@ -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)"`

View file

@ -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",

View file

@ -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,

View file

@ -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()

View file

@ -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',

View file

@ -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')}

View file

@ -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",

View file

@ -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 />

View file

@ -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>
)} )}