mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-22 00:07:52 +01:00
Change: Series show first cover on top #149
This commit is contained in:
parent
866ee18016
commit
7933eb369e
@ -1,11 +1,11 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="relative">
|
<div class="relative">
|
||||||
<div class="rounded-sm h-full overflow-hidden relative" :style="{ padding: `16px ${paddingX}px` }" @mouseover="isHovering = true" @mouseleave="isHovering = false" @click="clickCard">
|
<div class="rounded-sm h-full relative" :style="{ padding: `16px ${paddingX}px` }" @mouseover="mouseoverCard" @mouseleave="mouseleaveCard" @click="clickCard">
|
||||||
<nuxt-link :to="groupTo" class="cursor-pointer">
|
<nuxt-link :to="groupTo" class="cursor-pointer">
|
||||||
<div class="w-full relative" :class="isHovering ? 'bg-black-400' : 'bg-primary'" :style="{ height: height + 'px', width: height + 'px' }">
|
<div class="w-full relative" :class="isHovering ? 'bg-black-400' : 'bg-primary'" :style="{ height: height + 'px', width: height + 'px' }">
|
||||||
<cards-group-cover ref="groupcover" :name="groupName" :book-items="bookItems" :width="height" :height="height" />
|
<cards-group-cover ref="groupcover" :name="groupName" :book-items="bookItems" :width="height" :height="height" />
|
||||||
|
|
||||||
<div v-if="hasValidCovers" class="bg-black bg-opacity-60 absolute top-0 left-0 w-full h-full flex items-center justify-center text-center transition-opacity" :class="isHovering ? '' : 'opacity-0'" :style="{ padding: `${sizeMultiplier}rem` }">
|
<div v-if="hasValidCovers" class="bg-black bg-opacity-60 absolute top-0 left-0 w-full h-full flex items-center justify-center text-center transition-opacity z-30" :class="isHovering ? '' : 'opacity-0'" :style="{ padding: `${sizeMultiplier}rem` }">
|
||||||
<p class="font-book" :style="{ fontSize: sizeMultiplier + 'rem' }">{{ groupName }}</p>
|
<p class="font-book" :style="{ fontSize: sizeMultiplier + 'rem' }">{{ groupName }}</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -103,6 +103,14 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
mouseoverCard() {
|
||||||
|
this.isHovering = true
|
||||||
|
// if (this.$refs.groupcover) this.$refs.groupcover.setHover(true)
|
||||||
|
},
|
||||||
|
mouseleaveCard() {
|
||||||
|
this.isHovering = false
|
||||||
|
// if (this.$refs.groupcover) this.$refs.groupcover.setHover(false)
|
||||||
|
},
|
||||||
clickCard() {
|
clickCard() {
|
||||||
this.$emit('click', this.group)
|
this.$emit('click', this.group)
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,12 @@ export default {
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
noValidCovers: false,
|
noValidCovers: false,
|
||||||
coverDiv: null
|
coverDiv: null,
|
||||||
|
isHovering: false,
|
||||||
|
coverImageEls: [],
|
||||||
|
coverWidth: 0,
|
||||||
|
offsetIncrement: 0,
|
||||||
|
isFannedOut: false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
watch: {
|
watch: {
|
||||||
@ -40,10 +45,59 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
setHover(val) {
|
||||||
|
if (val && !this.isHovering) {
|
||||||
|
this.fanOutCovers()
|
||||||
|
} else if (!val && this.isHovering) {
|
||||||
|
this.reverseFan()
|
||||||
|
}
|
||||||
|
this.isHovering = val
|
||||||
|
},
|
||||||
|
fanOutCovers() {
|
||||||
|
if (this.coverImageEls.length < 2 || this.isFannedOut) return
|
||||||
|
this.isFannedOut = true
|
||||||
|
var fanCoverWidth = this.coverWidth * 0.75
|
||||||
|
var fanWidth = (this.coverImageEls.length - 1) * fanCoverWidth
|
||||||
|
var offsetLeft = (-1 * fanWidth) / 2
|
||||||
|
for (let i = 0; i < this.coverImageEls.length; i++) {
|
||||||
|
var coverEl = this.coverImageEls[i]
|
||||||
|
coverEl.style.transform = `translateX(${offsetLeft}px)`
|
||||||
|
offsetLeft += fanCoverWidth
|
||||||
|
|
||||||
|
var coverOverlay = document.createElement('div')
|
||||||
|
coverOverlay.className = 'absolute top-0 left-0 w-full h-full hover:bg-black hover:bg-opacity-40 text-white text-opacity-0 hover:text-opacity-100 flex items-center justify-center'
|
||||||
|
|
||||||
|
if (coverEl.dataset.volumeNumber) {
|
||||||
|
var pEl = document.createElement('p')
|
||||||
|
pEl.className = 'text-2xl'
|
||||||
|
pEl.textContent = `#${coverEl.dataset.volumeNumber}`
|
||||||
|
coverOverlay.appendChild(pEl)
|
||||||
|
}
|
||||||
|
|
||||||
|
let audiobookId = coverEl.dataset.audiobookId
|
||||||
|
coverOverlay.addEventListener('click', (e) => {
|
||||||
|
this.$router.push(`/audiobook/${audiobookId}`)
|
||||||
|
e.stopPropagation()
|
||||||
|
e.preventDefault()
|
||||||
|
})
|
||||||
|
coverEl.appendChild(coverOverlay)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
reverseFan() {
|
||||||
|
if (this.coverImageEls.length < 2 || !this.isFannedOut) return
|
||||||
|
this.isFannedOut = false
|
||||||
|
for (let i = 0; i < this.coverImageEls.length; i++) {
|
||||||
|
var coverEl = this.coverImageEls[i]
|
||||||
|
coverEl.style.transform = 'translateX(0px)'
|
||||||
|
if (coverEl.lastChild) coverEl.lastChild.remove()
|
||||||
|
}
|
||||||
|
},
|
||||||
getCoverUrl(book) {
|
getCoverUrl(book) {
|
||||||
return this.$store.getters['audiobooks/getBookCoverSrc'](book, '')
|
return this.$store.getters['audiobooks/getBookCoverSrc'](book, '')
|
||||||
},
|
},
|
||||||
async buildCoverImg(src, bgCoverWidth, offsetLeft, forceCoverBg = false) {
|
async buildCoverImg(coverData, bgCoverWidth, offsetLeft, zIndex, forceCoverBg = false) {
|
||||||
|
var src = coverData.coverUrl
|
||||||
|
|
||||||
var showCoverBg =
|
var showCoverBg =
|
||||||
forceCoverBg ||
|
forceCoverBg ||
|
||||||
(await new Promise((resolve) => {
|
(await new Promise((resolve) => {
|
||||||
@ -72,8 +126,11 @@ export default {
|
|||||||
imgdiv.style.height = this.height + 'px'
|
imgdiv.style.height = this.height + 'px'
|
||||||
imgdiv.style.width = bgCoverWidth + 'px'
|
imgdiv.style.width = bgCoverWidth + 'px'
|
||||||
imgdiv.style.left = offsetLeft + 'px'
|
imgdiv.style.left = offsetLeft + 'px'
|
||||||
imgdiv.className = 'absolute top-0 box-shadow-book'
|
imgdiv.style.zIndex = zIndex
|
||||||
imgdiv.style.boxShadow = '-4px 0px 4px #11111166'
|
imgdiv.dataset.audiobookId = coverData.id
|
||||||
|
imgdiv.dataset.volumeNumber = coverData.volumeNumber || ''
|
||||||
|
imgdiv.className = 'absolute top-0 box-shadow-book transition-transform'
|
||||||
|
imgdiv.style.boxShadow = '4px 0px 4px #11111166'
|
||||||
// imgdiv.style.transform = 'skew(0deg, 15deg)'
|
// imgdiv.style.transform = 'skew(0deg, 15deg)'
|
||||||
|
|
||||||
if (showCoverBg) {
|
if (showCoverBg) {
|
||||||
@ -105,7 +162,15 @@ export default {
|
|||||||
this.coverDiv.remove()
|
this.coverDiv.remove()
|
||||||
this.coverDiv = null
|
this.coverDiv = null
|
||||||
}
|
}
|
||||||
var validCovers = this.bookItems.map((bookItem) => this.getCoverUrl(bookItem)).filter((b) => b !== '')
|
var validCovers = this.bookItems
|
||||||
|
.map((bookItem) => {
|
||||||
|
return {
|
||||||
|
id: bookItem.id,
|
||||||
|
volumeNumber: bookItem.book ? bookItem.book.volumeNumber : null,
|
||||||
|
coverUrl: this.getCoverUrl(bookItem)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.filter((b) => b.coverUrl !== '')
|
||||||
if (!validCovers.length) {
|
if (!validCovers.length) {
|
||||||
this.noValidCovers = true
|
this.noValidCovers = true
|
||||||
return
|
return
|
||||||
@ -118,15 +183,21 @@ export default {
|
|||||||
coverWidth = this.height / 1.6
|
coverWidth = this.height / 1.6
|
||||||
widthPer = (this.width - coverWidth) / (validCovers.length - 1)
|
widthPer = (this.width - coverWidth) / (validCovers.length - 1)
|
||||||
}
|
}
|
||||||
|
this.coverWidth = coverWidth
|
||||||
|
this.offsetIncrement = widthPer
|
||||||
|
|
||||||
var outerdiv = document.createElement('div')
|
var outerdiv = document.createElement('div')
|
||||||
outerdiv.className = 'w-full h-full relative'
|
outerdiv.className = 'w-full h-full relative'
|
||||||
|
|
||||||
|
var coverImageEls = []
|
||||||
for (let i = 0; i < validCovers.length; i++) {
|
for (let i = 0; i < validCovers.length; i++) {
|
||||||
var offsetLeft = widthPer * i
|
var offsetLeft = widthPer * i
|
||||||
var img = await this.buildCoverImg(validCovers[i], coverWidth, offsetLeft, validCovers.length === 1)
|
var zIndex = validCovers.length - i
|
||||||
|
var img = await this.buildCoverImg(validCovers[i], coverWidth, offsetLeft, zIndex, validCovers.length === 1)
|
||||||
outerdiv.appendChild(img)
|
outerdiv.appendChild(img)
|
||||||
|
coverImageEls.push(img)
|
||||||
}
|
}
|
||||||
|
this.coverImageEls = coverImageEls
|
||||||
|
|
||||||
if (this.$refs.wrapper) {
|
if (this.$refs.wrapper) {
|
||||||
this.coverDiv = outerdiv
|
this.coverDiv = outerdiv
|
||||||
|
Loading…
Reference in New Issue
Block a user