koel/resources/assets/js/components/main-wrapper/extra/index.vue
2017-03-28 18:25:24 +08:00

169 lines
4.4 KiB
Vue

<template>
<section id="extra" :class="{ showing: state.showExtraPanel }">
<div class="tabs">
<div class="header clear">
<a @click.prevent="currentView = 'lyrics'"
:class="{ active: currentView === 'lyrics' }">Lyrics</a>
<a @click.prevent="currentView = 'artistInfo'"
:class="{ active: currentView === 'artistInfo' }">Artist</a>
<a @click.prevent="currentView = 'albumInfo'"
:class="{ active: currentView === 'albumInfo' }">Album</a>
<a @click.prevent="currentView = 'youtube'"
v-if="sharedState.useYouTube"
:class="{ active: currentView === 'youtube' }"><i class="fa fa-youtube-play"></i></a>
</div>
<div class="panes">
<lyrics :song="song" ref="lyrics" v-show="currentView === 'lyrics'"/>
<artist-info v-if="song.artist.id"
:artist="song.artist"
mode="sidebar"
ref="artist-info"
v-show="currentView === 'artistInfo'"/>
<album-info v-if="song.album.id"
:album="song.album"
mode="sidebar"
ref="album-info"
v-show="currentView === 'albumInfo'"/>
<youtube v-if="sharedState.useYouTube"
:song="song" :youtube="song.youtube"
ref="youtube"
v-show="currentView === 'youtube'"/>
</div>
</div>
</section>
</template>
<script>
import isMobile from 'ismobilejs'
import { event, $ } from '../../../utils'
import { sharedStore, songStore, preferenceStore as preferences } from '../../../stores'
import { songInfo } from '../../../services'
import lyrics from './lyrics.vue'
import artistInfo from './artist-info.vue'
import albumInfo from './album-info.vue'
import youtube from './youtube.vue'
export default {
name: 'main-wrapper--extra--index',
components: { lyrics, artistInfo, albumInfo, youtube },
data () {
return {
song: songStore.stub,
state: preferences.state,
sharedState: sharedStore.state,
currentView: 'lyrics'
}
},
watch: {
/**
* Watch the "showExtraPanel" property to add/remove the corresponding class
* to/from the html tag.
* Some element's CSS can then be controlled based on this class.
*/
'state.showExtraPanel' (newVal) {
if (newVal && !isMobile.any) {
$.addClass(document.documentElement, 'with-extra-panel')
} else {
$.removeClass(document.documentElement, 'with-extra-panel')
}
}
},
mounted () {
// On ready, add 'with-extra-panel' class.
if (!isMobile.any) {
$.addClass(document.documentElement, 'with-extra-panel')
}
if (isMobile.phone) {
// On a mobile device, we always hide the panel initially regardless of
// the saved preference.
preferences.showExtraPanel = false
}
},
methods: {
/**
* Reset all self and applicable child components' states.
*/
resetState () {
this.currentView = 'lyrics'
this.song = songStore.stub
}
},
created () {
event.on({
'main-content-view:load': view => {
// Hide the panel away if a main view is triggered on mobile.
if (isMobile.phone) {
preferences.showExtraPanel = false
}
},
'song:played': song => {
songInfo.fetch(song).then(song => {
this.song = song
})
}
})
}
}
</script>
<style lang="scss">
@import "../../../../sass/partials/_vars.scss";
@import "../../../../sass/partials/_mixins.scss";
#extra {
flex: 0 0 $extraPanelWidth;
padding: 24px 16px $footerHeight;
background: $colorExtraBgr;
max-height: calc(100vh - #{$headerHeight + $footerHeight});
display: none;
color: $color2ndText;
overflow: auto;
-ms-overflow-style: -ms-autohiding-scrollbar;
html.touchevents & {
// Enable scroll with momentum on touch devices
overflow-y: scroll;
-webkit-overflow-scrolling: touch;
}
&.showing {
display: block;
}
h1 {
font-weight: $fontWeight_UltraThin;
font-size: 2.2rem;
margin-bottom: 16px;
line-height: 2.8rem;
}
@media only screen and (max-width : 1024px) {
position: fixed;
height: calc(100vh - #{$headerHeight + $footerHeight});
padding-bottom: $footerHeight; // make sure the footer can never overlap the content
width: $extraPanelWidth;
z-index: 5;
top: $headerHeight;
right: -100%;
transition: right .3s ease-in;
&.showing {
right: 0;
}
}
@media only screen and (max-width : 667px) {
width: 100%;
}
}
</style>