mirror of
https://github.com/gedoor/legado.git
synced 2025-08-10 00:52:30 +00:00
197 lines
4.6 KiB
Vue
197 lines
4.6 KiB
Vue
<template>
|
||
<div class="books-wrapper">
|
||
<div class="wrapper">
|
||
<div
|
||
class="book"
|
||
v-for="book in books"
|
||
:key="book.bookUrl"
|
||
@click="handleClick(book)"
|
||
>
|
||
<div class="cover-img">
|
||
<img
|
||
class="cover"
|
||
:src="getCover(book)"
|
||
:key="book.coverUrl"
|
||
@error.once="proxyImage"
|
||
alt=""
|
||
loading="lazy"
|
||
/>
|
||
</div>
|
||
<div class="info">
|
||
<div class="name">{{ book.name }}</div>
|
||
<div class="sub">
|
||
<div class="author">
|
||
{{ book.author }}
|
||
</div>
|
||
<div class="tags" v-if="isSearch">
|
||
<el-tag
|
||
v-for="tag in book.kind?.split(',').slice(0, 2)"
|
||
:key="tag"
|
||
>
|
||
{{ tag }}
|
||
</el-tag>
|
||
</div>
|
||
<div class="update-info" v-if="!isSearch">
|
||
<div class="dot">•</div>
|
||
<div class="size">共{{ (book as Book).totalChapterNum }}章</div>
|
||
<div class="dot">•</div>
|
||
<div class="date">
|
||
{{ dateFormat((book as Book).lastCheckTime) }}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="intro" v-if="isSearch">{{ book.intro }}</div>
|
||
|
||
<div class="dur-chapter" v-if="!isSearch">
|
||
已读:{{ (book as Book).durChapterTitle }}
|
||
</div>
|
||
<div class="last-chapter">最新:{{ book.latestChapterTitle }}</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
<script setup lang="ts">
|
||
import type { Book, SeachBook } from '@/book'
|
||
import { dateFormat, isLegadoUrl } from '../utils/utils'
|
||
import API from '@api'
|
||
const props = defineProps<{
|
||
books: Array<Book | SeachBook>
|
||
isSearch: boolean
|
||
}>()
|
||
|
||
const emit = defineEmits(['bookClick'])
|
||
const handleClick = (book: Book | SeachBook) => emit('bookClick', book)
|
||
const getCover = ({ bookUrl, coverUrl }: Book | SeachBook) => {
|
||
if (coverUrl === undefined) return API.getProxyCoverUrl(bookUrl)
|
||
return isLegadoUrl(coverUrl) ? API.getProxyCoverUrl(coverUrl) : coverUrl
|
||
}
|
||
const proxyImage = (evt: Event) => {
|
||
const target = evt.target as HTMLImageElement
|
||
target.src = API.getProxyCoverUrl(target.src)
|
||
}
|
||
|
||
const subJustify = computed(() =>
|
||
props.isSearch ? 'space-between' : 'flex-start',
|
||
)
|
||
</script>
|
||
|
||
<style lang="scss" scoped>
|
||
.books-wrapper {
|
||
overflow: auto;
|
||
|
||
.wrapper {
|
||
display: grid;
|
||
grid-template-columns: repeat(auto-fill, 380px);
|
||
justify-content: space-around;
|
||
grid-gap: 10px;
|
||
|
||
.book {
|
||
user-select: none;
|
||
display: flex;
|
||
cursor: pointer;
|
||
margin-bottom: 18px;
|
||
padding: 24px 24px;
|
||
width: 360px;
|
||
flex-direction: row;
|
||
justify-content: space-around;
|
||
|
||
.cover-img {
|
||
width: 84px;
|
||
height: 112px;
|
||
|
||
.cover {
|
||
width: 84px;
|
||
height: 112px;
|
||
}
|
||
}
|
||
|
||
.info {
|
||
display: flex;
|
||
flex-direction: column;
|
||
justify-content: space-around;
|
||
align-items: left;
|
||
height: 112px;
|
||
margin-left: 20px;
|
||
flex: 1;
|
||
overflow: hidden;
|
||
|
||
.name {
|
||
width: fit-content;
|
||
font-size: 16px;
|
||
font-weight: 700;
|
||
color: #33373d;
|
||
}
|
||
|
||
.sub {
|
||
display: flex;
|
||
flex-direction: row;
|
||
align-items: baseline;
|
||
justify-content: v-bind('subJustify');
|
||
font-size: 12px;
|
||
font-weight: 600;
|
||
color: #6b6b6b;
|
||
.tags {
|
||
:deep(.el-tag) {
|
||
margin-right: 0.5em;
|
||
}
|
||
}
|
||
.update-info {
|
||
display: flex;
|
||
.dot {
|
||
margin: 0 7px;
|
||
}
|
||
}
|
||
}
|
||
|
||
.intro,
|
||
.dur-chapter,
|
||
.last-chapter {
|
||
color: #969ba3;
|
||
font-size: 13px;
|
||
margin-top: 3px;
|
||
font-weight: 500;
|
||
word-wrap: break-word;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
display: -webkit-box;
|
||
-webkit-box-orient: vertical;
|
||
-webkit-line-clamp: 1;
|
||
line-clamp: 1;
|
||
text-align: left;
|
||
}
|
||
}
|
||
}
|
||
|
||
.book:hover {
|
||
background: rgba(0, 0, 0, 0.1);
|
||
transition-duration: 0.5s;
|
||
}
|
||
}
|
||
|
||
.wrapper:last-child {
|
||
margin-right: auto;
|
||
}
|
||
}
|
||
|
||
.books-wrapper::-webkit-scrollbar {
|
||
width: 0 !important;
|
||
}
|
||
|
||
@media screen and (max-width: 750px) {
|
||
.books-wrapper {
|
||
.wrapper {
|
||
display: flex;
|
||
flex-direction: column;
|
||
|
||
.book {
|
||
box-sizing: border-box;
|
||
width: 100%;
|
||
margin-bottom: 0;
|
||
padding: 10px 20px;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
</style>
|