lx-music-desktop/src/renderer/views/songList/List/components/SongList.vue

221 lines
4.8 KiB
Vue

<template>
<div :class="$style.container">
<div v-show="!props.listInfo.noItemLabel" ref="dom_list_ref" :class="$style.listContent" class="scroll">
<ul>
<li v-for="item in props.listInfo.list" :key="item.id" :class="$style.item" @click="toDetail(item)">
<div :class="$style.image">
<img :class="$style.img" loading="lazy" decoding="async" :src="item.img">
</div>
<div :class="$style.desc">
<h4>{{ item.name }}</h4>
<div>
<p :class="$style.author">{{ item.author }}</p>
<p v-if="item.time" :class="$style.time">{{ item.time }}</p>
<div :class="$style.songlist_info">
<span v-if="item.total != null"><svg-icon name="music" />{{ item.total }}</span>
<span v-if="item.play_count != null"><svg-icon name="headphones" />{{ item.play_count }}</span>
<span v-if="visibleSource">{{ item.source }}</span>
</div>
</div>
</div>
</li>
<li v-for="(i, index) in 6" :key="index" :class="$style.item" style="margin-bottom: 0;height: 0;" />
</ul>
<div :class="$style.pagination">
<material-pagination :count="props.listInfo.total" :limit="props.listInfo.limit" :page="props.listInfo.page" @btn-click="togglePage" />
</div>
</div>
<transition enter-active-class="animated fadeIn" leave-active-class="animated fadeOut">
<div v-show="props.listInfo.noItemLabel" :class="$style.noitem">
<p v-text="props.listInfo.noItemLabel" />
</div>
</transition>
</div>
</template>
<script setup lang="ts">
import { ref, defineEmits, defineExpose } from '@common/utils/vueTools'
import type { ListInfo, ListInfoItem } from '@renderer/store/songList/state'
// import LX from '@renderer/types/lx'
import { useRoute, useRouter } from '@common/utils/vueRouter'
const props = withDefaults(defineProps<{
listInfo: ListInfo
visibleSource?: boolean
}>(), {
visibleSource: false,
})
const router = useRouter()
const route = useRoute()
const dom_list_ref = ref<HTMLElement | null>(null)
const emit = defineEmits(['toggle-page'])
const togglePage = (page: number) => {
emit('toggle-page', page)
}
const toDetail = (info: ListInfoItem) => {
router.push({
path: '/songList/detail',
query: {
source: info.source,
id: info.id,
picUrl: info.img,
fromName: route.name as string,
},
})
}
defineExpose({
scrollTo(top: number) {
dom_list_ref.value?.scrollTo({
top,
// behavior: 'smooth',
})
},
getScrollTop() {
return dom_list_ref.value?.scrollTop ?? 0
},
})
</script>
<style lang="less" module>
@import '@renderer/assets/styles/layout.less';
.container {
overflow: hidden;
height: 100%;
display: flex;
flex-flow: column nowrap;
position: relative;
}
.listContent {
position: absolute;
left: 0;
top: 0;
width: 100%;
height: 100%;
display: flex;
flex-flow: column nowrap;
font-size: 14px;
box-sizing: border-box;
padding: 15px 15px 0;
ul {
display: flex;
flex-flow: row wrap;
justify-content: space-between;
}
}
.item {
max-width: 360px;
width: 32%;
box-sizing: border-box;
display: flex;
// flex-flow: column nowrap;
// padding: 10px;
margin-bottom: 20px;
cursor: pointer;
transition: opacity @transition-normal;
&:hover {
opacity: .7;
}
}
.image {
flex: none;
width: 40%;
display: flex;
background-position: center;
background-size: cover;
border-radius: 4px;
overflow: hidden;
opacity: .9;
aspect-ratio: 1 / 1;
box-shadow: 0 0 2px 0 rgba(0,0,0,.2);
}
.img {
width: 100%;
height: 100%;
object-fit: cover;
}
.desc {
flex: auto;
padding: 2px 15px 2px 7px;
overflow: hidden;
h4 {
font-size: 14px;
// height: 2.6em;
text-align: justify;
line-height: 1.3;
.mixin-ellipsis-2;
}
}
.songlist_info {
display: flex;
flex-flow: row nowrap;
gap: 15px;
margin-top: 8px;
font-size: 12px;
.mixin-ellipsis-1;
text-align: justify;
line-height: 1.2;
// text-indent: 24px;
color: var(--color-font-label);
svg {
margin-right: 2px;
}
}
.author {
margin-top: 6px;
font-size: 12px;
.mixin-ellipsis-1;
text-align: justify;
line-height: 1.3;
// text-indent: 24px;
color: var(--color-font-label);
}
.time {
margin-top: 3px;
font-size: 12px;
.mixin-ellipsis-1;
text-align: justify;
line-height: 1.3;
// text-indent: 24px;
color: var(--color-font-label);
}
.pagination {
text-align: center;
padding: 15px 0;
// left: 50%;
// transform: translateX(-50%);
}
.noitem {
position: absolute;
top: 0;
left: 0;
height: 100%;
width: 100%;
display: flex;
flex-flow: column nowrap;
justify-content: center;
align-items: center;
// background-color: var(--color-000);
p {
font-size: 24px;
color: var(--color-font-label);
}
}
</style>