2023-08-20 00:12:24 +02:00
|
|
|
const { createNewSortInstance } = require('../libs/fastSort')
|
2023-07-05 01:14:44 +02:00
|
|
|
const Database = require('../Database')
|
2022-11-03 05:14:07 +01:00
|
|
|
const { getTitlePrefixAtEnd, isNullOrNaN, getTitleIgnorePrefix } = require('../utils/index')
|
2021-12-26 18:25:07 +01:00
|
|
|
const naturalSort = createNewSortInstance({
|
|
|
|
comparer: new Intl.Collator(undefined, { numeric: true, sensitivity: 'base' }).compare
|
|
|
|
})
|
2021-12-01 03:02:40 +01:00
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
decode(text) {
|
|
|
|
return Buffer.from(decodeURIComponent(text), 'base64').toString()
|
|
|
|
},
|
|
|
|
|
2023-07-17 23:48:46 +02:00
|
|
|
async getFilteredLibraryItems(libraryItems, filterBy, user) {
|
2022-11-28 00:42:02 +01:00
|
|
|
let filtered = libraryItems
|
2022-03-11 01:45:02 +01:00
|
|
|
|
2023-07-15 19:22:13 +02:00
|
|
|
const searchGroups = ['genres', 'tags', 'series', 'authors', 'progress', 'narrators', 'publishers', 'missing', 'languages', 'tracks', 'ebooks']
|
2022-11-28 00:42:02 +01:00
|
|
|
const group = searchGroups.find(_group => filterBy.startsWith(_group + '.'))
|
2022-03-11 01:45:02 +01:00
|
|
|
if (group) {
|
2022-11-28 00:42:02 +01:00
|
|
|
const filterVal = filterBy.replace(`${group}.`, '')
|
|
|
|
const filter = this.decode(filterVal)
|
2023-05-30 23:37:24 +02:00
|
|
|
if (group === 'genres') filtered = filtered.filter(li => li.media.metadata.genres?.includes(filter))
|
2022-03-11 01:45:02 +01:00
|
|
|
else if (group === 'tags') filtered = filtered.filter(li => li.media.tags.includes(filter))
|
|
|
|
else if (group === 'series') {
|
2022-11-28 00:54:40 +01:00
|
|
|
if (filter === 'no-series') filtered = filtered.filter(li => li.isBook && !li.media.metadata.series.length)
|
2022-03-13 01:50:31 +01:00
|
|
|
else {
|
2022-11-28 00:42:02 +01:00
|
|
|
filtered = filtered.filter(li => li.isBook && li.media.metadata.hasSeries(filter))
|
2022-03-13 01:50:31 +01:00
|
|
|
}
|
2022-03-11 01:45:02 +01:00
|
|
|
}
|
2022-11-28 00:42:02 +01:00
|
|
|
else if (group === 'authors') filtered = filtered.filter(li => li.isBook && li.media.metadata.hasAuthor(filter))
|
|
|
|
else if (group === 'narrators') filtered = filtered.filter(li => li.isBook && li.media.metadata.hasNarrator(filter))
|
2023-07-15 19:22:13 +02:00
|
|
|
else if (group === 'publishers') filtered = filtered.filter(li => li.isBook && li.media.metadata.publisher === filter)
|
2022-03-11 01:45:02 +01:00
|
|
|
else if (group === 'progress') {
|
|
|
|
filtered = filtered.filter(li => {
|
2022-11-28 00:42:02 +01:00
|
|
|
const itemProgress = user.getMediaProgress(li.id)
|
2022-11-28 00:54:40 +01:00
|
|
|
if (filter === 'finished' && (itemProgress && itemProgress.isFinished)) return true
|
2023-07-29 01:03:31 +02:00
|
|
|
if (filter === 'not-started' && (!itemProgress || itemProgress.notStarted)) return true
|
2022-11-28 00:54:40 +01:00
|
|
|
if (filter === 'not-finished' && (!itemProgress || !itemProgress.isFinished)) return true
|
|
|
|
if (filter === 'in-progress' && (itemProgress && itemProgress.inProgress)) return true
|
2022-03-11 01:45:02 +01:00
|
|
|
return false
|
|
|
|
})
|
2022-04-18 08:31:39 +02:00
|
|
|
} else if (group == 'missing') {
|
|
|
|
filtered = filtered.filter(li => {
|
2022-11-28 00:42:02 +01:00
|
|
|
if (li.isBook) {
|
2022-12-16 00:46:27 +01:00
|
|
|
if (filter === 'asin' && !li.media.metadata.asin) return true
|
|
|
|
if (filter === 'isbn' && !li.media.metadata.isbn) return true
|
|
|
|
if (filter === 'subtitle' && !li.media.metadata.subtitle) return true
|
|
|
|
if (filter === 'authors' && !li.media.metadata.authors.length) return true
|
|
|
|
if (filter === 'publishedYear' && !li.media.metadata.publishedYear) return true
|
|
|
|
if (filter === 'series' && !li.media.metadata.series.length) return true
|
|
|
|
if (filter === 'description' && !li.media.metadata.description) return true
|
|
|
|
if (filter === 'genres' && !li.media.metadata.genres.length) return true
|
|
|
|
if (filter === 'tags' && !li.media.tags.length) return true
|
|
|
|
if (filter === 'narrators' && !li.media.metadata.narrators.length) return true
|
|
|
|
if (filter === 'publisher' && !li.media.metadata.publisher) return true
|
|
|
|
if (filter === 'language' && !li.media.metadata.language) return true
|
|
|
|
if (filter === 'cover' && !li.media.coverPath) return true
|
2022-04-26 00:36:18 +02:00
|
|
|
} else {
|
|
|
|
return false
|
|
|
|
}
|
2022-04-18 08:31:39 +02:00
|
|
|
})
|
2022-03-11 01:45:02 +01:00
|
|
|
} else if (group === 'languages') {
|
2023-05-30 23:37:24 +02:00
|
|
|
filtered = filtered.filter(li => li.media.metadata.language === filter)
|
2022-11-28 00:42:02 +01:00
|
|
|
} else if (group === 'tracks') {
|
2023-08-05 21:01:16 +02:00
|
|
|
if (filter === 'none') filtered = filtered.filter(li => li.isBook && !li.media.numTracks)
|
|
|
|
else if (filter === 'single') filtered = filtered.filter(li => li.isBook && li.media.numTracks === 1)
|
2022-11-28 00:42:02 +01:00
|
|
|
else if (filter === 'multi') filtered = filtered.filter(li => li.isBook && li.media.numTracks > 1)
|
2023-06-10 22:59:44 +02:00
|
|
|
} else if (group === 'ebooks') {
|
|
|
|
if (filter === 'ebook') filtered = filtered.filter(li => li.media.ebookFile)
|
|
|
|
else if (filter === 'supplementary') filtered = filtered.filter(li => li.libraryFiles.some(lf => lf.isEBookFile && lf.ino !== li.media.ebookFile?.ino))
|
2022-03-11 01:45:02 +01:00
|
|
|
}
|
|
|
|
} else if (filterBy === 'issues') {
|
2022-04-25 01:05:15 +02:00
|
|
|
filtered = filtered.filter(li => li.hasIssues)
|
2022-08-06 14:58:19 +02:00
|
|
|
} else if (filterBy === 'feed-open') {
|
2023-07-17 23:48:46 +02:00
|
|
|
const libraryItemIdsWithFeed = await Database.models.feed.findAllLibraryItemIds()
|
|
|
|
filtered = filtered.filter(li => libraryItemIdsWithFeed.includes(li.id))
|
2023-03-23 00:05:43 +01:00
|
|
|
} else if (filterBy === 'abridged') {
|
|
|
|
filtered = filtered.filter(li => !!li.media.metadata?.abridged)
|
2023-05-30 23:37:24 +02:00
|
|
|
} else if (filterBy === 'ebook') {
|
|
|
|
filtered = filtered.filter(li => li.media.ebookFile)
|
2022-03-11 01:45:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return filtered
|
|
|
|
},
|
|
|
|
|
2022-10-29 22:33:38 +02:00
|
|
|
// Returns false if should be filtered out
|
|
|
|
checkFilterForSeriesLibraryItem(libraryItem, filterBy) {
|
2023-07-15 19:22:13 +02:00
|
|
|
const searchGroups = ['genres', 'tags', 'authors', 'progress', 'narrators', 'publishers', 'languages']
|
|
|
|
const group = searchGroups.find(_group => filterBy.startsWith(_group + '.'))
|
2022-10-29 22:33:38 +02:00
|
|
|
if (group) {
|
2023-07-15 19:22:13 +02:00
|
|
|
const filterVal = filterBy.replace(`${group}.`, '')
|
|
|
|
const filter = this.decode(filterVal)
|
2022-10-29 22:33:38 +02:00
|
|
|
|
2023-05-30 23:37:24 +02:00
|
|
|
if (group === 'genres') return libraryItem.media.metadata.genres.includes(filter)
|
2022-10-29 22:33:38 +02:00
|
|
|
else if (group === 'tags') return libraryItem.media.tags.includes(filter)
|
2023-05-30 23:37:24 +02:00
|
|
|
else if (group === 'authors') return libraryItem.isBook && libraryItem.media.metadata.hasAuthor(filter)
|
|
|
|
else if (group === 'narrators') return libraryItem.isBook && libraryItem.media.metadata.hasNarrator(filter)
|
2023-07-15 19:22:13 +02:00
|
|
|
else if (group === 'publishers') return libraryItem.isBook && libraryItem.media.metadata.publisher === filter
|
2022-10-29 22:33:38 +02:00
|
|
|
else if (group === 'languages') {
|
2023-05-30 23:37:24 +02:00
|
|
|
return libraryItem.media.metadata.language === filter
|
2022-10-29 22:33:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
},
|
|
|
|
|
|
|
|
// Return false to filter out series
|
|
|
|
checkSeriesProgressFilter(series, filterBy, user) {
|
|
|
|
const filter = this.decode(filterBy.split('.')[1])
|
|
|
|
|
2023-03-04 00:35:14 +01:00
|
|
|
let someBookHasProgress = false
|
|
|
|
let someBookIsUnfinished = false
|
2022-10-29 22:33:38 +02:00
|
|
|
for (const libraryItem of series.books) {
|
|
|
|
const itemProgress = user.getMediaProgress(libraryItem.id)
|
2023-03-04 00:35:14 +01:00
|
|
|
if (!itemProgress || !itemProgress.isFinished) someBookIsUnfinished = true
|
|
|
|
if (itemProgress && itemProgress.progress > 0) someBookHasProgress = true
|
|
|
|
|
|
|
|
if (filter === 'finished' && (!itemProgress || !itemProgress.isFinished)) return false
|
|
|
|
if (filter === 'not-started' && itemProgress) return false
|
2022-10-29 22:33:38 +02:00
|
|
|
}
|
|
|
|
|
2023-06-07 21:01:03 +02:00
|
|
|
if (!someBookIsUnfinished && (filter === 'not-finished' || filter === 'in-progress')) { // Completely finished series
|
2023-03-04 00:35:14 +01:00
|
|
|
return false
|
|
|
|
} else if (!someBookHasProgress && filter === 'in-progress') { // Series not started
|
2022-10-29 22:33:38 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
},
|
|
|
|
|
2023-06-30 00:55:17 +02:00
|
|
|
getSeriesFromBooks(books, allSeries, filterSeries, filterBy, user, minified, hideSingleBookSeries) {
|
2022-10-29 18:17:51 +02:00
|
|
|
const _series = {}
|
2022-10-29 22:33:38 +02:00
|
|
|
const seriesToFilterOut = {}
|
2022-03-13 01:50:31 +01:00
|
|
|
books.forEach((libraryItem) => {
|
2022-10-29 22:33:38 +02:00
|
|
|
// get all book series for item that is not already filtered out
|
|
|
|
const bookSeries = (libraryItem.media.metadata.series || []).filter(se => !seriesToFilterOut[se.id])
|
|
|
|
if (!bookSeries.length) return
|
|
|
|
|
|
|
|
if (filterBy && user && !filterBy.startsWith('progress.')) { // Series progress filters are evaluated after grouping
|
|
|
|
// If a single book in a series is filtered out then filter out the entire series
|
|
|
|
if (!this.checkFilterForSeriesLibraryItem(libraryItem, filterBy)) {
|
|
|
|
// filter out this library item
|
|
|
|
bookSeries.forEach((bookSeriesObj) => {
|
|
|
|
// flag series to filter it out
|
|
|
|
seriesToFilterOut[bookSeriesObj.id] = true
|
|
|
|
delete _series[bookSeriesObj.id]
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-29 18:17:51 +02:00
|
|
|
bookSeries.forEach((bookSeriesObj) => {
|
|
|
|
const series = allSeries.find(se => se.id === bookSeriesObj.id)
|
|
|
|
|
|
|
|
const abJson = minified ? libraryItem.toJSONMinified() : libraryItem.toJSONExpanded()
|
|
|
|
abJson.sequence = bookSeriesObj.sequence
|
2022-10-30 16:38:00 +01:00
|
|
|
if (filterSeries) {
|
|
|
|
abJson.filterSeriesSequence = libraryItem.media.metadata.getSeries(filterSeries).sequence
|
|
|
|
}
|
2022-10-29 18:17:51 +02:00
|
|
|
if (!_series[bookSeriesObj.id]) {
|
|
|
|
_series[bookSeriesObj.id] = {
|
|
|
|
id: bookSeriesObj.id,
|
|
|
|
name: bookSeriesObj.name,
|
2022-11-03 05:14:07 +01:00
|
|
|
nameIgnorePrefix: getTitlePrefixAtEnd(bookSeriesObj.name),
|
|
|
|
nameIgnorePrefixSort: getTitleIgnorePrefix(bookSeriesObj.name),
|
2022-03-27 23:16:08 +02:00
|
|
|
type: 'series',
|
2022-10-29 18:17:51 +02:00
|
|
|
books: [abJson],
|
|
|
|
addedAt: series ? series.addedAt : 0,
|
|
|
|
totalDuration: isNullOrNaN(abJson.media.duration) ? 0 : Number(abJson.media.duration)
|
2021-12-01 03:02:40 +01:00
|
|
|
}
|
2022-10-29 18:17:51 +02:00
|
|
|
|
2022-03-27 23:16:08 +02:00
|
|
|
} else {
|
2022-10-29 18:17:51 +02:00
|
|
|
_series[bookSeriesObj.id].books.push(abJson)
|
|
|
|
_series[bookSeriesObj.id].totalDuration += isNullOrNaN(abJson.media.duration) ? 0 : Number(abJson.media.duration)
|
2022-03-27 23:16:08 +02:00
|
|
|
}
|
|
|
|
})
|
2021-12-01 03:02:40 +01:00
|
|
|
})
|
2022-10-29 22:33:38 +02:00
|
|
|
|
2022-12-31 23:58:19 +01:00
|
|
|
let seriesItems = Object.values(_series)
|
2022-10-29 22:33:38 +02:00
|
|
|
|
2023-06-30 00:55:17 +02:00
|
|
|
// Library setting to hide series with only 1 book
|
|
|
|
if (hideSingleBookSeries) {
|
|
|
|
seriesItems = seriesItems.filter(se => se.books.length > 1)
|
|
|
|
}
|
|
|
|
|
2022-10-29 22:33:38 +02:00
|
|
|
// check progress filter
|
|
|
|
if (filterBy && filterBy.startsWith('progress.') && user) {
|
|
|
|
seriesItems = seriesItems.filter(se => this.checkSeriesProgressFilter(se, filterBy, user))
|
|
|
|
}
|
|
|
|
|
|
|
|
return seriesItems.map((series) => {
|
2022-03-13 01:50:31 +01:00
|
|
|
series.books = naturalSort(series.books).asc(li => li.sequence)
|
2021-12-06 23:18:26 +01:00
|
|
|
return series
|
|
|
|
})
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
|
2023-06-30 00:55:17 +02:00
|
|
|
collapseBookSeries(libraryItems, series, filterSeries, hideSingleBookSeries) {
|
2022-10-30 15:21:12 +01:00
|
|
|
// Get series from the library items. If this list is being collapsed after filtering for a series,
|
|
|
|
// don't collapse that series, only books that are in other series.
|
2022-12-31 17:59:12 +01:00
|
|
|
const seriesObjects = this
|
2023-06-30 00:55:17 +02:00
|
|
|
.getSeriesFromBooks(libraryItems, series, filterSeries, null, null, true, hideSingleBookSeries)
|
2022-10-30 15:21:12 +01:00
|
|
|
.filter(s => s.id != filterSeries)
|
|
|
|
|
2022-12-31 17:59:12 +01:00
|
|
|
const filteredLibraryItems = []
|
2022-04-10 02:44:46 +02:00
|
|
|
|
2022-10-30 03:54:31 +01:00
|
|
|
libraryItems.forEach((li) => {
|
2022-04-10 02:44:46 +02:00
|
|
|
if (li.mediaType != 'book') return
|
2022-10-30 03:54:31 +01:00
|
|
|
|
|
|
|
// Handle when this is the first book in a series
|
|
|
|
seriesObjects.filter(s => s.books[0].id == li.id).forEach(series => {
|
2022-10-30 15:21:12 +01:00
|
|
|
// Clone the library item as we need to attach data to it, but don't
|
|
|
|
// want to change the global copy of the library item
|
|
|
|
filteredLibraryItems.push(Object.assign(
|
|
|
|
Object.create(Object.getPrototypeOf(li)),
|
|
|
|
li, { collapsedSeries: series }))
|
2022-12-31 17:59:12 +01:00
|
|
|
})
|
2022-10-30 03:54:31 +01:00
|
|
|
|
2022-10-30 15:21:12 +01:00
|
|
|
// Only included books not contained in series
|
|
|
|
if (!seriesObjects.some(s => s.books.some(b => b.id == li.id)))
|
|
|
|
filteredLibraryItems.push(li)
|
2022-12-31 17:59:12 +01:00
|
|
|
})
|
2022-10-30 03:54:31 +01:00
|
|
|
|
2022-10-30 15:21:12 +01:00
|
|
|
return filteredLibraryItems
|
2022-04-10 02:44:46 +02:00
|
|
|
}
|
2023-02-19 22:39:28 +01:00
|
|
|
}
|