Newer
Older
Eliot Berriot
committed
<template>
<div :class="['ui', 'vertical', 'left', 'visible', 'wide', {'collapsed': isCollapsed}, 'sidebar',]">
Eliot Berriot
committed
<div class="ui inverted segment header-wrapper">
<search-bar @search="isCollapsed = false">
Eliot Berriot
committed
<router-link :title="'Funkwhale'" :to="{name: 'index'}">
<i class="logo bordered inverted orange big icon">
<logo class="logo"></logo>
</i>
slot="after"
@click="isCollapsed = !isCollapsed"
:class="['ui', 'basic', 'big', {'inverted': isCollapsed}, 'orange', 'icon', 'collapse', 'button']">
<i class="sidebar icon"></i></span>
Eliot Berriot
committed
</search-bar>
</div>
<div class="menu-area">
<div class="ui compact fluid two item inverted menu">
Eliot Berriot
committed
<a class="active item" @click="selectedTab = 'library'" data-tab="library">Browse</a>
<a class="item" @click="selectedTab = 'queue'" data-tab="queue">
Eliot Berriot
committed
<template v-if="queue.tracks.length === 0">
Eliot Berriot
committed
</template>
<translate v-else :translate-params="{index: queue.currentIndex + 1, length: queue.tracks.length}">
(%{ index } of %{ length })
</translate>
Eliot Berriot
committed
</a>
</div>
</div>
<div class="tabs">
Eliot Berriot
committed
<div class="ui bottom attached active tab" data-tab="library">
<div class="ui inverted vertical large 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>
<translate :translate-params="{username: $store.state.auth.username}">
Logged in as %{ username }
</translate>
</router-link>
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{name: 'logout'}"><i class="sign out icon"></i>{{ $gettext('Logout') }}</router-link>
<router-link class="item" v-else :to="{name: 'login'}"><i class="sign in icon"></i>{{ $gettext('Login') }}</router-link>
<router-link class="item" :to="{path: '/library'}"><i class="sound icon"> </i>{{ $gettext('Browse library') }}</router-link>
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{path: '/favorites'}"><i class="heart icon"></i>{{ $gettext('Favorites') }}</router-link>
<a
@click="$store.commit('playlists/chooseTrack', null)"
v-if="$store.state.auth.authenticated"
class="item">
</a>
<router-link
v-if="$store.state.auth.authenticated"
class="item" :to="{path: '/activity'}"><i class="bell icon"></i>{{ $gettext('Activity') }}</router-link>
<div class="item" v-if="showAdmin">
<router-link
class="item"
v-if="$store.state.auth.availablePermissions['library']"
:to="{name: 'manage.library.files'}">
<div
:class="['ui', {'teal': $store.state.ui.notifications.importRequests > 0}, 'label']"
{{ $store.state.ui.notifications.importRequests }}</div>
Eliot Berriot
committed
<router-link
class="item"
v-else-if="$store.state.auth.availablePermissions['upload']"
to="/library/import/launch">
Eliot Berriot
committed
</router-link>
<router-link
class="item"
v-if="$store.state.auth.availablePermissions['federation']"
:to="{path: '/manage/federation/libraries'}">
:class="['ui', {'teal': $store.state.ui.notifications.federation > 0}, 'label']"
{{ $store.state.ui.notifications.federation }}</div>
</router-link>
<router-link
class="item"
v-if="$store.state.auth.availablePermissions['settings']"
:to="{path: '/manage/settings'}">
<router-link
class="item"
v-if="$store.state.auth.availablePermissions['settings']"
:to="{name: 'manage.users.users.list'}">
Eliot Berriot
committed
</div>
</div>
<div v-if="queue.previousQueue " class="ui black icon message">
<i class="history icon"></i>
<div class="content">
<div class="header">
{{ $gettext('Do you want to restore your previous queue?') }}
Eliot Berriot
committed
</div>
<p>
<translate
translate-plural="%{ count } tracks"
:translate-n="queue.previousQueue.tracks.length"
:translate-params="{count: queue.previousQueue.tracks.length}">
%{ count } track
</translate>
</p>
Eliot Berriot
committed
<div class="ui two buttons">
<div @click="queue.restore()" class="ui basic inverted green button">{{ $gettext('Yes') }}</div>
<div @click="queue.removePrevious()" class="ui basic inverted red button">{{ $gettext('No') }}</div>
Eliot Berriot
committed
</div>
</div>
</div>
<div class="ui bottom attached tab" data-tab="queue">
<table class="ui compact inverted very basic fixed single line unstackable table">
Eliot Berriot
committed
<draggable v-model="tracks" element="tbody" @update="reorder">
<tr @click="$store.dispatch('queue/currentIndex', index)" v-for="(track, index) in 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="$store.getters['instance/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>
</td>
<td>
<i @click.stop="cleanTrack(index)" class="circular trash icon"></i>
</td>
</tr>
</draggable>
Eliot Berriot
committed
</table>
<div v-if="$store.state.radios.running" class="ui black message">
Eliot Berriot
committed
<div class="content">
<div class="header">
<i class="feed icon"></i> {{ $gettext('You have a radio playing') }}
Eliot Berriot
committed
</div>
<p>{{ $gettext('New tracks will be appended here automatically.') }}</p>
<div @click="$store.dispatch('radios/stop')" class="ui basic inverted red button">{{ $gettext('Stop radio') }}</div>
Eliot Berriot
committed
</div>
</div>
</div>
</div>
<player @next="scrollToCurrent" @previous="scrollToCurrent"></player>
Eliot Berriot
committed
</div>
</template>
<script>
import {mapState, mapActions} from 'vuex'
Eliot Berriot
committed
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'
Eliot Berriot
committed
import $ from 'jquery'
export default {
name: 'sidebar',
components: {
Player,
SearchBar,
Eliot Berriot
committed
},
data () {
return {
Eliot Berriot
committed
selectedTab: 'library',
backend: backend,
Eliot Berriot
committed
tracksChangeBuffer: null,
isCollapsed: true,
fetchInterval: null
Eliot Berriot
committed
}
},
mounted () {
$(this.$el).find('.menu .item').tab()
created () {
this.fetchNotificationsCount()
this.fetchInterval = setInterval(
this.fetchNotificationsCount, 1000 * 60 * 15)
},
destroy () {
if (this.fetchInterval) {
clearInterval(this.fetchInterval)
}
},
computed: {
...mapState({
queue: state => state.queue,
url: state => state.route.path
}),
showAdmin () {
let adminPermissions = [
this.$store.state.auth.availablePermissions['federation'],
Eliot Berriot
committed
this.$store.state.auth.availablePermissions['library'],
this.$store.state.auth.availablePermissions['upload']
]
return adminPermissions.filter(e => {
return e
}).length > 0
Eliot Berriot
committed
},
tracks: {
get () {
return this.$store.state.queue.tracks
},
set (value) {
this.tracksChangeBuffer = value
}
methods: {
...mapActions({
cleanTrack: 'queue/cleanTrack'
}),
fetchNotificationsCount () {
this.$store.dispatch('ui/fetchFederationNotificationsCount')
this.$store.dispatch('ui/fetchImportRequestsCount')
Eliot Berriot
committed
reorder: function (event) {
this.$store.commit('queue/reorder', {
Eliot Berriot
committed
tracks: this.tracksChangeBuffer, oldIndex: event.oldIndex, newIndex: event.newIndex})
Eliot Berriot
committed
},
scrollToCurrent () {
let current = $(this.$el).find('[data-tab="queue"] .active')[0]
if (!current) {
return
}
let container = $(this.$el).find('.tabs')[0]
// Position container at the top line then scroll current into view
container.scrollTop = 0
current.scrollIntoView(true)
// Scroll back nothing if element is at bottom of container else do it
// for half the height of the containers display area
var scrollBack = (container.scrollHeight - container.scrollTop <= container.clientHeight) ? 0 : container.clientHeight / 2
container.scrollTop = container.scrollTop - scrollBack
},
watch: {
url: function () {
this.isCollapsed = true
Eliot Berriot
committed
},
selectedTab: function (newValue) {
if (newValue === 'queue') {
this.scrollToCurrent()
}
},
'$store.state.queue.currentIndex': function () {
if (this.selectedTab !== 'queue') {
this.scrollToCurrent()
}
'$store.state.auth.availablePermissions': {
handler () {
this.fetchNotificationsCount()
},
deep: true
Eliot Berriot
committed
}
}
</script>
<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped lang="scss">
@import '../style/vendor/media';
Eliot Berriot
committed
$sidebar-color: #3d3e3f;
Eliot Berriot
committed
.sidebar {
background: $sidebar-color;
@include media(">tablet") {
display:flex;
flex-direction:column;
justify-content: space-between;
}
@include media(">desktop") {
.collapse.button {
display: none !important;
}
}
@include media("<desktop") {
position: static !important;
width: 100% !important;
&.collapsed {
.menu-area, .player-wrapper, .tabs {
display: none;
}
}
}
Eliot Berriot
committed
> div {
margin: 0;
background-color: $sidebar-color;
}
.menu.vertical {
background: $sidebar-color;
Eliot Berriot
committed
}
}
.menu-area {
.menu .item:not(.active):not(:hover) {
Eliot Berriot
committed
}
.menu .item {
border-radius: 0;
}
.menu .item.active {
background-color: $sidebar-color;
&:hover {
background-color: rgba(255, 255, 255, 0.06);
Eliot Berriot
committed
}
.vertical.menu {
.item .item {
font-size: 1em;
> i.icon {
float: none;
margin: 0 0.5em 0 0;
}
&:not(.active) {
color: rgba(255, 255, 255, 0.75);
}
Eliot Berriot
committed
.tabs {
Eliot Berriot
committed
flex: 1;
display: flex;
flex-direction: column;
Eliot Berriot
committed
overflow-y: auto;
Eliot Berriot
committed
justify-content: space-between;
@include media("<desktop") {
Eliot Berriot
committed
max-height: 500px;
Eliot Berriot
committed
}
Eliot Berriot
committed
.ui.tab.active {
display: flex;
}
Eliot Berriot
committed
.tab[data-tab="queue"] {
Eliot Berriot
committed
flex-direction: column;
Eliot Berriot
committed
tr {
cursor: pointer;
}
}
Eliot Berriot
committed
.tab[data-tab="library"] {
flex-direction: column;
flex: 1 1 auto;
> .menu {
flex: 1;
flex-grow: 1;
}
> .player-wrapper {
width: 100%;
}
}
Eliot Berriot
committed
.sidebar .segment {
margin: 0;
border-radius: 0;
}
.ui.inverted.segment.header-wrapper {
padding: 0;
}
.logo {
cursor: pointer;
display: inline-block;
Eliot Berriot
committed
}
.ui.search {
.collapse.button, .collapse.button:hover, .collapse.button:active {
box-shadow: none !important;
margin: 0px;
display: flex;
flex-direction: column;
justify-content: center;
Eliot Berriot
committed
}
.ui.message.black {
background: $sidebar-color;
}
</style>
<style lang="scss">
.sidebar {
.ui.search .input {
flex: 1;
.prompt {
border-radius: 0;
}
}
}