Merge pull request #1536 from mfcar/addSeasonInfo

Adding podcast type, season and episode info to the feed
This commit is contained in:
advplyr 2023-02-23 17:39:46 -06:00 committed by GitHub
commit 859a53e79a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
20 changed files with 104 additions and 32 deletions

View File

@ -29,10 +29,13 @@
</div> </div>
</div> </div>
<div class="flex flex-wrap"> <div class="flex flex-wrap">
<div class="w-full md:w-1/2 p-2"> <div class="md:w-1/4 p-2">
<ui-dropdown :label="$strings.LabelPodcastType" v-model="podcast.type" :items="podcastTypes" small />
</div>
<div class="md:w-1/4 p-2">
<ui-text-input-with-label v-model="podcast.language" :label="$strings.LabelLanguage" /> <ui-text-input-with-label v-model="podcast.language" :label="$strings.LabelLanguage" />
</div> </div>
<div class="flex-grow px-1 pt-6"> <div class="md:w-1/4 px-2 pt-7">
<ui-checkbox v-model="podcast.explicit" :label="$strings.LabelExplicit" checkbox-bg="primary" border-color="gray-600" label-class="pl-2 text-base font-semibold" /> <ui-checkbox v-model="podcast.explicit" :label="$strings.LabelExplicit" checkbox-bg="primary" border-color="gray-600" label-class="pl-2 text-base font-semibold" />
</div> </div>
</div> </div>
@ -92,7 +95,8 @@ export default {
itunesArtistId: '', itunesArtistId: '',
autoDownloadEpisodes: false, autoDownloadEpisodes: false,
language: '', language: '',
explicit: false explicit: false,
type: ''
} }
} }
}, },
@ -150,6 +154,9 @@ export default {
selectedFolderPath() { selectedFolderPath() {
if (!this.selectedFolder) return '' if (!this.selectedFolder) return ''
return this.selectedFolder.fullPath return this.selectedFolder.fullPath
},
podcastTypes() {
return this.$store.state.globals.podcastTypes || []
} }
}, },
methods: { methods: {
@ -181,7 +188,8 @@ export default {
itunesId: this.podcast.itunesId, itunesId: this.podcast.itunesId,
itunesArtistId: this.podcast.itunesArtistId, itunesArtistId: this.podcast.itunesArtistId,
language: this.podcast.language, language: this.podcast.language,
explicit: this.podcast.explicit explicit: this.podcast.explicit,
type: this.podcast.type
}, },
autoDownloadEpisodes: this.podcast.autoDownloadEpisodes autoDownloadEpisodes: this.podcast.autoDownloadEpisodes
} }
@ -218,6 +226,7 @@ export default {
this.podcast.itunesArtistId = this._podcastData.artistId || '' this.podcast.itunesArtistId = this._podcastData.artistId || ''
this.podcast.language = this._podcastData.language || this.feedMetadata.language || '' this.podcast.language = this._podcastData.language || this.feedMetadata.language || ''
this.podcast.autoDownloadEpisodes = false this.podcast.autoDownloadEpisodes = false
this.podcast.type = this._podcastData.type || this.feedMetadata.type || 'episodic'
this.podcast.explicit = this._podcastData.explicit || this.feedMetadata.explicit === 'yes' || this.feedMetadata.explicit == 'true' this.podcast.explicit = this._podcastData.explicit || this.feedMetadata.explicit === 'yes' || this.feedMetadata.explicit == 'true'
if (this.folderItems[0]) { if (this.folderItems[0]) {

View File

@ -8,7 +8,7 @@
<ui-text-input-with-label v-model="newEpisode.episode" :label="$strings.LabelEpisode" /> <ui-text-input-with-label v-model="newEpisode.episode" :label="$strings.LabelEpisode" />
</div> </div>
<div class="w-1/5 p-1"> <div class="w-1/5 p-1">
<ui-text-input-with-label v-model="newEpisode.episodeType" :label="$strings.LabelEpisodeType" /> <ui-dropdown v-model="newEpisode.episodeType" :label="$strings.LabelEpisodeType" :items="episodeTypes" small />
</div> </div>
<div class="w-2/5 p-1"> <div class="w-2/5 p-1">
<ui-text-input-with-label v-model="pubDateInput" @input="updatePubDate" type="datetime-local" :label="$strings.LabelPubDate" /> <ui-text-input-with-label v-model="pubDateInput" @input="updatePubDate" type="datetime-local" :label="$strings.LabelPubDate" />
@ -89,6 +89,9 @@ export default {
}, },
enclosureUrl() { enclosureUrl() {
return this.enclosure.url return this.enclosure.url
},
episodeTypes() {
return this.$store.state.globals.episodeTypes || []
} }
}, },
methods: { methods: {

View File

@ -39,6 +39,11 @@
</div> </div>
</div> </div>
</div> </div>
<div class="flex mt-2 -mx-1">
<div class="w-1/4 px-1">
<ui-dropdown :label="$strings.LabelPodcastType" v-model="details.type" :items="podcastTypes" small class="max-w-52" />
</div>
</div>
</form> </form>
</div> </div>
</template> </template>
@ -65,7 +70,8 @@ export default {
itunesId: null, itunesId: null,
itunesArtistId: null, itunesArtistId: null,
explicit: false, explicit: false,
language: null language: null,
type: null
}, },
newTags: [] newTags: []
} }
@ -93,6 +99,9 @@ export default {
}, },
filterData() { filterData() {
return this.$store.state.libraries.filterData || {} return this.$store.state.libraries.filterData || {}
},
podcastTypes() {
return this.$store.state.globals.podcastTypes || []
} }
}, },
methods: { methods: {
@ -219,6 +228,7 @@ export default {
this.details.itunesArtistId = this.mediaMetadata.itunesArtistId || '' this.details.itunesArtistId = this.mediaMetadata.itunesArtistId || ''
this.details.language = this.mediaMetadata.language || '' this.details.language = this.mediaMetadata.language || ''
this.details.explicit = !!this.mediaMetadata.explicit this.details.explicit = !!this.mediaMetadata.explicit
this.details.type = this.mediaMetadata.type || 'episodic'
this.newTags = [...(this.media.tags || [])] this.newTags = [...(this.media.tags || [])]
}, },

View File

@ -37,6 +37,15 @@ export const state = () => ({
value: 'yyyy-MM-dd' value: 'yyyy-MM-dd'
} }
], ],
podcastTypes: [
{ text: 'Episodic', value: 'episodic' },
{ text: 'Serial', value: 'serial' }
],
episodeTypes: [
{ text: 'Full', value: 'full' },
{ text: 'Trailer', value: 'trailer' },
{ text: 'Bonus', value: 'bonus' }
],
libraryIcons: ['database', 'audiobookshelf', 'books-1', 'books-2', 'book-1', 'microphone-1', 'microphone-3', 'radio', 'podcast', 'rss', 'headphones', 'music', 'file-picture', 'rocket', 'power', 'star', 'heart'] libraryIcons: ['database', 'audiobookshelf', 'books-1', 'books-2', 'book-1', 'microphone-1', 'microphone-3', 'radio', 'podcast', 'rss', 'headphones', 'music', 'file-picture', 'rocket', 'power', 'star', 'heart']
}) })

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Abspielmethode", "LabelPlayMethod": "Abspielmethode",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasts", "LabelPodcasts": "Podcasts",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Zu ignorierende(s) Vorwort(e) (Groß- und Kleinschreibung wird nicht berücksichtigt)", "LabelPrefixesToIgnore": "Zu ignorierende(s) Vorwort(e) (Groß- und Kleinschreibung wird nicht berücksichtigt)",
"LabelProgress": "Fortschritt", "LabelProgress": "Fortschritt",
"LabelProvider": "Anbieter", "LabelProvider": "Anbieter",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Play Method", "LabelPlayMethod": "Play Method",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasts", "LabelPodcasts": "Podcasts",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Prefixes to Ignore (case insensitive)", "LabelPrefixesToIgnore": "Prefixes to Ignore (case insensitive)",
"LabelProgress": "Progress", "LabelProgress": "Progress",
"LabelProvider": "Provider", "LabelProvider": "Provider",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Play Method", "LabelPlayMethod": "Play Method",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasts", "LabelPodcasts": "Podcasts",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Prefixes to Ignore (case insensitive)", "LabelPrefixesToIgnore": "Prefixes to Ignore (case insensitive)",
"LabelProgress": "Progress", "LabelProgress": "Progress",
"LabelProvider": "Provider", "LabelProvider": "Provider",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Méthode d'écoute", "LabelPlayMethod": "Méthode d'écoute",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasts", "LabelPodcasts": "Podcasts",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Préfixes à Ignorer (Insensible à la Casse)", "LabelPrefixesToIgnore": "Préfixes à Ignorer (Insensible à la Casse)",
"LabelProgress": "Progression", "LabelProgress": "Progression",
"LabelProvider": "Fournisseur", "LabelProvider": "Fournisseur",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Vrsta reprodukcije", "LabelPlayMethod": "Vrsta reprodukcije",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasts", "LabelPodcasts": "Podcasts",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Prefiksi za ignorirati (mala i velika slova nisu bitna)", "LabelPrefixesToIgnore": "Prefiksi za ignorirati (mala i velika slova nisu bitna)",
"LabelProgress": "Napredak", "LabelProgress": "Napredak",
"LabelProvider": "Dobavljač", "LabelProvider": "Dobavljač",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Metodo di riproduzione", "LabelPlayMethod": "Metodo di riproduzione",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasts", "LabelPodcasts": "Podcasts",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Suffissi da ignorare (specificando maiuscole e minuscole)", "LabelPrefixesToIgnore": "Suffissi da ignorare (specificando maiuscole e minuscole)",
"LabelProgress": "Cominciati", "LabelProgress": "Cominciati",
"LabelProvider": "Provider", "LabelProvider": "Provider",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Metoda odtwarzania", "LabelPlayMethod": "Metoda odtwarzania",
"LabelPodcast": "Podcast", "LabelPodcast": "Podcast",
"LabelPodcasts": "Podcasty", "LabelPodcasts": "Podcasty",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Ignorowane prefiksy (wielkość liter nie ma znaczenia)", "LabelPrefixesToIgnore": "Ignorowane prefiksy (wielkość liter nie ma znaczenia)",
"LabelProgress": "Postęp", "LabelProgress": "Postęp",
"LabelProvider": "Dostawca", "LabelProvider": "Dostawca",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "Метод Воспроизведения", "LabelPlayMethod": "Метод Воспроизведения",
"LabelPodcast": "Подкаст", "LabelPodcast": "Подкаст",
"LabelPodcasts": "Подкасты", "LabelPodcasts": "Подкасты",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "Игнорируемые Префиксы (без учета регистра)", "LabelPrefixesToIgnore": "Игнорируемые Префиксы (без учета регистра)",
"LabelProgress": "Прогресс", "LabelProgress": "Прогресс",
"LabelProvider": "Провайдер", "LabelProvider": "Провайдер",

