200 lines
5.6 KiB
Vue
200 lines
5.6 KiB
Vue
<template>
|
|
<div class="ui vertical left visible wide sidebar">
|
|
<div class="ui inverted segment header-wrapper">
|
|
<search-bar>
|
|
<router-link :title="'Funkwhale'" :to="{name: 'index'}">
|
|
<i class="logo bordered inverted orange big icon">
|
|
<logo class="logo"></logo>
|
|
</i>
|
|
</router-link>
|
|
|
|
</search-bar>
|
|
</div>
|
|
|
|
<div class="menu-area">
|
|
<div class="ui compact fluid two item inverted menu">
|
|
<a class="active item" data-tab="library">Browse</a>
|
|
<a class="item" data-tab="queue">
|
|
Queue
|
|
<template v-if="queue.tracks.length === 0">
|
|
(empty)
|
|
</template>
|
|
<template v-else>
|
|
({{ queue.currentIndex + 1}} of {{ queue.tracks.length }})
|
|
</template>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
<div class="tabs">
|
|
<div class="ui bottom attached active tab" data-tab="library">
|
|
<div class="ui inverted vertical fluid menu">
|
|
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{name: 'profile', params: {username: $store.state.auth.username}}"><i class="user icon"></i> Logged in as {{ $store.state.auth.username }}</router-link>
|
|
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{name: 'logout'}"><i class="sign out icon"></i> Logout</router-link>
|
|
<router-link class="item" v-else :to="{name: 'login'}"><i class="sign in icon"></i> Login</router-link>
|
|
<router-link class="item" :to="{path: '/library'}"><i class="sound icon"> </i>Browse library</router-link>
|
|
<router-link class="item" :to="{path: '/favorites'}"><i class="heart icon"></i> Favorites</router-link>
|
|
</div>
|
|
</div>
|
|
<div v-if="queue.previousQueue " class="ui black icon message">
|
|
<i class="history icon"></i>
|
|
<div class="content">
|
|
<div class="header">
|
|
Do you want to restore your previous queue?
|
|
</div>
|
|
<p>{{ queue.previousQueue.tracks.length }} tracks</p>
|
|
<div class="ui two buttons">
|
|
<div @click="queue.restore()" class="ui basic inverted green button">Yes</div>
|
|
<div @click="queue.removePrevious()" class="ui basic inverted red button">No</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="ui bottom attached tab" data-tab="queue">
|
|
<table class="ui compact inverted very basic fixed single line table">
|
|
<draggable v-model="queue.tracks" element="tbody" @update="reorder">
|
|
<tr @click="$store.dispatch('queue/currentIndex', index)" v-for="(track, index) in queue.tracks" :key="index" :class="[{'active': index === queue.currentIndex}]">
|
|
<td class="right aligned">{{ index + 1}}</td>
|
|
<td class="center aligned">
|
|
<img class="ui mini image" v-if="track.album.cover" :src="backend.absoluteUrl(track.album.cover)">
|
|
<img class="ui mini image" v-else src="../assets/audio/default-cover.png">
|
|
</td>
|
|
<td colspan="4">
|
|
<strong>{{ track.title }}</strong><br />
|
|
{{ track.artist.name }}
|
|
</td>
|
|
<td>
|
|
<template v-if="$store.getters['favorites/isFavorite'](track.id)">
|
|
<i class="pink heart icon"></i>
|
|
</template
|
|
</td>
|
|
<td>
|
|
<i @click.stop="cleanTrack(index)" class="circular trash icon"></i>
|
|
</td>
|
|
</tr>
|
|
</draggable>
|
|
</table>
|
|
<div v-if="$store.state.radios.running" class="ui black message">
|
|
|
|
<div class="content">
|
|
<div class="header">
|
|
<i class="feed icon"></i> You have a radio playing
|
|
</div>
|
|
<p>New tracks will be appended here automatically.</p>
|
|
<div @click="$store.dispatch('radios/stop')" class="ui basic inverted red button">Stop radio</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="ui inverted segment player-wrapper">
|
|
<player></player>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import {mapState, mapActions} from 'vuex'
|
|
|
|
import Player from '@/components/audio/Player'
|
|
import Logo from '@/components/Logo'
|
|
import SearchBar from '@/components/audio/SearchBar'
|
|
import backend from '@/audio/backend'
|
|
import draggable from 'vuedraggable'
|
|
|
|
import $ from 'jquery'
|
|
|
|
export default {
|
|
name: 'sidebar',
|
|
components: {
|
|
Player,
|
|
SearchBar,
|
|
Logo,
|
|
draggable
|
|
},
|
|
data () {
|
|
return {
|
|
backend: backend
|
|
}
|
|
},
|
|
mounted () {
|
|
$(this.$el).find('.menu .item').tab()
|
|
},
|
|
computed: {
|
|
...mapState({
|
|
queue: state => state.queue
|
|
})
|
|
},
|
|
methods: {
|
|
...mapActions({
|
|
cleanTrack: 'queue/cleanTrack'
|
|
}),
|
|
reorder: function (oldValue, newValue) {
|
|
this.$store.commit('queue/reorder', {oldValue, newValue})
|
|
}
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<!-- Add "scoped" attribute to limit CSS to this component only -->
|
|
<style scoped lang="scss">
|
|
|
|
$sidebar-color: #1B1C1D;
|
|
|
|
.sidebar {
|
|
display:flex;
|
|
flex-direction:column;
|
|
justify-content: space-between;
|
|
|
|
> div {
|
|
margin: 0;
|
|
background-color: $sidebar-color;
|
|
}
|
|
.menu {
|
|
}
|
|
}
|
|
|
|
.menu-area {
|
|
padding: 0.5rem;
|
|
.menu .item:not(.active):not(:hover) {
|
|
background-color: rgba(255, 255, 255, 0.06);
|
|
}
|
|
|
|
}
|
|
.tabs {
|
|
overflow-y: auto;
|
|
height: 0px;
|
|
}
|
|
.tab[data-tab="queue"] {
|
|
tr {
|
|
cursor: pointer;
|
|
}
|
|
}
|
|
.sidebar .segment {
|
|
margin: 0;
|
|
border-radius: 0;
|
|
}
|
|
|
|
.ui.inverted.segment.header-wrapper {
|
|
padding: 0;
|
|
padding-bottom: 1rem;
|
|
}
|
|
.tabs {
|
|
flex: 1;
|
|
}
|
|
|
|
.player-wrapper {
|
|
border-top: 1px solid rgba(255, 255, 255, 0.1) !important;
|
|
background-color: rgb(46, 46, 46) !important;
|
|
}
|
|
|
|
.logo {
|
|
cursor: pointer;
|
|
display: inline-block;
|
|
}
|
|
|
|
.ui.search {
|
|
display: inline-block;
|
|
> a {
|
|
margin-right: 1.5rem;
|
|
}
|
|
}
|
|
</style>
|