diff --git a/client/components/controls/LibraryFilterSelect.vue b/client/components/controls/LibraryFilterSelect.vue index 2bc6109a61..ba96f103bd 100644 --- a/client/components/controls/LibraryFilterSelect.vue +++ b/client/components/controls/LibraryFilterSelect.vue @@ -189,6 +189,12 @@ export default { value: 'publishers', sublist: true }, + { + text: this.$strings.LabelPublishedDecade, + textPlural: this.$strings.LabelPublishedDecades, + value: 'publishedDecades', + sublist: true + }, { text: this.$strings.LabelLanguage, textPlural: this.$strings.LabelLanguages, @@ -338,6 +344,9 @@ export default { publishers() { return this.filterData.publishers || [] }, + publishedDecades() { + return this.filterData.publishedDecades || [] + }, progress() { return [ { diff --git a/client/store/libraries.js b/client/store/libraries.js index b92800baf2..81d3257749 100644 --- a/client/store/libraries.js +++ b/client/store/libraries.js @@ -240,7 +240,8 @@ export const mutations = { series: [], narrators: [], languages: [], - publishers: [] + publishers: [], + publishedDecades: [] } */ const mediaMetadata = libraryItem.media.metadata @@ -307,6 +308,16 @@ export const mutations = { state.filterData.publishers.sort((a, b) => a.localeCompare(b)) } + // Add publishedDecades + if (mediaMetadata.publishedYear) { + const publishedYear = parseInt(mediaMetadata.publishedYear, 10) + const decade = Math.floor(publishedYear / 10) * 10 + if (!state.filterData.publishedDecades.includes(decade)) { + state.filterData.publishedDecades.push(decade) + state.filterData.publishedDecades.sort((a, b) => a - b) + } + } + // Add language if (mediaMetadata.language && !state.filterData.languages.includes(mediaMetadata.language)) { state.filterData.languages.push(mediaMetadata.language) diff --git a/client/store/user.js b/client/store/user.js index 10dc8ef662..0e4cc0ccf9 100644 --- a/client/store/user.js +++ b/client/store/user.js @@ -90,7 +90,7 @@ export const actions = { if (state.settings.orderBy == 'media.metadata.publishedYear') { settingsUpdate.orderBy = 'media.metadata.title' } - const invalidFilters = ['series', 'authors', 'narrators', 'publishers', 'languages', 'progress', 'issues', 'ebooks', 'abridged'] + const invalidFilters = ['series', 'authors', 'narrators', 'publishers', 'publishedDecades', 'languages', 'progress', 'issues', 'ebooks', 'abridged'] const filterByFirstPart = (state.settings.filterBy || '').split('.').shift() if (invalidFilters.includes(filterByFirstPart)) { settingsUpdate.filterBy = 'all' diff --git a/client/strings/en-us.json b/client/strings/en-us.json index a6026ab419..4d4a340b86 100644 --- a/client/strings/en-us.json +++ b/client/strings/en-us.json @@ -465,6 +465,8 @@ "LabelPubDate": "Pub Date", "LabelPublishYear": "Publish Year", "LabelPublishedDate": "Published {0}", + "LabelPublishedDecade": "Published Decade", + "LabelPublishedDecades": "Published Decades", "LabelPublisher": "Publisher", "LabelPublishers": "Publishers", "LabelRSSFeedCustomOwnerEmail": "Custom owner Email", diff --git a/server/Database.js b/server/Database.js index e7bad49ba7..6ae7259553 100644 --- a/server/Database.js +++ b/server/Database.js @@ -601,6 +601,11 @@ class Database { this.libraryFilterData[libraryId].publishers.push(publisher) } + addPublishedDecadeToFilterData(libraryId, decade) { + if (!this.libraryFilterData[libraryId] || !decade || this.libraryFilterData[libraryId].publishedDecades.includes(decade)) return + this.libraryFilterData[libraryId].publishedDecades.push(decade) + } + addLanguageToFilterData(libraryId, language) { if (!this.libraryFilterData[libraryId] || !language || this.libraryFilterData[libraryId].languages.includes(language)) return this.libraryFilterData[libraryId].languages.push(language) diff --git a/server/scanner/BookScanner.js b/server/scanner/BookScanner.js index 279fcf64da..f0737dac05 100644 --- a/server/scanner/BookScanner.js +++ b/server/scanner/BookScanner.js @@ -590,6 +590,10 @@ class BookScanner { Database.addPublisherToFilterData(libraryItemData.libraryId, libraryItem.book.publisher) Database.addLanguageToFilterData(libraryItemData.libraryId, libraryItem.book.language) + const publishedYear = libraryItem.book.publishedYear + const decade = publishedYear ? `${Math.floor(publishedYear / 10) * 10}` : null + Database.addPublishedDecadeToFilterData(libraryItemData.libraryId, decade) + // Load for emitting to client libraryItem.media = await libraryItem.getMedia({ include: [ diff --git a/server/utils/queries/libraryFilters.js b/server/utils/queries/libraryFilters.js index 5c7b7dc0a9..fc04fefc5f 100644 --- a/server/utils/queries/libraryFilters.js +++ b/server/utils/queries/libraryFilters.js @@ -26,7 +26,7 @@ module.exports = { let filterValue = null let filterGroup = null if (filterBy) { - const searchGroups = ['genres', 'tags', 'series', 'authors', 'progress', 'narrators', 'publishers', 'missing', 'languages', 'tracks', 'ebooks'] + const searchGroups = ['genres', 'tags', 'series', 'authors', 'progress', 'narrators', 'publishers', 'publishedDecades', 'missing', 'languages', 'tracks', 'ebooks'] const group = searchGroups.find((_group) => filterBy.startsWith(_group + '.')) filterGroup = group || filterBy filterValue = group ? this.decode(filterBy.replace(`${group}.`, '')) : null @@ -458,6 +458,7 @@ module.exports = { narrators: new Set(), languages: new Set(), publishers: new Set(), + publishedDecades: new Set(), numIssues: 0 } @@ -492,7 +493,7 @@ module.exports = { libraryId: libraryId } }, - attributes: ['tags', 'genres', 'publisher', 'narrators', 'language'] + attributes: ['tags', 'genres', 'publisher', 'publishedYear', 'narrators', 'language'] }) for (const book of books) { if (book.libraryItem.isMissing || book.libraryItem.isInvalid) data.numIssues++ @@ -506,6 +507,11 @@ module.exports = { book.narrators.forEach((narrator) => data.narrators.add(narrator)) } if (book.publisher) data.publishers.add(book.publisher) + // Check if published year exists and is valid + if (book.publishedYear && !isNaN(book.publishedYear) && book.publishedYear > 0 && book.publishedYear < 3000 && book.publishedYear.toString().length === 4) { + const decade = Math.floor(book.publishedYear / 10) * 10 + data.publishedDecades.add(decade.toString()) + } if (book.language) data.languages.add(book.language) } @@ -532,6 +538,7 @@ module.exports = { data.series = naturalSort(data.series).asc((se) => se.name) data.narrators = naturalSort([...data.narrators]).asc() data.publishers = naturalSort([...data.publishers]).asc() + data.publishedDecades = naturalSort([...data.publishedDecades]).asc() data.languages = naturalSort([...data.languages]).asc() data.loadedAt = Date.now() Database.libraryFilterData[libraryId] = data diff --git a/server/utils/queries/libraryItemsBookFilters.js b/server/utils/queries/libraryItemsBookFilters.js index ae1ccc03bc..20886456da 100644 --- a/server/utils/queries/libraryItemsBookFilters.js +++ b/server/utils/queries/libraryItemsBookFilters.js @@ -228,6 +228,11 @@ module.exports = { } else if (value === 'series') { mediaWhere['$series.id$'] = null } + } else if (group === 'publishedDecades') { + const year = parseInt(value, 10) + mediaWhere['publishedYear'] = { + [Sequelize.Op.between]: year >= 1000 ? [year, year + 9] : [year * 10, (year + 1) * 10 - 1] + } } return { mediaWhere, replacements }