198 lines
6.0 KiB
JavaScript
198 lines
6.0 KiB
JavaScript
import Vue from 'vue'
|
|
import dateFormat from 'dateformat'
|
|
import FtLoader from '../../components/ft-loader/ft-loader.vue'
|
|
import FtCard from '../../components/ft-card/ft-card.vue'
|
|
import PlaylistInfo from '../../components/playlist-info/playlist-info.vue'
|
|
import FtListVideo from '../../components/ft-list-video/ft-list-video.vue'
|
|
import FtFlexBox from '../../components/ft-flex-box/ft-flex-box.vue'
|
|
|
|
export default Vue.extend({
|
|
name: 'Search',
|
|
components: {
|
|
'ft-loader': FtLoader,
|
|
'ft-card': FtCard,
|
|
'playlist-info': PlaylistInfo,
|
|
'ft-list-video': FtListVideo,
|
|
'ft-flex-box': FtFlexBox
|
|
},
|
|
data: function () {
|
|
return {
|
|
isLoading: false,
|
|
playlistId: null,
|
|
nextPageRef: '',
|
|
lastSearchQuery: '',
|
|
playlistPage: 1,
|
|
infoData: {},
|
|
playlistItems: []
|
|
}
|
|
},
|
|
computed: {
|
|
backendPreference: function () {
|
|
return this.$store.getters.getBackendPreference
|
|
},
|
|
backendFallback: function () {
|
|
return this.$store.getters.getBackendFallback
|
|
},
|
|
invidiousInstance: function () {
|
|
return this.$store.getters.getInvidiousInstance
|
|
}
|
|
},
|
|
watch: {
|
|
$route () {
|
|
// react to route changes...
|
|
const payload = {
|
|
query: this.$route.params.query,
|
|
options: {},
|
|
nextPage: false
|
|
}
|
|
|
|
if (typeof (this.sessionSearchHistory[this.query]) !== 'undefined' && this.query !== this.$route.params.query) {
|
|
this.isLoading = true
|
|
this.shownResults = []
|
|
// Replacing the data right away causes a strange error where the data
|
|
// Shown is mixed from 2 different search results. So we'll wait a moment
|
|
// Before showing the results.
|
|
setTimeout(this.replaceShownResults, 100, this.sessionSearchHistory[this.query])
|
|
} else {
|
|
this.performSearch(payload)
|
|
}
|
|
}
|
|
},
|
|
mounted: function () {
|
|
this.playlistId = this.$route.params.id
|
|
|
|
switch (this.backendPreference) {
|
|
case 'local':
|
|
this.getPlaylistLocal()
|
|
break
|
|
case 'invidious':
|
|
this.getPlaylistInvidious()
|
|
break
|
|
}
|
|
},
|
|
methods: {
|
|
getPlaylistLocal: function () {
|
|
this.isLoading = true
|
|
|
|
this.$store.dispatch('ytGetPlaylistInfo', this.playlistId).then((result) => {
|
|
console.log('done')
|
|
console.log(result)
|
|
|
|
const randomVideoIndex = Math.floor((Math.random() * result.items.length))
|
|
|
|
this.infoData = {
|
|
id: result.id,
|
|
title: result.title,
|
|
description: result.description ? result.description : '',
|
|
randomVideoId: result.items[randomVideoIndex].id,
|
|
viewCount: result.views,
|
|
videoCount: result.estimatedItemCount,
|
|
lastUpdated: result.lastUpdated ? result.lastUpdated : '',
|
|
channelName: result.author ? result.author.name : '',
|
|
channelThumbnail: result.author ? result.author.bestAvatar.url : '',
|
|
channelId: result.author ? result.author.channelID : '',
|
|
infoSource: 'local'
|
|
}
|
|
|
|
this.playlistItems = result.items.map((video) => {
|
|
if (typeof video.author !== 'undefined') {
|
|
const channelName = video.author.name
|
|
const channelId = video.author.channelID ? video.author.channelID : channelName
|
|
video.author = channelName
|
|
video.authorId = channelId
|
|
} else {
|
|
video.author = ''
|
|
video.authorId = ''
|
|
}
|
|
video.videoId = video.id
|
|
video.lengthSeconds = video.duration
|
|
return video
|
|
})
|
|
|
|
this.isLoading = false
|
|
}).catch((err) => {
|
|
console.log(err)
|
|
if (this.backendPreference === 'local' && this.backendFallback) {
|
|
console.log('Falling back to Invidious API')
|
|
this.getPlaylistInvidious()
|
|
} else {
|
|
this.isLoading = false
|
|
}
|
|
})
|
|
},
|
|
|
|
getPlaylistInvidious: function () {
|
|
this.isLoading = true
|
|
|
|
const payload = {
|
|
resource: 'playlists',
|
|
id: this.playlistId,
|
|
params: {
|
|
page: this.playlistPage
|
|
}
|
|
}
|
|
|
|
this.$store.dispatch('invidiousGetPlaylistInfo', payload).then((result) => {
|
|
console.log('done')
|
|
console.log(result)
|
|
|
|
const randomVideoIndex = Math.floor((Math.random() * result.videos.length) + 1)
|
|
|
|
this.infoData = {
|
|
id: result.playlistId,
|
|
title: result.title,
|
|
description: result.description,
|
|
randomVideoId: result.videos[randomVideoIndex].videoId,
|
|
viewCount: result.viewCount,
|
|
videoCount: result.videoCount,
|
|
channelName: result.author,
|
|
channelThumbnail: result.authorThumbnails[2].url.replace('https://yt3.ggpht.com', `${this.invidiousInstance}/ggpht/`),
|
|
channelId: result.authorId,
|
|
infoSource: 'invidious'
|
|
}
|
|
|
|
const dateString = new Date(result.updated * 1000)
|
|
dateString.setDate(dateString.getDate() + 1)
|
|
this.infoData.lastUpdated = dateFormat(dateString, 'mmm dS, yyyy')
|
|
|
|
this.playlistItems = this.playlistItems.concat(result.videos)
|
|
|
|
if (this.playlistItems.length < result.videoCount) {
|
|
console.log('getting next page')
|
|
this.playlistPage++
|
|
this.getPlaylistInvidious()
|
|
} else {
|
|
this.isLoading = false
|
|
}
|
|
}).catch((err) => {
|
|
console.log(err)
|
|
if (this.backendPreference === 'invidious' && this.backendFallback) {
|
|
console.log('Error getting data with Invidious, falling back to local backend')
|
|
this.getPlaylistLocal()
|
|
} else {
|
|
this.isLoading = false
|
|
// TODO: Show toast with error message
|
|
}
|
|
})
|
|
},
|
|
|
|
nextPage: function () {
|
|
const payload = {
|
|
query: this.query,
|
|
options: {
|
|
nextpageRef: this.nextPageRef
|
|
},
|
|
nextPage: true
|
|
}
|
|
|
|
this.performSearch(payload)
|
|
},
|
|
|
|
replaceShownResults: function (history) {
|
|
this.shownResults = history.data
|
|
this.nextPageRef = history.nextPageRef
|
|
this.isLoading = false
|
|
}
|
|
}
|
|
})
|