2022-03-27 22:37:04 +02:00
< template >
< modals -modal v-model ="show" name="podcast-episodes-modal" :width="1200" :height="'unset'" :processing ="processing" >
< template # outer >
< div class = "absolute top-0 left-0 p-5 w-2/3 overflow-hidden" >
2023-02-11 22:02:56 +01:00
< p class = "text-3xl text-white truncate" > { { title } } < / p >
2022-03-27 22:37:04 +02:00
< / div >
< / template >
< div ref = "wrapper" id = "podcast-wrapper" class = "p-4 w-full text-sm py-2 rounded-lg bg-bg shadow-lg border border-black-300 relative overflow-hidden" >
2023-02-19 21:07:32 +01:00
< div class = "w-full py-3 mx-auto flex" >
< form @submit.prevent ="submit" class = "flex flex-grow" >
< ui -text -input v -model = " search " @input ="inputUpdate" placeholder = "Search episode..." class = "flex-grow mr-2 text-sm md:text-base" / >
< / form >
< / div >
2022-03-27 22:37:04 +02:00
< div ref = "episodeContainer" id = "episodes-scroll" class = "w-full overflow-x-hidden overflow-y-auto" >
< div
2023-02-19 21:07:32 +01:00
v - for = "(episode, index) in episodesList"
2022-03-27 22:37:04 +02:00
: key = "index"
class = "relative"
2022-08-28 15:48:41 +02:00
: class = "itemEpisodeMap[episode.enclosure.url] ? 'bg-primary bg-opacity-40' : selectedEpisodes[String(index)] ? 'cursor-pointer bg-success bg-opacity-10' : index % 2 == 0 ? 'cursor-pointer bg-primary bg-opacity-25 hover:bg-opacity-40' : 'cursor-pointer bg-primary bg-opacity-5 hover:bg-opacity-25'"
@ click = "toggleSelectEpisode(index, episode)"
2022-03-27 22:37:04 +02:00
>
< div class = "absolute top-0 left-0 h-full flex items-center p-2" >
2022-08-28 15:48:41 +02:00
< span v-if ="itemEpisodeMap[episode.enclosure.url]" class="material-icons text-success text-xl" > download_done < / span >
< ui -checkbox v -else v -model = " selectedEpisodes [ String ( index ) ] " small checkbox -bg = " primary " border -color = " gray -600 " / >
2022-03-27 22:37:04 +02:00
< / div >
< div class = "px-8 py-2" >
< p v-if ="episode.episode" class="font-semibold text-gray-200" > # {{ episode.episode }} < / p >
< p class = "break-words mb-1" > { { episode . title } } < / p >
< p v-if ="episode.subtitle" class="break-words mb-1 text-sm text-gray-300 episode-subtitle" > {{ episode.subtitle }} < / p >
< p class = "text-xs text-gray-300" > Published { { episode . publishedAt ? $dateDistanceFromNow ( episode . publishedAt ) : 'Unknown' } } < / p >
< / div >
< / div >
< / div >
< div class = "flex justify-end pt-4" >
2022-08-28 15:48:41 +02:00
< ui -checkbox v -if = " ! allDownloaded " v -model = " selectAll " @input ="toggleSelectAll" label = "Select all episodes" small checkbox -bg = " primary " border -color = " gray -600 " class = "mx-8" / >
< ui -btn v-if ="!allDownloaded" :disabled="!episodesSelected.length" @click="submit" > {{ buttonText }} < / ui -btn >
< p v -else class = "text-success text-base px-2 py-4" > All episodes are downloaded < / p >
2022-03-27 22:37:04 +02:00
< / div >
< / div >
< / m o d a l s - m o d a l >
< / template >
< script >
export default {
props : {
value : Boolean ,
libraryItem : {
type : Object ,
default : ( ) => { }
} ,
episodes : {
type : Array ,
default : ( ) => [ ]
}
} ,
data ( ) {
return {
processing : false ,
2022-08-28 15:48:41 +02:00
selectedEpisodes : { } ,
2023-02-19 21:07:32 +01:00
selectAll : false ,
search : null ,
searchTimeout : null ,
searchText : null ,
2022-03-27 22:37:04 +02:00
}
} ,
2022-04-10 18:01:50 +02:00
watch : {
show : {
immediate : true ,
handler ( newVal ) {
if ( newVal ) this . init ( )
}
}
} ,
2022-03-27 22:37:04 +02:00
computed : {
show : {
get ( ) {
return this . value
} ,
set ( val ) {
this . $emit ( 'input' , val )
}
} ,
title ( ) {
if ( ! this . libraryItem ) return ''
return this . libraryItem . media . metadata . title || 'Unknown'
} ,
2022-04-09 11:44:31 +02:00
allDownloaded ( ) {
2022-08-28 15:48:41 +02:00
return ! this . episodes . some ( ( episode ) => ! this . itemEpisodeMap [ episode . enclosure . url ] )
2022-04-09 11:44:31 +02:00
} ,
2022-03-27 22:37:04 +02:00
episodesSelected ( ) {
return Object . keys ( this . selectedEpisodes ) . filter ( ( key ) => ! ! this . selectedEpisodes [ key ] )
} ,
buttonText ( ) {
if ( ! this . episodesSelected . length ) return 'No Episodes Selected'
return ` Download ${ this . episodesSelected . length } Episode ${ this . episodesSelected . length > 1 ? 's' : '' } `
} ,
itemEpisodes ( ) {
if ( ! this . libraryItem ) return [ ]
return this . libraryItem . media . episodes || [ ]
} ,
itemEpisodeMap ( ) {
var map = { }
this . itemEpisodes . forEach ( ( item ) => {
if ( item . enclosure ) map [ item . enclosure . url ] = true
} )
return map
2023-02-19 21:07:32 +01:00
} ,
episodesList ( ) {
return this . episodes . filter ( ( episode ) => {
if ( ! this . searchText ) return true
return (
( episode . title && episode . title . toLowerCase ( ) . includes ( this . searchText ) ) ||
( episode . subtitle && episode . subtitle . toLowerCase ( ) . includes ( this . searchText ) )
)
} )
2022-03-27 22:37:04 +02:00
}
} ,
methods : {
2023-02-19 21:07:32 +01:00
inputUpdate ( ) {
clearTimeout ( this . searchTimeout )
this . searchTimeout = setTimeout ( ( ) => {
if ( ! this . search || ! this . search . trim ( ) ) {
this . searchText = ''
return
}
this . searchText = this . search . toLowerCase ( ) . trim ( )
} , 500 )
} ,
2022-08-28 15:48:41 +02:00
toggleSelectAll ( val ) {
for ( let i = 0 ; i < this . episodes . length ; i ++ ) {
const episode = this . episodes [ i ]
if ( this . itemEpisodeMap [ episode . enclosure . url ] ) this . selectedEpisodes [ String ( i ) ] = false
else this . $set ( this . selectedEpisodes , String ( i ) , val )
}
} ,
checkSetIsSelectedAll ( ) {
for ( let i = 0 ; i < this . episodes . length ; i ++ ) {
const episode = this . episodes [ i ]
if ( ! this . itemEpisodeMap [ episode . enclosure . url ] && ! this . selectedEpisodes [ String ( i ) ] ) {
this . selectAll = false
return
}
}
this . selectAll = true
2022-03-27 22:37:04 +02:00
} ,
2022-08-28 15:48:41 +02:00
toggleSelectEpisode ( index , episode ) {
if ( this . itemEpisodeMap [ episode . enclosure . url ] ) return
this . $set ( this . selectedEpisodes , String ( index ) , ! this . selectedEpisodes [ String ( index ) ] )
this . checkSetIsSelectedAll ( )
2022-08-28 14:22:51 +02:00
} ,
2022-03-27 22:37:04 +02:00
submit ( ) {
var episodesToDownload = [ ]
if ( this . episodesSelected . length ) {
episodesToDownload = this . episodesSelected . map ( ( episodeIndex ) => this . episodes [ Number ( episodeIndex ) ] )
}
2022-04-24 02:41:06 +02:00
var payloadSize = JSON . stringify ( episodesToDownload ) . length
var sizeInMb = payloadSize / 1024 / 1024
var sizeInMbPretty = sizeInMb . toFixed ( 2 ) + 'MB'
console . log ( 'Request size' , sizeInMb )
if ( sizeInMb > 4.99 ) {
return this . $toast . error ( ` Request is too large ( ${ sizeInMbPretty } ) should be < 5Mb ` )
}
2022-03-27 22:37:04 +02:00
this . processing = true
this . $axios
. $post ( ` /api/podcasts/ ${ this . libraryItem . id } /download-episodes ` , episodesToDownload )
. then ( ( ) => {
this . processing = false
this . $toast . success ( 'Started downloading episodes' )
this . show = false
} )
. catch ( ( error ) => {
var errorMsg = error . response && error . response . data ? error . response . data : 'Failed to download episodes'
console . error ( 'Failed to download episodes' , error )
this . processing = false
this . $toast . error ( errorMsg )
2022-08-28 15:48:41 +02:00
this . selectedEpisodes = { }
this . selectAll = false
2022-03-27 22:37:04 +02:00
} )
2022-04-10 18:01:50 +02:00
} ,
init ( ) {
2022-08-28 15:48:41 +02:00
this . episodes . sort ( ( a , b ) => ( a . publishedAt < b . publishedAt ? 1 : - 1 ) )
this . selectAll = false
this . selectedEpisodes = { }
2022-03-27 22:37:04 +02:00
}
} ,
mounted ( ) { }
}
< / script >
< style scoped >
# podcast - wrapper {
min - height : 400 px ;
max - height : 80 vh ;
}
# episodes - scroll {
max - height : calc ( 80 vh - 200 px ) ;
}
2022-08-28 05:40:02 +02:00
< / style >