Change: Series show first cover on top #149

This commit is contained in:
advplyr 2021-10-28 19:34:29 -05:00
parent 866ee18016
commit 7933eb369e
2 changed files with 87 additions and 8 deletions

View File

@ -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)
} }

View File

@ -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