You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
194 lines
4.6 KiB
194 lines
4.6 KiB
<template>
|
|
<div class="fm" :style="{ background }" data-theme="dark">
|
|
<img :src="nextTrackCover" style="display: none" loading="lazy" />
|
|
<img
|
|
class="cover"
|
|
:src="track.album && track.album.picUrl | resizeImage(512)"
|
|
loading="lazy"
|
|
@click="goToAlbum"
|
|
/>
|
|
<div class="right-part">
|
|
<div class="info">
|
|
<div class="title">{{ track.name }}</div>
|
|
<div class="artist"><ArtistsInLine :artists="artists" /></div>
|
|
</div>
|
|
<div class="controls">
|
|
<div class="buttons">
|
|
<button-icon title="不喜欢" @click.native="moveToFMTrash">
|
|
<svg-icon id="thumbs-down" icon-class="thumbs-down" />
|
|
</button-icon>
|
|
<button-icon
|
|
:title="$t(isPlaying ? 'player.pause' : 'player.play')"
|
|
class="play"
|
|
@click.native="play"
|
|
>
|
|
<svg-icon :icon-class="isPlaying ? 'pause' : 'play'" />
|
|
</button-icon>
|
|
<button-icon :title="$t('player.next')" @click.native="next">
|
|
<svg-icon icon-class="next" />
|
|
</button-icon>
|
|
</div>
|
|
<div class="card-name"><svg-icon icon-class="fm" />私人FM</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import ButtonIcon from '@/components/ButtonIcon.vue';
|
|
import ArtistsInLine from '@/components/ArtistsInLine.vue';
|
|
import { mapState } from 'vuex';
|
|
import * as Vibrant from 'node-vibrant/dist/vibrant.worker.min.js';
|
|
import Color from 'color';
|
|
|
|
export default {
|
|
name: 'FMCard',
|
|
components: { ButtonIcon, ArtistsInLine },
|
|
data() {
|
|
return {
|
|
background: '',
|
|
};
|
|
},
|
|
computed: {
|
|
...mapState(['player']),
|
|
track() {
|
|
return this.player.personalFMTrack;
|
|
},
|
|
isPlaying() {
|
|
return this.player.playing && this.player.isPersonalFM;
|
|
},
|
|
artists() {
|
|
return this.track.artists || this.track.ar || [];
|
|
},
|
|
nextTrackCover() {
|
|
return `${this.player._personalFMNextTrack?.album?.picUrl.replace(
|
|
'http://',
|
|
'https://'
|
|
)}?param=512y512`;
|
|
},
|
|
},
|
|
watch: {
|
|
track() {
|
|
this.getColor();
|
|
},
|
|
},
|
|
created() {
|
|
this.getColor();
|
|
window.ok = this.getColor;
|
|
},
|
|
methods: {
|
|
play() {
|
|
this.player.playPersonalFM();
|
|
},
|
|
next() {
|
|
this.player.playNextFMTrack();
|
|
},
|
|
goToAlbum() {
|
|
if (this.track.album.id === 0) return;
|
|
this.$router.push({ path: '/album/' + this.track.album.id });
|
|
},
|
|
moveToFMTrash() {
|
|
this.player.moveToFMTrash();
|
|
},
|
|
getColor() {
|
|
if (!this.player.personalFMTrack?.album?.picUrl) return;
|
|
const cover = `${this.player.personalFMTrack.album.picUrl.replace(
|
|
'http://',
|
|
'https://'
|
|
)}?param=512y512`;
|
|
Vibrant.from(cover, { colorCount: 1 })
|
|
.getPalette()
|
|
.then(palette => {
|
|
const color = Color.rgb(palette.Vibrant._rgb)
|
|
.darken(0.1)
|
|
.rgb()
|
|
.string();
|
|
const color2 = Color.rgb(palette.Vibrant._rgb)
|
|
.lighten(0.28)
|
|
.rotate(-30)
|
|
.rgb()
|
|
.string();
|
|
this.background = `linear-gradient(to top left, ${color}, ${color2})`;
|
|
});
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.fm {
|
|
padding: 1rem;
|
|
background: var(--color-secondary-bg);
|
|
border-radius: 1rem;
|
|
display: flex;
|
|
height: 198px;
|
|
box-sizing: border-box;
|
|
}
|
|
.cover {
|
|
height: 100%;
|
|
clip-path: border-box;
|
|
border-radius: 0.75rem;
|
|
margin-right: 1.2rem;
|
|
cursor: pointer;
|
|
user-select: none;
|
|
}
|
|
.right-part {
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: space-between;
|
|
color: var(--color-text);
|
|
width: 100%;
|
|
.title {
|
|
font-size: 1.6rem;
|
|
font-weight: 600;
|
|
margin-bottom: 0.6rem;
|
|
display: -webkit-box;
|
|
-webkit-box-orient: vertical;
|
|
-webkit-line-clamp: 2;
|
|
overflow: hidden;
|
|
word-break: break-all;
|
|
}
|
|
.artist {
|
|
opacity: 0.68;
|
|
display: -webkit-box;
|
|
-webkit-box-orient: vertical;
|
|
-webkit-line-clamp: 2;
|
|
overflow: hidden;
|
|
word-break: break-all;
|
|
}
|
|
.controls {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: baseline;
|
|
margin-left: -0.4rem;
|
|
.buttons {
|
|
display: flex;
|
|
}
|
|
.button-icon {
|
|
margin: 0 8px 0 0;
|
|
}
|
|
.svg-icon {
|
|
width: 24px;
|
|
height: 24px;
|
|
}
|
|
.svg-icon#thumbs-down {
|
|
width: 22px;
|
|
height: 22px;
|
|
}
|
|
.card-name {
|
|
font-size: 1rem;
|
|
opacity: 0.18;
|
|
display: flex;
|
|
align-items: center;
|
|
font-weight: 600;
|
|
user-select: none;
|
|
.svg-icon {
|
|
width: 18px;
|
|
height: 18px;
|
|
margin-right: 6px;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|