View File

@ -301,6 +301,7 @@
"LabelPlayMethod": "播放方法", "LabelPlayMethod": "播放方法",
"LabelPodcast": "播客", "LabelPodcast": "播客",
"LabelPodcasts": "播客", "LabelPodcasts": "播客",
"LabelPodcastType": "Podcast Type",
"LabelPrefixesToIgnore": "忽略的前缀 (不区分大小写)", "LabelPrefixesToIgnore": "忽略的前缀 (不区分大小写)",
"LabelProgress": "进度", "LabelProgress": "进度",
"LabelProvider": "供应商", "LabelProvider": "供应商",

View File

@ -106,6 +106,8 @@ class Feed {
this.meta.feedUrl = feedUrl this.meta.feedUrl = feedUrl
this.meta.link = `${serverAddress}/item/${libraryItem.id}` this.meta.link = `${serverAddress}/item/${libraryItem.id}`
this.meta.explicit = !!mediaMetadata.explicit this.meta.explicit = !!mediaMetadata.explicit
this.meta.type = mediaMetadata.type
this.meta.language = mediaMetadata.language
this.episodes = [] this.episodes = []
if (isPodcast) { // PODCAST EPISODES if (isPodcast) { // PODCAST EPISODES
@ -142,6 +144,8 @@ class Feed {
this.meta.author = author this.meta.author = author
this.meta.imageUrl = media.coverPath ? `${this.serverAddress}/feed/${this.slug}/cover` : `${this.serverAddress}/Logo.png` this.meta.imageUrl = media.coverPath ? `${this.serverAddress}/feed/${this.slug}/cover` : `${this.serverAddress}/Logo.png`
this.meta.explicit = !!mediaMetadata.explicit this.meta.explicit = !!mediaMetadata.explicit
this.meta.type = mediaMetadata.type
this.meta.language = mediaMetadata.language
this.episodes = [] this.episodes = []
if (isPodcast) { // PODCAST EPISODES if (isPodcast) { // PODCAST EPISODES

View File

@ -14,6 +14,9 @@ class FeedEpisode {
this.author = null this.author = null
this.explicit = null this.explicit = null
this.duration = null this.duration = null
this.season = null
this.episode = null
this.episodeType = null
this.libraryItemId = null this.libraryItemId = null
this.episodeId = null this.episodeId = null
@ -35,6 +38,9 @@ class FeedEpisode {
this.author = episode.author this.author = episode.author
this.explicit = episode.explicit this.explicit = episode.explicit
this.duration = episode.duration this.duration = episode.duration
this.season = episode.season
this.episode = episode.episode
this.episodeType = episode.episodeType
this.libraryItemId = episode.libraryItemId this.libraryItemId = episode.libraryItemId
this.episodeId = episode.episodeId || null this.episodeId = episode.episodeId || null
this.trackIndex = episode.trackIndex || 0 this.trackIndex = episode.trackIndex || 0
@ -52,6 +58,9 @@ class FeedEpisode {
author: this.author, author: this.author,
explicit: this.explicit, explicit: this.explicit,
duration: this.duration, duration: this.duration,
season: this.season,
episode: this.episode,
episodeType: this.episodeType,
libraryItemId: this.libraryItemId, libraryItemId: this.libraryItemId,
episodeId: this.episodeId, episodeId: this.episodeId,
trackIndex: this.trackIndex, trackIndex: this.trackIndex,
@ -77,6 +86,9 @@ class FeedEpisode {
this.author = meta.author this.author = meta.author
this.explicit = mediaMetadata.explicit this.explicit = mediaMetadata.explicit
this.duration = episode.duration this.duration = episode.duration
this.season = episode.season
this.episode = episode.episode
this.episodeType = episode.episodeType
this.libraryItemId = libraryItem.id this.libraryItemId = libraryItem.id
this.episodeId = episode.id this.episodeId = episode.id
this.trackIndex = 0 this.trackIndex = 0
@ -144,7 +156,10 @@ class FeedEpisode {
{ 'itunes:summary': this.description || '' }, { 'itunes:summary': this.description || '' },
{ {
"itunes:explicit": !!this.explicit "itunes:explicit": !!this.explicit
} },
{"itunes:episodeType": this.episodeType},
{"itunes:season": this.season},
{"itunes:episode": this.episode}
] ]
} }
} }

View File

@ -7,6 +7,8 @@ class FeedMeta {
this.feedUrl = null this.feedUrl = null
this.link = null this.link = null
this.explicit = null this.explicit = null
this.type = null
this.language = null
if (meta) { if (meta) {
this.construct(meta) this.construct(meta)
@ -21,6 +23,8 @@ class FeedMeta {
this.feedUrl = meta.feedUrl this.feedUrl = meta.feedUrl
this.link = meta.link this.link = meta.link
this.explicit = meta.explicit this.explicit = meta.explicit
this.type = meta.type
this.language = meta.language
} }
toJSON() { toJSON() {
@ -31,7 +35,9 @@ class FeedMeta {
imageUrl: this.imageUrl, imageUrl: this.imageUrl,
feedUrl: this.feedUrl, feedUrl: this.feedUrl,
link: this.link, link: this.link,
explicit: this.explicit explicit: this.explicit,
type: this.type,
language: this.language
} }
} }
@ -43,16 +49,17 @@ class FeedMeta {
feed_url: this.feedUrl, feed_url: this.feedUrl,
site_url: this.link, site_url: this.link,
image_url: this.imageUrl, image_url: this.imageUrl,
language: 'en',
custom_namespaces: { custom_namespaces: {
'itunes': 'http://www.itunes.com/dtds/podcast-1.0.dtd', 'itunes': 'http://www.itunes.com/dtds/podcast-1.0.dtd',
'psc': 'http://podlove.org/simple-chapters', 'psc': 'http://podlove.org/simple-chapters',
'podcast': 'https://podcastindex.org/namespace/1.0' 'podcast': 'https://podcastindex.org/namespace/1.0'
}, },
custom_elements: [ custom_elements: [
{ 'language': this.language || 'en' },
{ 'author': this.author || 'advplyr' }, { 'author': this.author || 'advplyr' },
{ 'itunes:author': this.author || 'advplyr' }, { 'itunes:author': this.author || 'advplyr' },
{ 'itunes:summary': this.description || '' }, { 'itunes:summary': this.description || '' },
{ 'itunes:type': this.type },
{ {
'itunes:image': { 'itunes:image': {
_attr: { _attr: {

View File

@ -117,7 +117,7 @@ class PodcastEpisode {
this.enclosure = data.enclosure ? { ...data.enclosure } : null this.enclosure = data.enclosure ? { ...data.enclosure } : null
this.season = data.season || '' this.season = data.season || ''
this.episode = data.episode || '' this.episode = data.episode || ''
this.episodeType = data.episodeType || '' this.episodeType = data.episodeType || 'full'
this.publishedAt = data.publishedAt || 0 this.publishedAt = data.publishedAt || 0
this.addedAt = Date.now() this.addedAt = Date.now()
this.updatedAt = Date.now() this.updatedAt = Date.now()

View File

@ -15,6 +15,7 @@ class PodcastMetadata {
this.itunesArtistId = null this.itunesArtistId = null
this.explicit = false this.explicit = false
this.language = null this.language = null
this.type = null
if (metadata) { if (metadata) {
this.construct(metadata) this.construct(metadata)
@ -34,6 +35,7 @@ class PodcastMetadata {
this.itunesArtistId = metadata.itunesArtistId this.itunesArtistId = metadata.itunesArtistId
this.explicit = metadata.explicit this.explicit = metadata.explicit
this.language = metadata.language || null this.language = metadata.language || null
this.type = metadata.type || 'episodic'
} }
toJSON() { toJSON() {
@ -49,7 +51,8 @@ class PodcastMetadata {
itunesId: this.itunesId, itunesId: this.itunesId,
itunesArtistId: this.itunesArtistId, itunesArtistId: this.itunesArtistId,
explicit: this.explicit, explicit: this.explicit,
language: this.language language: this.language,
type: this.type
} }
} }
@ -67,7 +70,8 @@ class PodcastMetadata {
itunesId: this.itunesId, itunesId: this.itunesId,
itunesArtistId: this.itunesArtistId, itunesArtistId: this.itunesArtistId,
explicit: this.explicit, explicit: this.explicit,
language: this.language language: this.language,
type: this.type
} }
} }
@ -112,6 +116,7 @@ class PodcastMetadata {
this.itunesArtistId = mediaMetadata.itunesArtistId || null this.itunesArtistId = mediaMetadata.itunesArtistId || null
this.explicit = !!mediaMetadata.explicit this.explicit = !!mediaMetadata.explicit
this.language = mediaMetadata.language || null this.language = mediaMetadata.language || null
this.type = mediaMetadata.type || null
if (mediaMetadata.genres && mediaMetadata.genres.length) { if (mediaMetadata.genres && mediaMetadata.genres.length) {
this.genres = [...mediaMetadata.genres] this.genres = [...mediaMetadata.genres]
} }

View File

@ -899,7 +899,7 @@ class Scanner {
description: episodeToMatch.description || '', description: episodeToMatch.description || '',
enclosure: episodeToMatch.enclosure || null, enclosure: episodeToMatch.enclosure || null,
episode: episodeToMatch.episode || '', episode: episodeToMatch.episode || '',
episodeType: episodeToMatch.episodeType || '', episodeType: episodeToMatch.episodeType || 'full',
season: episodeToMatch.season || '', season: episodeToMatch.season || '',
pubDate: episodeToMatch.pubDate || '', pubDate: episodeToMatch.pubDate || '',
publishedAt: episodeToMatch.publishedAt publishedAt: episodeToMatch.publishedAt

View File

@ -46,7 +46,8 @@ function extractPodcastMetadata(channel) {
categories: extractCategories(channel), categories: extractCategories(channel),
feedUrl: null, feedUrl: null,
description: null, description: null,
descriptionPlain: null descriptionPlain: null,
type: null
} }
if (channel['itunes:new-feed-url']) { if (channel['itunes:new-feed-url']) {
@ -61,7 +62,7 @@ function extractPodcastMetadata(channel) {
metadata.descriptionPlain = htmlSanitizer.stripAllTags(rawDescription) metadata.descriptionPlain = htmlSanitizer.stripAllTags(rawDescription)
} }
var arrayFields = ['title', 'language', 'itunes:explicit', 'itunes:author', 'pubDate', 'link'] var arrayFields = ['title', 'language', 'itunes:explicit', 'itunes:author', 'pubDate', 'link', 'itunes:type']
arrayFields.forEach((key) => { arrayFields.forEach((key) => {
var cleanKey = key.split(':').pop() var cleanKey = key.split(':').pop()
metadata[cleanKey] = extractFirstArrayItem(channel, key) metadata[cleanKey] = extractFirstArrayItem(channel, key)