<template> <div class="w-full h-full"> <div v-show="showPageMenu" v-click-outside="clickOutside" class="pagemenu absolute top-9 left-8 rounded-md overflow-y-auto bg-bg shadow-lg z-20 border border-gray-400" :style="{ width: pageMenuWidth + 'px' }"> <div v-for="(file, index) in cleanedPageNames" :key="file" class="w-full cursor-pointer hover:bg-black-200 px-2 py-1" :class="page === index ? 'bg-black-200' : ''" @click="setPage(index + 1)"> <p class="text-sm truncate">{{ file }}</p> </div> </div> <div v-show="showInfoMenu" v-click-outside="clickOutside" class="pagemenu absolute top-9 left-20 rounded-md overflow-y-auto bg-bg shadow-lg z-20 border border-gray-400 w-96"> <div v-for="key in comicMetadataKeys" :key="key" class="w-full px-2 py-1"> <p class="text-xs"> <strong>{{ key }}</strong >: {{ comicMetadata[key] }} </p> </div> </div> <a v-if="pages && numPages" :href="mainImg" :download="pages[page - 1]" class="absolute top-0 bg-bg text-gray-100 border-b border-l border-r border-gray-400 hover:bg-black-200 cursor-pointer rounded-b-md w-10 h-9 flex items-center justify-center text-center z-20" :class="comicMetadata ? 'left-32' : 'left-20'"> <span class="material-icons text-xl">download</span> </a> <div v-if="comicMetadata" class="absolute top-0 left-20 bg-bg text-gray-100 border-b border-l border-r border-gray-400 hover:bg-black-200 cursor-pointer rounded-b-md w-10 h-9 flex items-center justify-center text-center z-20" @mousedown.prevent @click.stop.prevent="clickShowInfoMenu"> <span class="material-icons text-xl">more</span> </div> <div v-if="numPages" class="absolute top-0 left-8 bg-bg text-gray-100 border-b border-l border-r border-gray-400 hover:bg-black-200 cursor-pointer rounded-b-md w-10 h-9 flex items-center justify-center text-center z-20" @mousedown.prevent @click.stop.prevent="clickShowPageMenu"> <span class="material-icons text-xl">menu</span> </div> <div v-if="numPages" class="absolute top-0 right-16 bg-bg text-gray-100 border-b border-l border-r border-gray-400 rounded-b-md px-2 h-9 flex items-center text-center z-20"> <p class="font-mono">{{ page }} / {{ numPages }}</p> </div> <div class="overflow-hidden w-full h-full relative"> <div v-show="canGoPrev" class="absolute top-0 left-0 h-full w-1/2 lg:w-1/3 hover:opacity-100 opacity-0 z-10 cursor-pointer" @click.stop.prevent="prev" @mousedown.prevent> <div class="flex items-center justify-center h-full w-1/2"> <span v-show="loadedFirstPage" class="material-icons text-5xl text-white cursor-pointer text-opacity-30 hover:text-opacity-90">arrow_back_ios</span> </div> </div> <div v-show="canGoNext" class="absolute top-0 right-0 h-full w-1/2 lg:w-1/3 hover:opacity-100 opacity-0 z-10 cursor-pointer" @click.stop.prevent="next" @mousedown.prevent> <div class="flex items-center justify-center h-full w-1/2 ml-auto"> <span v-show="loadedFirstPage" class="material-icons text-5xl text-white cursor-pointer text-opacity-30 hover:text-opacity-90">arrow_forward_ios</span> </div> </div> <div class="h-full flex justify-center"> <img v-if="mainImg" :src="mainImg" class="object-contain h-full m-auto" /> </div> <div v-show="loading" class="w-full h-full absolute top-0 left-0 flex items-center justify-center z-10"> <ui-loading-indicator /> </div> </div> </div> </template> <script> import Path from 'path' import { Archive } from 'libarchive.js/main.js' import { CompressedFile } from 'libarchive.js/src/compressed-file' Archive.init({ workerUrl: '/libarchive/worker-bundle.js' }) export default { props: { libraryItem: { type: Object, default: () => {} }, playerOpen: Boolean, keepProgress: Boolean, fileId: String }, data() { return { loading: false, pages: null, filesObject: null, mainImg: null, page: 0, numPages: 0, pageMenuWidth: 256, showPageMenu: false, showInfoMenu: false, loadTimeout: null, loadedFirstPage: false, comicMetadata: null } }, watch: { url: { immediate: true, handler() { this.extract() } } }, computed: { userToken() { return this.$store.getters['user/getToken'] }, libraryItemId() { return this.libraryItem?.id }, ebookUrl() { if (this.fileId) { return `/api/items/${this.libraryItemId}/ebook/${this.fileId}` } return `/api/items/${this.libraryItemId}/ebook` }, comicMetadataKeys() { return this.comicMetadata ? Object.keys(this.comicMetadata) : [] }, canGoNext() { return this.page < this.numPages }, canGoPrev() { return this.page > 1 }, userMediaProgress() { if (!this.libraryItemId) return return this.$store.getters['user/getUserMediaProgress'](this.libraryItemId) }, savedPage() { if (!this.keepProgress) return 0 // Validate ebookLocation is a number if (!this.userMediaProgress?.ebookLocation || isNaN(this.userMediaProgress.ebookLocation)) return 0 return Number(this.userMediaProgress.ebookLocation) }, cleanedPageNames() { return ( this.pages?.map((p) => { if (p.length > 50) { let firstHalf = p.slice(0, 22) let lastHalf = p.slice(p.length - 23) return `${firstHalf} ... ${lastHalf}` } return p }) || [] ) } }, methods: { clickShowPageMenu() { this.showInfoMenu = false this.showPageMenu = !this.showPageMenu }, clickShowInfoMenu() { this.showPageMenu = false this.showInfoMenu = !this.showInfoMenu }, updateProgress() { if (!this.keepProgress) return if (!this.numPages) { console.error('Num pages not loaded') return } if (this.savedPage === this.page) { return } const payload = { ebookLocation: this.page, ebookProgress: Math.max(0, Math.min(1, (Number(this.page) - 1) / Number(this.numPages))) } this.$axios.$patch(`/api/me/progress/${this.libraryItemId}`, payload).catch((error) => { console.error('ComicReader.updateProgress failed:', error) }) }, clickOutside() { if (this.showPageMenu) this.showPageMenu = false if (this.showInfoMenu) this.showInfoMenu = false }, next() { if (!this.canGoNext) return this.setPage(this.page + 1) }, prev() { if (!this.canGoPrev) return this.setPage(this.page - 1) }, setPage(page) { if (page <= 0 || page > this.numPages) { return } this.showPageMenu = false this.showInfoMenu = false const filename = this.pages[page - 1] this.page = page this.updateProgress() return this.extractFile(filename) }, setLoadTimeout() { this.loadTimeout = setTimeout(() => { this.loading = true }, 150) }, extractFile(filename) { return new Promise(async (resolve) => { this.setLoadTimeout() var file = await this.filesObject[filename].extract() var reader = new FileReader() reader.onload = (e) => { this.mainImg = e.target.result this.loading = false resolve() } reader.onerror = (e) => { console.error(e) this.$toast.error('Read page file failed') this.loading = false resolve() } reader.readAsDataURL(file) clearTimeout(this.loadTimeout) }) }, async extract() { this.loading = true var buff = await this.$axios.$get(this.ebookUrl, { responseType: 'blob', headers: { Authorization: `Bearer ${this.userToken}` } }) const archive = await Archive.open(buff) const originalFilesObject = await archive.getFilesObject() // to support images in subfolders we need to flatten the object // ref: https://github.com/advplyr/audiobookshelf/issues/811 this.filesObject = this.flattenFilesObject(originalFilesObject) console.log('Extracted files object', this.filesObject) var filenames = Object.keys(this.filesObject) this.parseFilenames(filenames) var xmlFile = filenames.find((f) => (Path.extname(f) || '').toLowerCase() === '.xml') if (xmlFile) await this.extractXmlFile(xmlFile) this.numPages = this.pages.length // Calculate page menu size const largestFilename = this.cleanedPageNames .map((p) => p) .sort((a, b) => a.length - b.length) .pop() const pEl = document.createElement('p') pEl.innerText = largestFilename pEl.style.fontSize = '0.875rem' pEl.style.opacity = 0 pEl.style.position = 'absolute' document.body.appendChild(pEl) const textWidth = pEl.getBoundingClientRect()?.width if (textWidth) { this.pageMenuWidth = textWidth + (16 + 5 + 2 + 5) } pEl.remove() if (this.pages.length) { this.loading = false const startPage = this.savedPage > 0 && this.savedPage <= this.numPages ? this.savedPage : 1 await this.setPage(startPage) this.loadedFirstPage = true } else { this.$toast.error('Unable to extract pages') this.loading = false } }, flattenFilesObject(filesObject) { const flattenObject = (obj, prefix = '') => { var _obj = {} for (const key in obj) { const newKey = prefix ? prefix + '/' + key : key if (obj[key] instanceof CompressedFile) { _obj[newKey] = obj[key] } else if (!key.startsWith('_') && typeof obj[key] === 'object' && !Array.isArray(obj[key])) { _obj = { ..._obj, ...flattenObject(obj[key], newKey) } } else { _obj[newKey] = obj[key] } } return _obj } return flattenObject(filesObject) }, async extractXmlFile(filename) { console.log('extracting xml filename', filename) try { var file = await this.filesObject[filename].extract() var reader = new FileReader() reader.onload = (e) => { this.comicMetadata = this.$xmlToJson(e.target.result) console.log('Metadata', this.comicMetadata) } reader.onerror = (e) => { console.error(e) } reader.readAsText(file) } catch (error) { console.error(error) } }, parseImageFilename(filename) { var basename = Path.basename(filename, Path.extname(filename)) var numbersinpath = basename.match(/\d+/g) if (!numbersinpath?.length) { return { index: -1, filename } } else { return { index: Number(numbersinpath[numbersinpath.length - 1]), filename } } }, parseFilenames(filenames) { const acceptableImages = ['.jpeg', '.jpg', '.png'] var imageFiles = filenames.filter((f) => { return acceptableImages.includes((Path.extname(f) || '').toLowerCase()) }) var imageFileObjs = imageFiles.map((img) => { return this.parseImageFilename(img) }) var imagesWithNum = imageFileObjs.filter((i) => i.index >= 0) var orderedImages = imagesWithNum.sort((a, b) => a.index - b.index).map((i) => i.filename) var noNumImages = imageFileObjs.filter((i) => i.index < 0) orderedImages = orderedImages.concat(noNumImages.map((i) => i.filename)) this.pages = orderedImages } }, mounted() {}, beforeDestroy() {} } </script> <style scoped> .pagemenu { max-height: calc(100% - 48px); } </style>