2021-11-29 02:36:44 +01:00
|
|
|
<template>
|
2024-06-03 08:04:03 +02:00
|
|
|
<div id="bookshelf" ref="bookshelf" class="w-full overflow-y-auto" :style="{ fontSize: sizeMultiplier + 'rem' }">
|
2021-11-29 02:36:44 +01:00
|
|
|
<template v-for="shelf in totalShelves">
|
2024-06-23 18:15:39 +02:00
|
|
|
<div :key="shelf" :id="`shelf-${shelf - 1}`" class="w-full px-4e sm:px-8e relative" :class="{ bookshelfRow: !isAlternativeBookshelfView }" :style="{ height: shelfHeight + 'px' }">
|
2024-12-21 16:42:32 +01:00
|
|
|
<!-- Card skeletons -->
|
|
|
|
<template v-for="entityIndex in entitiesInShelf(shelf)">
|
|
|
|
<div :key="entityIndex" class="w-full h-full absolute rounded z-5 top-0 left-0 bg-primary" :style="{ transform: entityTransform(entityIndex), width: cardWidth + 'px', height: coverHeight + 'px' }" />
|
|
|
|
</template>
|
2024-06-23 18:15:39 +02:00
|
|
|
<div v-if="!isAlternativeBookshelfView" class="bookshelfDivider w-full absolute bottom-0 left-0 right-0 z-20 h-6e" />
|
2021-11-29 02:36:44 +01:00
|
|
|
</div>
|
|
|
|
</template>
|
2021-12-01 03:02:40 +01:00
|
|
|
|
2023-01-04 01:00:01 +01:00
|
|
|
<div v-if="initialized && !totalShelves && !hasFilter && entityName === 'items'" class="w-full flex flex-col items-center justify-center py-12">
|
2023-02-11 22:02:56 +01:00
|
|
|
<p class="text-center text-2xl mb-4 py-4">{{ $getString('MessageXLibraryIsEmpty', [libraryName]) }}</p>
|
2022-05-04 02:16:16 +02:00
|
|
|
<div v-if="userIsAdminOrUp" class="flex">
|
2022-11-19 18:44:08 +01:00
|
|
|
<ui-btn to="/config" color="primary" class="w-52 mr-2">{{ $strings.ButtonConfigureScanner }}</ui-btn>
|
2024-05-13 23:31:30 +02:00
|
|
|
<ui-btn color="success" class="w-52" :loading="isScanningLibrary || tempIsScanning" @click="scan">{{ $strings.ButtonScanLibrary }}</ui-btn>
|
2021-12-02 02:07:03 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div v-else-if="!totalShelves && initialized" class="w-full py-16">
|
2021-12-01 03:02:40 +01:00
|
|
|
<p class="text-xl text-center">{{ emptyMessage }}</p>
|
2022-04-21 15:30:44 +02:00
|
|
|
<!-- Clear filter only available on Library bookshelf -->
|
2023-01-04 01:00:01 +01:00
|
|
|
<div v-if="entityName === 'items'" class="flex justify-center mt-2">
|
2022-11-19 18:44:08 +01:00
|
|
|
<ui-btn v-if="hasFilter" color="primary" @click="clearFilter">{{ $strings.ButtonClearFilter }}</ui-btn>
|
2022-01-10 01:37:16 +01:00
|
|
|
</div>
|
2021-12-01 03:02:40 +01:00
|
|
|
</div>
|
2021-12-02 02:07:03 +01:00
|
|
|
|
2024-06-23 18:15:39 +02:00
|
|
|
<widgets-cover-size-widget class="fixed right-4 z-50" :style="{ bottom: streamLibraryItem ? '181px' : '16px' }" />
|
2021-11-29 02:36:44 +01:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2021-12-01 03:02:40 +01:00
|
|
|
import bookshelfCardsHelpers from '@/mixins/bookshelfCardsHelpers'
|
2021-11-29 02:36:44 +01:00
|
|
|
|
|
|
|
export default {
|
2021-12-01 03:02:40 +01:00
|
|
|
props: {
|
2021-12-02 02:07:03 +01:00
|
|
|
page: String,
|
|
|
|
seriesId: String
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
mixins: [bookshelfCardsHelpers],
|
2021-11-29 02:36:44 +01:00
|
|
|
data() {
|
|
|
|
return {
|
2022-05-19 01:37:38 +02:00
|
|
|
routeFullPath: null,
|
2021-11-29 02:36:44 +01:00
|
|
|
initialized: false,
|
|
|
|
bookshelfHeight: 0,
|
|
|
|
bookshelfWidth: 0,
|
|
|
|
shelvesPerPage: 0,
|
2021-12-01 03:02:40 +01:00
|
|
|
entitiesPerShelf: 8,
|
2021-11-29 02:36:44 +01:00
|
|
|
currentPage: 0,
|
2021-12-01 03:02:40 +01:00
|
|
|
totalEntities: 0,
|
|
|
|
entities: [],
|
2021-11-29 02:36:44 +01:00
|
|
|
pagesLoaded: {},
|
2021-12-01 03:02:40 +01:00
|
|
|
entityIndexesMounted: [],
|
|
|
|
entityComponentRefs: {},
|
2021-12-02 02:07:03 +01:00
|
|
|
currentBookWidth: 0,
|
2021-12-01 03:02:40 +01:00
|
|
|
isFetchingEntities: false,
|
2021-11-29 02:36:44 +01:00
|
|
|
scrollTimeout: null,
|
2024-06-03 08:04:03 +02:00
|
|
|
booksPerFetch: 0,
|
2021-11-29 02:36:44 +01:00
|
|
|
totalShelves: 0,
|
2021-12-01 03:02:40 +01:00
|
|
|
bookshelfMarginLeft: 0,
|
|
|
|
isSelectionMode: false,
|
|
|
|
currentSFQueryString: null,
|
|
|
|
pendingReset: false,
|
2021-12-02 02:07:03 +01:00
|
|
|
keywordFilter: null,
|
2021-12-13 02:48:29 +01:00
|
|
|
currScrollTop: 0,
|
|
|
|
resizeTimeout: null,
|
2022-10-16 00:17:40 +02:00
|
|
|
mountWindowWidth: 0,
|
2024-05-13 23:31:30 +02:00
|
|
|
lastItemIndexSelected: -1,
|
2024-06-03 08:04:03 +02:00
|
|
|
tempIsScanning: false,
|
|
|
|
cardWidth: 0,
|
|
|
|
cardHeight: 0,
|
2024-12-21 16:42:32 +01:00
|
|
|
coverHeight: 0,
|
2024-12-16 18:21:44 +01:00
|
|
|
resizeObserver: null,
|
|
|
|
lastScrollTop: 0,
|
|
|
|
lastTimestamp: 0,
|
|
|
|
postScrollTimeout: null,
|
|
|
|
currFirstEntityIndex: -1,
|
|
|
|
currLastEntityIndex: -1
|
2021-12-02 02:07:03 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
watch: {
|
|
|
|
'$route.query.filter'() {
|
|
|
|
if (this.$route.query.filter && this.$route.query.filter !== this.filterBy) {
|
|
|
|
this.$store.dispatch('user/updateUserSettings', { filterBy: this.$route.query.filter })
|
|
|
|
} else if (!this.$route.query.filter && this.filterBy) {
|
|
|
|
this.$store.dispatch('user/updateUserSettings', { filterBy: 'all' })
|
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
2022-05-04 02:16:16 +02:00
|
|
|
userIsAdminOrUp() {
|
|
|
|
return this.$store.getters['user/getIsAdminOrUp']
|
2021-12-02 02:07:03 +01:00
|
|
|
},
|
2023-01-04 01:00:01 +01:00
|
|
|
libraryMediaType() {
|
|
|
|
return this.$store.getters['libraries/getCurrentLibraryMediaType']
|
|
|
|
},
|
2022-04-12 02:42:09 +02:00
|
|
|
isPodcast() {
|
2023-01-04 01:00:01 +01:00
|
|
|
return this.libraryMediaType === 'podcast'
|
2022-04-12 02:42:09 +02:00
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
emptyMessage() {
|
2022-11-19 18:44:08 +01:00
|
|
|
if (this.page === 'series') return this.$strings.MessageBookshelfNoSeries
|
|
|
|
if (this.page === 'collections') return this.$strings.MessageBookshelfNoCollections
|
2022-11-27 00:24:46 +01:00
|
|
|
if (this.page === 'playlists') return this.$strings.MessageNoUserPlaylists
|
2024-10-06 17:25:08 +02:00
|
|
|
if (this.page === 'authors') return this.$strings.MessageNoAuthors
|
2022-04-24 02:41:06 +02:00
|
|
|
if (this.hasFilter) {
|
2022-11-19 18:44:08 +01:00
|
|
|
if (this.filterName === 'Issues') return this.$strings.MessageNoIssues
|
|
|
|
else if (this.filterName === 'Feed-open') return this.$strings.MessageBookshelfNoRSSFeeds
|
|
|
|
return this.$getString('MessageBookshelfNoResultsForFilter', [this.filterName, this.filterValue])
|
2022-04-24 02:41:06 +02:00
|
|
|
}
|
2022-11-19 18:44:08 +01:00
|
|
|
return this.$strings.MessageNoResults
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
entityName() {
|
2023-01-04 01:00:01 +01:00
|
|
|
if (!this.page) return 'items'
|
2021-12-02 02:07:03 +01:00
|
|
|
return this.page
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
2022-10-29 01:10:19 +02:00
|
|
|
seriesSortBy() {
|
2022-12-17 22:10:25 +01:00
|
|
|
return this.$store.getters['user/getUserSetting']('seriesSortBy')
|
2022-10-29 01:10:19 +02:00
|
|
|
},
|
|
|
|
seriesSortDesc() {
|
2022-12-17 22:10:25 +01:00
|
|
|
return this.$store.getters['user/getUserSetting']('seriesSortDesc')
|
2022-10-29 01:10:19 +02:00
|
|
|
},
|
2022-10-29 22:33:38 +02:00
|
|
|
seriesFilterBy() {
|
2022-12-17 22:10:25 +01:00
|
|
|
return this.$store.getters['user/getUserSetting']('seriesFilterBy')
|
2022-10-29 22:33:38 +02:00
|
|
|
},
|
2024-10-06 17:25:08 +02:00
|
|
|
authorSortBy() {
|
|
|
|
return this.$store.getters['user/getUserSetting']('authorSortBy')
|
|
|
|
},
|
|
|
|
authorSortDesc() {
|
|
|
|
return !!this.$store.getters['user/getUserSetting']('authorSortDesc')
|
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
orderBy() {
|
2021-11-29 02:36:44 +01:00
|
|
|
return this.$store.getters['user/getUserSetting']('orderBy')
|
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
orderDesc() {
|
2021-11-29 02:36:44 +01:00
|
|
|
return this.$store.getters['user/getUserSetting']('orderDesc')
|
|
|
|
},
|
|
|
|
filterBy() {
|
|
|
|
return this.$store.getters['user/getUserSetting']('filterBy')
|
|
|
|
},
|
2022-01-25 01:03:54 +01:00
|
|
|
collapseSeries() {
|
|
|
|
return this.$store.getters['user/getUserSetting']('collapseSeries')
|
|
|
|
},
|
2022-11-03 14:11:33 +01:00
|
|
|
collapseBookSeries() {
|
|
|
|
return this.$store.getters['user/getUserSetting']('collapseBookSeries')
|
|
|
|
},
|
2021-12-02 22:49:03 +01:00
|
|
|
coverAspectRatio() {
|
2022-08-13 20:56:37 +02:00
|
|
|
return this.$store.getters['libraries/getBookCoverAspectRatio']
|
2021-12-02 22:49:03 +01:00
|
|
|
},
|
2022-02-14 23:01:53 +01:00
|
|
|
sortingIgnorePrefix() {
|
|
|
|
return this.$store.getters['getServerSetting']('sortingIgnorePrefix')
|
|
|
|
},
|
2021-12-02 22:49:03 +01:00
|
|
|
isCoverSquareAspectRatio() {
|
2022-08-13 20:56:37 +02:00
|
|
|
return this.coverAspectRatio == 1
|
2021-12-02 22:49:03 +01:00
|
|
|
},
|
2022-08-14 01:18:42 +02:00
|
|
|
bookshelfView() {
|
|
|
|
return this.$store.getters['getBookshelfView']
|
|
|
|
},
|
2021-12-28 22:50:17 +01:00
|
|
|
isAlternativeBookshelfView() {
|
2022-10-22 16:13:20 +02:00
|
|
|
return this.bookshelfView === this.$constants.BookshelfView.DETAIL
|
2021-12-28 22:50:17 +01:00
|
|
|
},
|
2021-12-02 02:07:03 +01:00
|
|
|
hasFilter() {
|
|
|
|
return this.filterBy && this.filterBy !== 'all'
|
|
|
|
},
|
2022-01-10 01:37:16 +01:00
|
|
|
filterName() {
|
|
|
|
if (!this.filterBy) return ''
|
|
|
|
var filter = this.filterBy.split('.')[0]
|
|
|
|
filter = filter.substr(0, 1).toUpperCase() + filter.substr(1)
|
|
|
|
return filter
|
|
|
|
},
|
|
|
|
filterValue() {
|
|
|
|
if (!this.filterBy) return ''
|
|
|
|
if (!this.filterBy.includes('.')) return ''
|
|
|
|
return this.$decode(this.filterBy.split('.')[1])
|
|
|
|
},
|
2021-11-29 02:36:44 +01:00
|
|
|
currentLibraryId() {
|
|
|
|
return this.$store.state.libraries.currentLibraryId
|
|
|
|
},
|
2022-03-27 22:37:04 +02:00
|
|
|
libraryName() {
|
|
|
|
return this.$store.getters['libraries/getCurrentLibraryName']
|
|
|
|
},
|
2021-12-02 02:07:03 +01:00
|
|
|
bookWidth() {
|
2024-06-03 08:04:03 +02:00
|
|
|
return this.cardWidth
|
2021-12-02 22:49:03 +01:00
|
|
|
},
|
|
|
|
bookHeight() {
|
2024-06-03 08:04:03 +02:00
|
|
|
return this.cardHeight
|
2021-12-02 02:07:03 +01:00
|
|
|
},
|
2021-12-18 03:52:11 +01:00
|
|
|
shelfPadding() {
|
2024-06-03 08:04:03 +02:00
|
|
|
if (this.bookshelfWidth < 640) return 32 * this.sizeMultiplier
|
|
|
|
return 64 * this.sizeMultiplier
|
2021-12-18 03:52:11 +01:00
|
|
|
},
|
|
|
|
totalPadding() {
|
|
|
|
return this.shelfPadding * 2
|
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
entityWidth() {
|
2024-06-03 08:04:03 +02:00
|
|
|
return this.cardWidth
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
2021-12-02 22:49:03 +01:00
|
|
|
entityHeight() {
|
2024-06-03 08:04:03 +02:00
|
|
|
return this.cardHeight
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
2024-06-03 08:04:03 +02:00
|
|
|
shelfPaddingHeight() {
|
|
|
|
return 16
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
|
|
|
shelfHeight() {
|
2024-06-03 08:04:03 +02:00
|
|
|
const dividerHeight = this.isAlternativeBookshelfView ? 0 : 24 // h-6
|
|
|
|
return this.cardHeight + (this.shelfPaddingHeight + dividerHeight) * this.sizeMultiplier
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
totalEntityCardWidth() {
|
2021-11-29 02:36:44 +01:00
|
|
|
// Includes margin
|
2024-06-03 08:04:03 +02:00
|
|
|
return this.entityWidth + 24 * this.sizeMultiplier
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
2022-12-01 00:09:00 +01:00
|
|
|
selectedMediaItems() {
|
|
|
|
return this.$store.state.globals.selectedMediaItems || []
|
2021-12-28 22:50:17 +01:00
|
|
|
},
|
|
|
|
sizeMultiplier() {
|
2024-06-03 08:04:03 +02:00
|
|
|
return this.$store.getters['user/getSizeMultiplier']
|
2023-12-23 17:50:04 +01:00
|
|
|
},
|
|
|
|
streamLibraryItem() {
|
|
|
|
return this.$store.state.streamLibraryItem
|
2024-05-13 23:31:30 +02:00
|
|
|
},
|
|
|
|
isScanningLibrary() {
|
|
|
|
return !!this.$store.getters['tasks/getRunningLibraryScanTask'](this.currentLibraryId)
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
methods: {
|
2022-01-10 01:37:16 +01:00
|
|
|
clearFilter() {
|
|
|
|
this.$store.dispatch('user/updateUserSettings', { filterBy: 'all' })
|
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
editEntity(entity) {
|
2023-01-04 01:00:01 +01:00
|
|
|
if (this.entityName === 'items' || this.entityName === 'series-books') {
|
2023-01-03 01:02:04 +01:00
|
|
|
const bookIds = this.entities.map((e) => e.id)
|
2021-12-01 03:02:40 +01:00
|
|
|
this.$store.commit('setBookshelfBookIds', bookIds)
|
|
|
|
this.$store.commit('showEditModal', entity)
|
2021-12-02 02:07:03 +01:00
|
|
|
} else if (this.entityName === 'collections') {
|
|
|
|
this.$store.commit('globals/setEditCollection', entity)
|
2022-11-27 18:53:48 +01:00
|
|
|
} else if (this.entityName === 'playlists') {
|
|
|
|
this.$store.commit('globals/setEditPlaylist', entity)
|
2024-10-06 17:25:08 +02:00
|
|
|
} else if (this.entityName === 'authors') {
|
|
|
|
this.$store.commit('globals/showEditAuthorModal', entity)
|
2021-12-01 03:02:40 +01:00
|
|
|
}
|
|
|
|
},
|
2021-12-02 02:07:03 +01:00
|
|
|
clearSelectedEntities() {
|
2021-12-01 03:02:40 +01:00
|
|
|
this.updateBookSelectionMode(false)
|
|
|
|
this.isSelectionMode = false
|
|
|
|
},
|
2022-10-16 00:17:40 +02:00
|
|
|
selectEntity(entity, shiftKey) {
|
2023-01-04 01:00:01 +01:00
|
|
|
if (this.entityName === 'items' || this.entityName === 'series-books') {
|
2022-12-02 00:39:23 +01:00
|
|
|
const indexOf = this.entities.findIndex((ent) => ent && ent.id === entity.id)
|
2022-10-16 00:17:40 +02:00
|
|
|
const lastLastItemIndexSelected = this.lastItemIndexSelected
|
2022-12-01 00:09:00 +01:00
|
|
|
if (!this.selectedMediaItems.some((i) => i.id === entity.id)) {
|
2022-10-16 00:17:40 +02:00
|
|
|
this.lastItemIndexSelected = indexOf
|
|
|
|
} else {
|
|
|
|
this.lastItemIndexSelected = -1
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shiftKey && lastLastItemIndexSelected >= 0) {
|
2022-12-02 00:39:23 +01:00
|
|
|
let loopStart = indexOf
|
|
|
|
let loopEnd = lastLastItemIndexSelected
|
2022-10-16 00:17:40 +02:00
|
|
|
if (indexOf > lastLastItemIndexSelected) {
|
|
|
|
loopStart = lastLastItemIndexSelected
|
|
|
|
loopEnd = indexOf
|
|
|
|
}
|
|
|
|
|
2022-12-02 00:39:23 +01:00
|
|
|
let isSelecting = false
|
2022-10-16 00:17:40 +02:00
|
|
|
// If any items in this range is not selected then select all otherwise unselect all
|
|
|
|
for (let i = loopStart; i <= loopEnd; i++) {
|
|
|
|
const thisEntity = this.entities[i]
|
|
|
|
if (thisEntity && !thisEntity.collapsedSeries) {
|
2022-12-01 00:09:00 +01:00
|
|
|
if (!this.selectedMediaItems.some((i) => i.id === thisEntity.id)) {
|
2022-10-16 00:17:40 +02:00
|
|
|
isSelecting = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (isSelecting) this.lastItemIndexSelected = indexOf
|
|
|
|
|
|
|
|
for (let i = loopStart; i <= loopEnd; i++) {
|
|
|
|
const thisEntity = this.entities[i]
|
|
|
|
if (thisEntity.collapsedSeries) {
|
|
|
|
console.warn('Ignoring collapsed series')
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
const entityComponentRef = this.entityComponentRefs[i]
|
|
|
|
if (thisEntity && entityComponentRef) {
|
|
|
|
entityComponentRef.selected = isSelecting
|
2022-12-01 00:09:00 +01:00
|
|
|
|
|
|
|
const mediaItem = {
|
|
|
|
id: thisEntity.id,
|
|
|
|
mediaType: thisEntity.mediaType,
|
2023-01-04 01:00:01 +01:00
|
|
|
hasTracks: thisEntity.mediaType === 'podcast' || thisEntity.media.audioFile || thisEntity.media.numTracks || (thisEntity.media.tracks && thisEntity.media.tracks.length)
|
2022-12-01 00:09:00 +01:00
|
|
|
}
|
|
|
|
this.$store.commit('globals/setMediaItemSelected', { item: mediaItem, selected: isSelecting })
|
2022-10-16 00:17:40 +02:00
|
|
|
} else {
|
|
|
|
console.error('Invalid entity index', i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2022-12-01 00:09:00 +01:00
|
|
|
const mediaItem = {
|
|
|
|
id: entity.id,
|
|
|
|
mediaType: entity.mediaType,
|
2023-01-04 01:00:01 +01:00
|
|
|
hasTracks: entity.mediaType === 'podcast' || entity.media.audioFile || entity.media.numTracks || (entity.media.tracks && entity.media.tracks.length)
|
2022-12-01 00:09:00 +01:00
|
|
|
}
|
|
|
|
this.$store.commit('globals/toggleMediaItemSelected', mediaItem)
|
2022-10-16 00:17:40 +02:00
|
|
|
}
|
2021-12-01 03:02:40 +01:00
|
|
|
|
2022-12-01 00:09:00 +01:00
|
|
|
const newIsSelectionMode = !!this.selectedMediaItems.length
|
2021-12-01 03:02:40 +01:00
|
|
|
if (this.isSelectionMode !== newIsSelectionMode) {
|
|
|
|
this.isSelectionMode = newIsSelectionMode
|
|
|
|
this.updateBookSelectionMode(newIsSelectionMode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
updateBookSelectionMode(isSelectionMode) {
|
|
|
|
for (const key in this.entityComponentRefs) {
|
|
|
|
if (this.entityIndexesMounted.includes(Number(key))) {
|
|
|
|
this.entityComponentRefs[key].setSelectionMode(isSelectionMode)
|
|
|
|
}
|
|
|
|
}
|
2022-10-16 00:17:40 +02:00
|
|
|
if (!isSelectionMode) {
|
|
|
|
this.lastItemIndexSelected = -1
|
|
|
|
}
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
async fetchEntites(page = 0) {
|
2023-01-03 01:02:04 +01:00
|
|
|
const startIndex = page * this.booksPerFetch
|
2021-11-29 02:36:44 +01:00
|
|
|
|
2021-12-01 03:02:40 +01:00
|
|
|
this.isFetchingEntities = true
|
|
|
|
|
|
|
|
if (!this.initialized) {
|
|
|
|
this.currentSFQueryString = this.buildSearchParams()
|
|
|
|
}
|
|
|
|
|
2023-07-31 00:51:44 +02:00
|
|
|
let entityPath = this.entityName === 'series-books' ? 'items' : this.entityName
|
2022-11-27 00:24:46 +01:00
|
|
|
const sfQueryString = this.currentSFQueryString ? this.currentSFQueryString + '&' : ''
|
2024-07-02 00:26:13 +02:00
|
|
|
const fullQueryString = `?${sfQueryString}limit=${this.booksPerFetch}&page=${page}&minified=1&include=rssfeed,numEpisodesIncomplete,share`
|
2021-12-26 21:40:00 +01:00
|
|
|
|
2022-11-27 00:24:46 +01:00
|
|
|
const payload = await this.$axios.$get(`/api/libraries/${this.currentLibraryId}/${entityPath}${fullQueryString}`).catch((error) => {
|
2023-07-15 21:45:08 +02:00
|
|
|
console.error('failed to fetch items', error)
|
2021-11-29 02:36:44 +01:00
|
|
|
return null
|
|
|
|
})
|
2022-02-13 22:00:59 +01:00
|
|
|
|
2021-12-01 03:02:40 +01:00
|
|
|
this.isFetchingEntities = false
|
|
|
|
if (this.pendingReset) {
|
|
|
|
this.pendingReset = false
|
|
|
|
this.resetEntities()
|
|
|
|
return
|
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
if (payload) {
|
|
|
|
if (!this.initialized) {
|
|
|
|
this.initialized = true
|
2021-12-01 03:02:40 +01:00
|
|
|
this.totalEntities = payload.total
|
|
|
|
this.totalShelves = Math.ceil(this.totalEntities / this.entitiesPerShelf)
|
|
|
|
this.entities = new Array(this.totalEntities)
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < payload.results.length; i++) {
|
2022-12-31 17:33:38 +01:00
|
|
|
const index = i + startIndex
|
2021-12-01 03:02:40 +01:00
|
|
|
this.entities[index] = payload.results[i]
|
|
|
|
if (this.entityComponentRefs[index]) {
|
|
|
|
this.entityComponentRefs[index].setEntity(this.entities[index])
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
}
|
2022-10-30 16:38:00 +01:00
|
|
|
|
2022-11-13 20:25:20 +01:00
|
|
|
this.$eventBus.$emit('bookshelf-total-entities', this.totalEntities)
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
loadPage(page) {
|
2024-12-16 18:21:44 +01:00
|
|
|
if (!this.pagesLoaded[page]) this.pagesLoaded[page] = this.fetchEntites(page)
|
|
|
|
return this.pagesLoaded[page]
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
|
|
|
showHideBookPlaceholder(index, show) {
|
|
|
|
var el = document.getElementById(`book-${index}-placeholder`)
|
|
|
|
if (el) el.style.display = show ? 'flex' : 'none'
|
|
|
|
},
|
2024-12-16 18:21:44 +01:00
|
|
|
mountEntities(fromIndex, toIndex) {
|
2021-11-29 02:36:44 +01:00
|
|
|
for (let i = fromIndex; i < toIndex; i++) {
|
2021-12-01 03:02:40 +01:00
|
|
|
if (!this.entityIndexesMounted.includes(i)) {
|
|
|
|
this.cardsHelpers.mountEntityCard(i)
|
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
},
|
2024-12-16 18:21:44 +01:00
|
|
|
getVisibleIndices(scrollTop) {
|
|
|
|
const firstShelfIndex = Math.floor(scrollTop / this.shelfHeight)
|
|
|
|
const lastShelfIndex = Math.min(Math.ceil((scrollTop + this.bookshelfHeight) / this.shelfHeight), this.totalShelves - 1)
|
|
|
|
const firstEntityIndex = firstShelfIndex * this.entitiesPerShelf
|
|
|
|
const lastEntityIndex = Math.min(lastShelfIndex * this.entitiesPerShelf + this.entitiesPerShelf, this.totalEntities)
|
|
|
|
return { firstEntityIndex, lastEntityIndex }
|
|
|
|
},
|
|
|
|
postScroll() {
|
|
|
|
const { firstEntityIndex, lastEntityIndex } = this.getVisibleIndices(this.currScrollTop)
|
2021-12-01 03:02:40 +01:00
|
|
|
this.entityIndexesMounted = this.entityIndexesMounted.filter((_index) => {
|
2024-12-16 18:21:44 +01:00
|
|
|
if (_index < firstEntityIndex || _index >= lastEntityIndex) {
|
|
|
|
var el = this.entityComponentRefs[_index]
|
|
|
|
if (el && el.$el) el.$el.remove()
|
2021-11-29 02:36:44 +01:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
2024-12-16 18:21:44 +01:00
|
|
|
},
|
|
|
|
handleScroll(scrollTop) {
|
|
|
|
this.currScrollTop = scrollTop
|
|
|
|
const { firstEntityIndex, lastEntityIndex } = this.getVisibleIndices(scrollTop)
|
|
|
|
if (firstEntityIndex === this.currFirstEntityIndex && lastEntityIndex === this.currLastEntityIndex) return
|
|
|
|
this.currFirstEntityIndex = firstEntityIndex
|
|
|
|
this.currLastEntityIndex = lastEntityIndex
|
|
|
|
|
|
|
|
clearTimeout(this.postScrollTimeout)
|
|
|
|
const firstPage = Math.floor(firstEntityIndex / this.booksPerFetch)
|
|
|
|
const lastPage = Math.floor(lastEntityIndex / this.booksPerFetch)
|
|
|
|
Promise.all([this.loadPage(firstPage), this.loadPage(lastPage)])
|
|
|
|
.then(() => this.mountEntities(firstEntityIndex, lastEntityIndex))
|
|
|
|
.catch((error) => console.error('Failed to load page', error))
|
|
|
|
|
|
|
|
this.postScrollTimeout = setTimeout(this.postScroll, 500)
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
async resetEntities() {
|
|
|
|
if (this.isFetchingEntities) {
|
|
|
|
this.pendingReset = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this.destroyEntityComponents()
|
|
|
|
this.pagesLoaded = {}
|
|
|
|
this.entities = []
|
|
|
|
this.totalShelves = 0
|
|
|
|
this.totalEntities = 0
|
|
|
|
this.currentPage = 0
|
|
|
|
this.isSelectionMode = false
|
|
|
|
this.initialized = false
|
|
|
|
|
2021-12-02 02:07:03 +01:00
|
|
|
this.initSizeData()
|
2024-12-16 18:21:44 +01:00
|
|
|
await this.loadPage(0)
|
2021-12-01 03:02:40 +01:00
|
|
|
var lastBookIndex = Math.min(this.totalEntities, this.shelvesPerPage * this.entitiesPerShelf)
|
2024-12-16 18:21:44 +01:00
|
|
|
this.mountEntities(0, lastBookIndex)
|
2021-12-02 02:07:03 +01:00
|
|
|
},
|
2024-12-16 18:21:44 +01:00
|
|
|
async rebuild() {
|
2021-12-13 02:48:29 +01:00
|
|
|
this.initSizeData()
|
|
|
|
|
2024-06-03 08:04:03 +02:00
|
|
|
var lastBookIndex = Math.min(this.totalEntities, this.booksPerFetch)
|
2024-12-16 18:21:44 +01:00
|
|
|
this.destroyEntityComponents()
|
|
|
|
await this.loadPage(0)
|
2021-12-13 02:48:29 +01:00
|
|
|
var bookshelfEl = document.getElementById('bookshelf')
|
|
|
|
if (bookshelfEl) {
|
|
|
|
bookshelfEl.scrollTop = 0
|
|
|
|
}
|
2024-12-16 18:21:44 +01:00
|
|
|
this.mountEntities(0, lastBookIndex)
|
2021-12-13 02:48:29 +01:00
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
buildSearchParams() {
|
2022-10-29 01:10:19 +02:00
|
|
|
if (this.page === 'search' || this.page === 'collections') {
|
2021-12-01 03:02:40 +01:00
|
|
|
return ''
|
|
|
|
}
|
|
|
|
|
|
|
|
let searchParams = new URLSearchParams()
|
2022-10-29 01:10:19 +02:00
|
|
|
if (this.page === 'series') {
|
|
|
|
searchParams.set('sort', this.seriesSortBy)
|
|
|
|
searchParams.set('desc', this.seriesSortDesc ? 1 : 0)
|
2022-10-29 22:33:38 +02:00
|
|
|
searchParams.set('filter', this.seriesFilterBy)
|
2022-10-29 01:10:19 +02:00
|
|
|
} else if (this.page === 'series-books') {
|
2022-03-13 01:50:31 +01:00
|
|
|
searchParams.set('filter', `series.${this.$encode(this.seriesId)}`)
|
2022-11-03 14:11:33 +01:00
|
|
|
if (this.collapseBookSeries) {
|
|
|
|
searchParams.set('collapseseries', 1)
|
|
|
|
}
|
2024-10-06 17:25:08 +02:00
|
|
|
} else if (this.page === 'authors') {
|
|
|
|
searchParams.set('sort', this.authorSortBy)
|
|
|
|
searchParams.set('desc', this.authorSortDesc ? 1 : 0)
|
2021-12-26 21:40:00 +01:00
|
|
|
} else {
|
|
|
|
if (this.filterBy && this.filterBy !== 'all') {
|
|
|
|
searchParams.set('filter', this.filterBy)
|
|
|
|
}
|
|
|
|
if (this.orderBy) {
|
|
|
|
searchParams.set('sort', this.orderBy)
|
|
|
|
searchParams.set('desc', this.orderDesc ? 1 : 0)
|
|
|
|
}
|
2022-04-12 02:42:09 +02:00
|
|
|
if (this.collapseSeries && !this.isPodcast) {
|
2022-01-25 01:03:54 +01:00
|
|
|
searchParams.set('collapseseries', 1)
|
|
|
|
}
|
2021-12-01 03:02:40 +01:00
|
|
|
}
|
|
|
|
return searchParams.toString()
|
|
|
|
},
|
|
|
|
checkUpdateSearchParams() {
|
|
|
|
var newSearchParams = this.buildSearchParams()
|
|
|
|
var currentQueryString = window.location.search
|
2021-12-02 02:07:03 +01:00
|
|
|
if (currentQueryString && currentQueryString.startsWith('?')) currentQueryString = currentQueryString.slice(1)
|
2021-12-01 03:02:40 +01:00
|
|
|
|
|
|
|
if (newSearchParams === '') {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if (newSearchParams !== this.currentSFQueryString || newSearchParams !== currentQueryString) {
|
|
|
|
let newurl = window.location.protocol + '//' + window.location.host + window.location.pathname + '?' + newSearchParams
|
|
|
|
window.history.replaceState({ path: newurl }, '', newurl)
|
2022-05-19 01:37:38 +02:00
|
|
|
|
|
|
|
this.routeFullPath = window.location.pathname + (window.location.search || '') // Update for saving scroll position
|
2021-12-01 03:02:40 +01:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
},
|
2022-10-29 01:10:19 +02:00
|
|
|
seriesSortUpdated() {
|
|
|
|
var wasUpdated = this.checkUpdateSearchParams()
|
|
|
|
if (wasUpdated) {
|
|
|
|
this.resetEntities()
|
|
|
|
}
|
|
|
|
},
|
2024-06-03 08:04:03 +02:00
|
|
|
async settingsUpdated(settings) {
|
|
|
|
await this.cardsHelpers.setCardSize()
|
2022-12-17 22:10:25 +01:00
|
|
|
const wasUpdated = this.checkUpdateSearchParams()
|
2021-12-01 03:02:40 +01:00
|
|
|
if (wasUpdated) {
|
|
|
|
this.resetEntities()
|
2021-12-02 02:07:03 +01:00
|
|
|
} else if (settings.bookshelfCoverSize !== this.currentBookWidth) {
|
2024-12-16 18:21:44 +01:00
|
|
|
this.rebuild()
|
2021-12-01 03:02:40 +01:00
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
2024-12-16 18:21:44 +01:00
|
|
|
getScrollRate() {
|
|
|
|
const currentTimestamp = Date.now()
|
|
|
|
const timeDelta = currentTimestamp - this.lastTimestamp
|
|
|
|
const scrollDelta = this.currScrollTop - this.lastScrollTop
|
|
|
|
const scrollRate = Math.abs(scrollDelta) / (timeDelta || 1)
|
|
|
|
this.lastScrollTop = this.currScrollTop
|
|
|
|
this.lastTimestamp = currentTimestamp
|
|
|
|
return scrollRate
|
|
|
|
},
|
2021-11-29 02:36:44 +01:00
|
|
|
scroll(e) {
|
|
|
|
if (!e || !e.target) return
|
2024-12-16 18:21:44 +01:00
|
|
|
clearTimeout(this.scrollTimeout)
|
|
|
|
const { scrollTop } = e.target
|
|
|
|
const scrollRate = this.getScrollRate()
|
|
|
|
if (scrollRate > 5) {
|
|
|
|
this.scrollTimeout = setTimeout(() => {
|
|
|
|
this.handleScroll(scrollTop)
|
|
|
|
}, 25)
|
|
|
|
return
|
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
this.handleScroll(scrollTop)
|
|
|
|
},
|
2022-03-13 00:45:32 +01:00
|
|
|
libraryItemAdded(libraryItem) {
|
|
|
|
console.log('libraryItem added', libraryItem)
|
2021-12-02 02:07:03 +01:00
|
|
|
// TODO: Check if audiobook would be on this shelf
|
|
|
|
this.resetEntities()
|
|
|
|
},
|
2022-03-13 00:45:32 +01:00
|
|
|
libraryItemUpdated(libraryItem) {
|
|
|
|
console.log('Item updated', libraryItem)
|
2023-01-04 01:00:01 +01:00
|
|
|
if (this.entityName === 'items' || this.entityName === 'series-books') {
|
2022-03-13 00:45:32 +01:00
|
|
|
var indexOf = this.entities.findIndex((ent) => ent && ent.id === libraryItem.id)
|
2021-12-02 02:07:03 +01:00
|
|
|
if (indexOf >= 0) {
|
2022-03-13 00:45:32 +01:00
|
|
|
this.entities[indexOf] = libraryItem
|
2021-12-02 02:07:03 +01:00
|
|
|
if (this.entityComponentRefs[indexOf]) {
|
2022-03-13 00:45:32 +01:00
|
|
|
this.entityComponentRefs[indexOf].setEntity(libraryItem)
|
2021-12-02 02:07:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2022-03-13 00:45:32 +01:00
|
|
|
libraryItemRemoved(libraryItem) {
|
2023-01-04 01:00:01 +01:00
|
|
|
if (this.entityName === 'items' || this.entityName === 'series-books') {
|
2022-03-13 00:45:32 +01:00
|
|
|
var indexOf = this.entities.findIndex((ent) => ent && ent.id === libraryItem.id)
|
2021-12-02 02:07:03 +01:00
|
|
|
if (indexOf >= 0) {
|
2022-03-13 00:45:32 +01:00
|
|
|
this.entities = this.entities.filter((ent) => ent.id !== libraryItem.id)
|
2022-11-13 20:25:20 +01:00
|
|
|
this.totalEntities--
|
|
|
|
this.$eventBus.$emit('bookshelf-total-entities', this.totalEntities)
|
2022-03-13 00:45:32 +01:00
|
|
|
this.executeRebuild()
|
2021-12-02 02:07:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2022-03-13 00:45:32 +01:00
|
|
|
libraryItemsAdded(libraryItems) {
|
|
|
|
console.log('items added', libraryItems)
|
2021-12-02 02:07:03 +01:00
|
|
|
// TODO: Check if audiobook would be on this shelf
|
|
|
|
this.resetEntities()
|
|
|
|
},
|
2022-03-13 00:45:32 +01:00
|
|
|
libraryItemsUpdated(libraryItems) {
|
|
|
|
libraryItems.forEach((ab) => {
|
|
|
|
this.libraryItemUpdated(ab)
|
2021-12-02 02:07:03 +01:00
|
|
|
})
|
|
|
|
},
|
2022-04-21 15:30:44 +02:00
|
|
|
collectionAdded(collection) {
|
|
|
|
if (this.entityName !== 'collections') return
|
|
|
|
console.log(`[LazyBookshelf] collectionAdded ${collection.id}`, collection)
|
|
|
|
this.resetEntities()
|
|
|
|
},
|
|
|
|
collectionUpdated(collection) {
|
|
|
|
if (this.entityName !== 'collections') return
|
|
|
|
console.log(`[LazyBookshelf] collectionUpdated ${collection.id}`, collection)
|
|
|
|
var indexOf = this.entities.findIndex((ent) => ent && ent.id === collection.id)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
this.entities[indexOf] = collection
|
|
|
|
if (this.entityComponentRefs[indexOf]) {
|
|
|
|
this.entityComponentRefs[indexOf].setEntity(collection)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
collectionRemoved(collection) {
|
|
|
|
if (this.entityName !== 'collections') return
|
|
|
|
console.log(`[LazyBookshelf] collectionRemoved ${collection.id}`, collection)
|
|
|
|
var indexOf = this.entities.findIndex((ent) => ent && ent.id === collection.id)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
this.entities = this.entities.filter((ent) => ent.id !== collection.id)
|
2022-11-13 20:25:20 +01:00
|
|
|
this.totalEntities--
|
|
|
|
this.$eventBus.$emit('bookshelf-total-entities', this.totalEntities)
|
2022-04-21 15:30:44 +02:00
|
|
|
this.executeRebuild()
|
|
|
|
}
|
|
|
|
},
|
2022-11-27 20:38:08 +01:00
|
|
|
playlistAdded(playlist) {
|
|
|
|
if (this.entityName !== 'playlists') return
|
|
|
|
console.log(`[LazyBookshelf] playlistAdded ${playlist.id}`, playlist)
|
|
|
|
this.resetEntities()
|
|
|
|
},
|
|
|
|
playlistUpdated(playlist) {
|
|
|
|
if (this.entityName !== 'playlists') return
|
|
|
|
console.log(`[LazyBookshelf] playlistUpdated ${playlist.id}`, playlist)
|
|
|
|
var indexOf = this.entities.findIndex((ent) => ent && ent.id === playlist.id)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
this.entities[indexOf] = playlist
|
|
|
|
if (this.entityComponentRefs[indexOf]) {
|
|
|
|
this.entityComponentRefs[indexOf].setEntity(playlist)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
playlistRemoved(playlist) {
|
|
|
|
if (this.entityName !== 'playlists') return
|
|
|
|
console.log(`[LazyBookshelf] playlistRemoved ${playlist.id}`, playlist)
|
|
|
|
var indexOf = this.entities.findIndex((ent) => ent && ent.id === playlist.id)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
this.entities = this.entities.filter((ent) => ent.id !== playlist.id)
|
|
|
|
this.totalEntities--
|
|
|
|
this.$eventBus.$emit('bookshelf-total-entities', this.totalEntities)
|
|
|
|
this.executeRebuild()
|
|
|
|
}
|
|
|
|
},
|
2024-10-06 17:25:08 +02:00
|
|
|
authorAdded(author) {
|
|
|
|
if (this.entityName !== 'authors') return
|
|
|
|
console.log(`[LazyBookshelf] authorAdded ${author.id}`, author)
|
|
|
|
this.resetEntities()
|
|
|
|
},
|
|
|
|
authorUpdated(author) {
|
|
|
|
if (this.entityName !== 'authors') return
|
|
|
|
console.log(`[LazyBookshelf] authorUpdated ${author.id}`, author)
|
|
|
|
const indexOf = this.entities.findIndex((ent) => ent && ent.id === author.id)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
this.entities[indexOf] = author
|
|
|
|
if (this.entityComponentRefs[indexOf]) {
|
|
|
|
this.entityComponentRefs[indexOf].setEntity(author)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
authorRemoved(author) {
|
|
|
|
if (this.entityName !== 'authors') return
|
|
|
|
console.log(`[LazyBookshelf] authorRemoved ${author.id}`, author)
|
|
|
|
const indexOf = this.entities.findIndex((ent) => ent && ent.id === author.id)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
this.entities = this.entities.filter((ent) => ent.id !== author.id)
|
|
|
|
this.totalEntities--
|
|
|
|
this.$eventBus.$emit('bookshelf-total-entities', this.totalEntities)
|
|
|
|
this.executeRebuild()
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2024-07-07 22:51:50 +02:00
|
|
|
shareOpen(mediaItemShare) {
|
|
|
|
if (this.entityName === 'items' || this.entityName === 'series-books') {
|
|
|
|
var indexOf = this.entities.findIndex((ent) => ent?.media?.id === mediaItemShare.mediaItemId)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
if (this.entityComponentRefs[indexOf]) {
|
|
|
|
const libraryItem = { ...this.entityComponentRefs[indexOf].libraryItem }
|
|
|
|
libraryItem.mediaItemShare = mediaItemShare
|
|
|
|
this.entityComponentRefs[indexOf].setEntity?.(libraryItem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
shareClosed(mediaItemShare) {
|
|
|
|
if (this.entityName === 'items' || this.entityName === 'series-books') {
|
|
|
|
var indexOf = this.entities.findIndex((ent) => ent?.media?.id === mediaItemShare.mediaItemId)
|
|
|
|
if (indexOf >= 0) {
|
|
|
|
if (this.entityComponentRefs[indexOf]) {
|
|
|
|
const libraryItem = { ...this.entityComponentRefs[indexOf].libraryItem }
|
|
|
|
libraryItem.mediaItemShare = null
|
|
|
|
this.entityComponentRefs[indexOf].setEntity?.(libraryItem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2024-06-03 08:04:03 +02:00
|
|
|
updatePagesLoaded() {
|
|
|
|
let numPages = Math.ceil(this.totalEntities / this.booksPerFetch)
|
2024-12-16 18:21:44 +01:00
|
|
|
this.pagesLoaded = {}
|
2024-06-03 08:04:03 +02:00
|
|
|
for (let page = 0; page < numPages; page++) {
|
|
|
|
let numEntities = Math.min(this.totalEntities - page * this.booksPerFetch, this.booksPerFetch)
|
2024-12-16 18:21:44 +01:00
|
|
|
this.pagesLoaded[page] = Promise.resolve()
|
2024-06-03 08:04:03 +02:00
|
|
|
for (let i = 0; i < numEntities; i++) {
|
|
|
|
const index = page * this.booksPerFetch + i
|
|
|
|
if (!this.entities[index]) {
|
2024-12-16 18:21:44 +01:00
|
|
|
if (this.pagesLoaded[page]) delete this.pagesLoaded[page]
|
2024-06-03 08:04:03 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2021-12-02 02:07:03 +01:00
|
|
|
initSizeData(_bookshelf) {
|
|
|
|
var bookshelf = _bookshelf || document.getElementById('bookshelf')
|
|
|
|
if (!bookshelf) {
|
|
|
|
console.error('Failed to init size data')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var entitiesPerShelfBefore = this.entitiesPerShelf
|
2021-12-01 03:02:40 +01:00
|
|
|
|
2021-11-29 02:36:44 +01:00
|
|
|
var { clientHeight, clientWidth } = bookshelf
|
2021-12-13 02:48:29 +01:00
|
|
|
this.mountWindowWidth = window.innerWidth
|
2021-11-29 02:36:44 +01:00
|
|
|
this.bookshelfHeight = clientHeight
|
|
|
|
this.bookshelfWidth = clientWidth
|
2021-12-18 03:52:11 +01:00
|
|
|
this.entitiesPerShelf = Math.max(1, Math.floor((this.bookshelfWidth - this.shelfPadding) / this.totalEntityCardWidth))
|
2021-11-29 02:36:44 +01:00
|
|
|
this.shelvesPerPage = Math.ceil(this.bookshelfHeight / this.shelfHeight) + 2
|
2021-12-01 03:02:40 +01:00
|
|
|
this.bookshelfMarginLeft = (this.bookshelfWidth - this.entitiesPerShelf * this.totalEntityCardWidth) / 2
|
2024-06-03 08:04:03 +02:00
|
|
|
const booksPerFetch = this.entitiesPerShelf * this.shelvesPerPage
|
|
|
|
if (booksPerFetch !== this.booksPerFetch) {
|
|
|
|
this.booksPerFetch = booksPerFetch
|
|
|
|
if (this.totalEntities) {
|
|
|
|
this.updatePagesLoaded()
|
|
|
|
}
|
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
|
2021-12-02 02:07:03 +01:00
|
|
|
this.currentBookWidth = this.bookWidth
|
|
|
|
if (this.totalEntities) {
|
|
|
|
this.totalShelves = Math.ceil(this.totalEntities / this.entitiesPerShelf)
|
|
|
|
}
|
|
|
|
return entitiesPerShelfBefore < this.entitiesPerShelf // Books per shelf has changed
|
|
|
|
},
|
|
|
|
async init(bookshelf) {
|
|
|
|
this.initSizeData(bookshelf)
|
2024-06-03 08:04:03 +02:00
|
|
|
this.checkUpdateSearchParams()
|
2021-12-02 02:07:03 +01:00
|
|
|
|
2024-12-16 18:21:44 +01:00
|
|
|
await this.loadPage(0)
|
2021-12-01 03:02:40 +01:00
|
|
|
var lastBookIndex = Math.min(this.totalEntities, this.shelvesPerPage * this.entitiesPerShelf)
|
2024-12-16 18:21:44 +01:00
|
|
|
this.mountEntities(0, lastBookIndex)
|
2022-05-19 01:37:38 +02:00
|
|
|
|
|
|
|
// Set last scroll position for this bookshelf page
|
|
|
|
if (this.$store.state.lastBookshelfScrollData[this.page] && window.bookshelf) {
|
|
|
|
const { path, scrollTop } = this.$store.state.lastBookshelfScrollData[this.page]
|
|
|
|
if (path === this.routeFullPath) {
|
|
|
|
// Exact path match with query so use scroll position
|
|
|
|
window.bookshelf.scrollTop = scrollTop
|
|
|
|
}
|
|
|
|
}
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
2021-12-21 02:26:22 +01:00
|
|
|
executeRebuild() {
|
2021-12-13 02:48:29 +01:00
|
|
|
clearTimeout(this.resizeTimeout)
|
|
|
|
this.resizeTimeout = setTimeout(() => {
|
|
|
|
this.rebuild()
|
|
|
|
}, 200)
|
|
|
|
},
|
2021-12-21 02:26:22 +01:00
|
|
|
windowResize() {
|
|
|
|
this.executeRebuild()
|
|
|
|
},
|
|
|
|
socketInit() {
|
|
|
|
// Server settings are set on socket init
|
|
|
|
this.executeRebuild()
|
|
|
|
},
|
2021-12-01 03:02:40 +01:00
|
|
|
initListeners() {
|
2021-12-13 02:48:29 +01:00
|
|
|
window.addEventListener('resize', this.windowResize)
|
|
|
|
|
|
|
|
this.$nextTick(() => {
|
|
|
|
var bookshelf = document.getElementById('bookshelf')
|
|
|
|
if (bookshelf) {
|
|
|
|
this.init(bookshelf)
|
2024-12-16 18:21:44 +01:00
|
|
|
bookshelf.addEventListener('scroll', this.scroll, { passive: true })
|
2021-12-13 02:48:29 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-10-29 01:10:19 +02:00
|
|
|
this.$eventBus.$on('bookshelf_clear_selection', this.clearSelectedEntities)
|
2021-12-21 02:26:22 +01:00
|
|
|
this.$eventBus.$on('socket_init', this.socketInit)
|
2022-12-17 21:50:01 +01:00
|
|
|
this.$eventBus.$on('user-settings', this.settingsUpdated)
|
2021-12-02 02:07:03 +01:00
|
|
|
|
|
|
|
if (this.$root.socket) {
|
2022-03-13 00:45:32 +01:00
|
|
|
this.$root.socket.on('item_updated', this.libraryItemUpdated)
|
|
|
|
this.$root.socket.on('item_added', this.libraryItemAdded)
|
|
|
|
this.$root.socket.on('item_removed', this.libraryItemRemoved)
|
|
|
|
this.$root.socket.on('items_updated', this.libraryItemsUpdated)
|
|
|
|
this.$root.socket.on('items_added', this.libraryItemsAdded)
|
2022-04-21 15:30:44 +02:00
|
|
|
this.$root.socket.on('collection_added', this.collectionAdded)
|
|
|
|
this.$root.socket.on('collection_updated', this.collectionUpdated)
|
|
|
|
this.$root.socket.on('collection_removed', this.collectionRemoved)
|
2022-11-27 20:38:08 +01:00
|
|
|
this.$root.socket.on('playlist_added', this.playlistAdded)
|
|
|
|
this.$root.socket.on('playlist_updated', this.playlistUpdated)
|
|
|
|
this.$root.socket.on('playlist_removed', this.playlistRemoved)
|
2024-10-06 17:25:08 +02:00
|
|
|
this.$root.socket.on('author_added', this.authorAdded)
|
|
|
|
this.$root.socket.on('author_updated', this.authorUpdated)
|
|
|
|
this.$root.socket.on('author_removed', this.authorRemoved)
|
2024-07-07 22:51:50 +02:00
|
|
|
this.$root.socket.on('share_open', this.shareOpen)
|
|
|
|
this.$root.socket.on('share_closed', this.shareClosed)
|
2021-12-02 02:07:03 +01:00
|
|
|
} else {
|
|
|
|
console.error('Bookshelf - Socket not initialized')
|
|
|
|
}
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
removeListeners() {
|
2021-12-13 02:48:29 +01:00
|
|
|
window.removeEventListener('resize', this.windowResize)
|
2021-12-01 03:02:40 +01:00
|
|
|
var bookshelf = document.getElementById('bookshelf')
|
|
|
|
if (bookshelf) {
|
|
|
|
bookshelf.removeEventListener('scroll', this.scroll)
|
|
|
|
}
|
2022-10-29 01:10:19 +02:00
|
|
|
|
|
|
|
this.$eventBus.$off('bookshelf_clear_selection', this.clearSelectedEntities)
|
2021-12-21 02:26:22 +01:00
|
|
|
this.$eventBus.$off('socket_init', this.socketInit)
|
2022-12-17 21:50:01 +01:00
|
|
|
this.$eventBus.$off('user-settings', this.settingsUpdated)
|
2021-12-02 02:07:03 +01:00
|
|
|
|
|
|
|
if (this.$root.socket) {
|
2022-03-13 00:45:32 +01:00
|
|
|
this.$root.socket.off('item_updated', this.libraryItemUpdated)
|
|
|
|
this.$root.socket.off('item_added', this.libraryItemAdded)
|
|
|
|
this.$root.socket.off('item_removed', this.libraryItemRemoved)
|
|
|
|
this.$root.socket.off('items_updated', this.libraryItemsUpdated)
|
|
|
|
this.$root.socket.off('items_added', this.libraryItemsAdded)
|
2022-04-21 15:30:44 +02:00
|
|
|
this.$root.socket.off('collection_added', this.collectionAdded)
|
|
|
|
this.$root.socket.off('collection_updated', this.collectionUpdated)
|
|
|
|
this.$root.socket.off('collection_removed', this.collectionRemoved)
|
2022-11-27 20:38:08 +01:00
|
|
|
this.$root.socket.off('playlist_added', this.playlistAdded)
|
|
|
|
this.$root.socket.off('playlist_updated', this.playlistUpdated)
|
|
|
|
this.$root.socket.off('playlist_removed', this.playlistRemoved)
|
2024-10-06 17:25:08 +02:00
|
|
|
this.$root.socket.off('author_added', this.authorAdded)
|
|
|
|
this.$root.socket.off('author_updated', this.authorUpdated)
|
|
|
|
this.$root.socket.off('author_removed', this.authorRemoved)
|
2024-07-07 22:51:50 +02:00
|
|
|
this.$root.socket.off('share_open', this.shareOpen)
|
|
|
|
this.$root.socket.off('share_closed', this.shareClosed)
|
2021-12-02 02:07:03 +01:00
|
|
|
} else {
|
|
|
|
console.error('Bookshelf - Socket not initialized')
|
|
|
|
}
|
2021-12-01 03:02:40 +01:00
|
|
|
},
|
|
|
|
destroyEntityComponents() {
|
|
|
|
for (const key in this.entityComponentRefs) {
|
2024-12-16 18:21:44 +01:00
|
|
|
const ref = this.entityComponentRefs[key]
|
|
|
|
if (ref && ref.destroy) {
|
|
|
|
if (ref.$el) ref.$el.remove()
|
|
|
|
ref.destroy()
|
2021-12-01 03:02:40 +01:00
|
|
|
}
|
|
|
|
}
|
2024-12-16 18:21:44 +01:00
|
|
|
this.entityComponentRefs = {}
|
|
|
|
this.entityIndexesMounted = []
|
2021-12-02 02:07:03 +01:00
|
|
|
},
|
|
|
|
scan() {
|
2024-05-13 23:31:30 +02:00
|
|
|
this.tempIsScanning = true
|
2022-05-18 23:33:24 +02:00
|
|
|
this.$store
|
|
|
|
.dispatch('libraries/requestLibraryScan', { libraryId: this.currentLibraryId })
|
|
|
|
.catch((error) => {
|
|
|
|
console.error('Failed to start scan', error)
|
2024-05-13 23:31:30 +02:00
|
|
|
this.$toast.error(this.$strings.ToastLibraryScanFailedToStart)
|
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
this.tempIsScanning = false
|
2022-05-18 23:33:24 +02:00
|
|
|
})
|
2024-12-21 16:42:32 +01:00
|
|
|
},
|
|
|
|
entitiesInShelf(shelf) {
|
|
|
|
return shelf == this.totalShelves ? this.totalEntities % this.entitiesPerShelf : this.entitiesPerShelf
|
|
|
|
},
|
|
|
|
entityTransform(entityIndex) {
|
|
|
|
const shelfOffsetY = this.shelfPaddingHeight * this.sizeMultiplier
|
|
|
|
const shelfOffsetX = (entityIndex - 1) * this.totalEntityCardWidth + this.bookshelfMarginLeft
|
|
|
|
return `translate3d(${shelfOffsetX}px, ${shelfOffsetY}px, 0px)`
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
},
|
2024-06-03 08:04:03 +02:00
|
|
|
async mounted() {
|
|
|
|
await this.cardsHelpers.setCardSize()
|
2021-12-01 03:02:40 +01:00
|
|
|
this.initListeners()
|
2022-05-19 01:37:38 +02:00
|
|
|
|
|
|
|
this.routeFullPath = window.location.pathname + (window.location.search || '')
|
2021-11-29 02:36:44 +01:00
|
|
|
},
|
2021-12-13 02:48:29 +01:00
|
|
|
updated() {
|
2022-05-19 01:37:38 +02:00
|
|
|
this.routeFullPath = window.location.pathname + (window.location.search || '')
|
|
|
|
|
2021-12-18 03:52:11 +01:00
|
|
|
setTimeout(() => {
|
|
|
|
if (window.innerWidth > 0 && window.innerWidth !== this.mountWindowWidth) {
|
|
|
|
console.log('Updated window width', window.innerWidth, 'from', this.mountWindowWidth)
|
2021-12-21 02:26:22 +01:00
|
|
|
this.executeRebuild()
|
2021-12-18 03:52:11 +01:00
|
|
|
}
|
|
|
|
}, 50)
|
2021-12-13 02:48:29 +01:00
|
|
|
},
|
2021-11-29 02:36:44 +01:00
|
|
|
beforeDestroy() {
|
2021-12-01 03:02:40 +01:00
|
|
|
this.destroyEntityComponents()
|
|
|
|
this.removeListeners()
|
2022-05-19 01:37:38 +02:00
|
|
|
|
|
|
|
// Set bookshelf scroll position for specific bookshelf page and query
|
|
|
|
if (window.bookshelf) {
|
|
|
|
this.$store.commit('setLastBookshelfScrollData', { scrollTop: window.bookshelf.scrollTop || 0, path: this.routeFullPath, name: this.page })
|
|
|
|
}
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style>
|
|
|
|
.bookshelfRow {
|
|
|
|
background-image: var(--bookshelf-texture-img);
|
|
|
|
}
|
2022-10-30 16:38:00 +01:00
|
|
|
|
2021-11-29 02:36:44 +01:00
|
|
|
.bookshelfDivider {
|
|
|
|
background: rgb(149, 119, 90);
|
|
|
|
background: var(--bookshelf-divider-bg);
|
2024-06-03 08:04:03 +02:00
|
|
|
box-shadow: 0.125em 0.875em 0.5em #111111aa;
|
2021-11-29 02:36:44 +01:00
|
|
|
}
|
2024-05-13 23:31:30 +02:00
|
|
|
</style>
|