funkwhale/front/src/components/Sidebar.vue

493 wiersze
17 KiB
Vue
Czysty Zwykły widok Historia

<template>
<aside :class="['ui', 'vertical', 'left', 'visible', 'wide', {'collapsed': isCollapsed}, 'sidebar',]">
<header class="ui inverted segment header-wrapper">
<search-bar @search="isCollapsed = false">
2018-07-17 11:09:13 +00:00
<router-link :title="'Funkwhale'" :to="{name: logoUrl}">
<i class="logo bordered inverted orange big icon">
<logo class="logo"></logo>
</i>
</router-link><span
slot="after"
@click="isCollapsed = !isCollapsed"
:class="['ui', 'basic', 'big', {'inverted': isCollapsed}, 'orange', 'icon', 'collapse', 'button']">
<i class="sidebar icon"></i></span>
</search-bar>
</header>
<div class="menu-area">
<div class="ui compact fluid two item inverted menu">
<a :class="[{active: selectedTab === 'library'}, 'item']" role="button" @click.prevent.stop="selectedTab = 'library'" data-tab="library"><translate translate-context="*/Library/*/Verb">Browse</translate></a>
2018-12-20 10:49:10 +00:00
<a :class="[{active: selectedTab === 'queue'}, 'item']" role="button" @click.prevent.stop="selectedTab = 'queue'" data-tab="queue">
2019-03-08 11:37:02 +00:00
<translate translate-context="Sidebar/Queue/Tab.Title/Noun">Queue</translate>&nbsp;
<template v-if="queue.tracks.length === 0">
2019-03-08 11:37:02 +00:00
<translate translate-context="Sidebar/Queue/Tab.Title">(empty)</translate>
</template>
2019-03-08 11:37:02 +00:00
<translate translate-context="Sidebar/Queue/Tab.Title" v-else :translate-params="{index: queue.currentIndex + 1, length: queue.tracks.length}">
2018-06-30 13:28:47 +00:00
(%{ index } of %{ length })
</translate>
</a>
</div>
</div>
<div class="tabs">
2018-12-20 10:49:10 +00:00
<section :class="['ui', 'bottom', 'attached', {active: selectedTab === 'library'}, 'tab']" :aria-label="labels.mainMenu">
<nav class="ui inverted vertical large fluid menu" role="navigation" :aria-label="labels.mainMenu">
2018-04-28 12:58:25 +00:00
<div class="item">
2019-03-08 11:37:02 +00:00
<header class="header"><translate translate-context="Sidebar/Profile/Title">My account</translate></header>
2018-04-28 12:58:25 +00:00
<div class="menu">
2018-06-30 13:28:47 +00:00
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{name: 'profile', params: {username: $store.state.auth.username}}">
<i class="user icon"></i>
2019-03-08 11:37:02 +00:00
<translate translate-context="Sidebar/Profile/List item.Link" :translate-params="{username: $store.state.auth.username}">
2018-06-30 13:28:47 +00:00
Logged in as %{ username }
</translate>
2018-12-20 11:37:51 +00:00
<img class="ui right floated circular tiny avatar image" v-if="$store.state.auth.profile.avatar.square_crop" v-lazy="$store.getters['instance/absoluteUrl']($store.state.auth.profile.avatar.square_crop)" />
2018-06-30 13:28:47 +00:00
</router-link>
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{path: '/settings'}"><i class="setting icon"></i><translate translate-context="*/*/*/Noun">Settings</translate></router-link>
2018-09-13 15:18:23 +00:00
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{name: 'notifications'}">
<i class="feed icon"></i>
<translate translate-context="*/Notifications/*">Notifications</translate>
2018-09-13 15:18:23 +00:00
<div
v-if="$store.state.ui.notifications.inbox + additionalNotifications > 0"
2018-09-13 15:18:23 +00:00
:class="['ui', 'teal', 'label']">
{{ $store.state.ui.notifications.inbox + additionalNotifications }}</div>
2018-09-13 15:18:23 +00:00
</router-link>
2019-03-08 11:37:02 +00:00
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{name: 'logout'}"><i class="sign out icon"></i><translate translate-context="Sidebar/Login/List item.Link/Verb">Logout</translate></router-link>
2018-09-07 15:08:07 +00:00
<template v-else>
<router-link class="item" :to="{name: 'login'}"><i class="sign in icon"></i><translate translate-context="*/Login/*/Verb">Login</translate></router-link>
2018-09-07 15:08:07 +00:00
<router-link class="item" :to="{path: '/signup'}">
2018-10-05 17:30:21 +00:00
<i class="corner add icon"></i>
<translate translate-context="*/Signup/Link/Verb">Create an account</translate>
2018-09-07 15:08:07 +00:00
</router-link>
</template>
2018-04-28 12:58:25 +00:00
</div>
</div>
<div class="item">
<header class="header"><translate translate-context="*/*/*/Noun">Music</translate></header>
2018-04-28 12:58:25 +00:00
<div class="menu">
2019-03-08 11:37:02 +00:00
<router-link class="item" :to="{path: '/library'}"><i class="sound icon"></i><translate translate-context="Sidebar/Library/List item.Link/Verb">Browse library</translate></router-link>
<router-link class="item" v-if="$store.state.auth.authenticated" :to="{path: '/favorites'}"><i class="heart icon"></i><translate translate-context="Sidebar/Favorites/List item.Link/Noun">Favorites</translate></router-link>
2018-04-28 12:58:25 +00:00
<a
@click="$store.commit('playlists/chooseTrack', null)"
v-if="$store.state.auth.authenticated"
class="item">
<i class="list icon"></i><translate translate-context="*/*/*">Playlists</translate>
2018-04-28 12:58:25 +00:00
</a>
<router-link
v-if="$store.state.auth.authenticated"
class="item" :to="{name: 'content.index'}"><i class="upload icon"></i><translate translate-context="*/Library/*/Verb">Add content</translate></router-link>
2018-04-28 12:58:25 +00:00
</div>
</div>
<div class="item" v-if="$store.state.auth.availablePermissions['settings'] || $store.state.auth.availablePermissions['moderation']">
2019-03-08 11:37:02 +00:00
<header class="header"><translate translate-context="Sidebar/Admin/Title/Noun">Administration</translate></header>
2018-04-28 12:58:25 +00:00
<div class="menu">
<router-link
2019-04-17 12:17:59 +00:00
v-if="$store.state.auth.availablePermissions['library']"
2018-06-19 16:50:22 +00:00
class="item"
2019-04-17 12:17:59 +00:00
:to="{name: 'manage.library.edits', query: {q: 'is_approved:null'}}">
2019-10-01 13:19:55 +00:00
<i class="book icon"></i><translate translate-context="*/*/*/Noun">Library</translate>
2019-04-17 12:17:59 +00:00
<div
v-if="$store.state.ui.notifications.pendingReviewEdits > 0"
:title="labels.pendingReviewEdits"
:class="['ui', 'teal', 'label']">
{{ $store.state.ui.notifications.pendingReviewEdits }}</div>
2018-06-19 16:50:22 +00:00
</router-link>
<router-link
v-if="$store.state.auth.availablePermissions['moderation']"
class="item"
:to="{name: 'manage.moderation.reports.list', query: {q: 'resolved:no'}}">
<i class="shield icon"></i><translate translate-context="*/Moderation/*">Moderation</translate>
<div
v-if="$store.state.ui.notifications.pendingReviewReports > 0"
:title="labels.pendingReviewReports"
:class="['ui', 'teal', 'label']">{{ $store.state.ui.notifications.pendingReviewReports }}</div>
</router-link>
<router-link
2019-04-17 12:17:59 +00:00
v-if="$store.state.auth.availablePermissions['settings']"
class="item"
2019-04-17 12:17:59 +00:00
:to="{name: 'manage.users.users.list'}">
<i class="users icon"></i><translate translate-context="*/*/*/Noun">Users</translate>
</router-link>
<router-link
v-if="$store.state.auth.availablePermissions['settings']"
class="item"
:to="{path: '/manage/settings'}">
<i class="settings icon"></i><translate translate-context="*/*/*/Noun">Settings</translate>
</router-link>
2018-04-28 12:58:25 +00:00
</div>
</div>
</nav>
</section>
<div v-if="queue.previousQueue " class="ui black icon message">
<i class="history icon"></i>
<div class="content">
<div class="header">
2019-03-08 11:37:02 +00:00
<translate translate-context="Sidebar/Queue/Message">Do you want to restore your previous queue?</translate>
</div>
2018-06-30 13:28:47 +00:00
<p>
<translate translate-context="*/*/*"
2018-06-30 13:28:47 +00:00
translate-plural="%{ count } tracks"
:translate-n="queue.previousQueue.tracks.length"
:translate-params="{count: queue.previousQueue.tracks.length}">
%{ count } track
</translate>
</p>
<div class="ui two buttons">
2019-03-08 11:37:02 +00:00
<div @click="queue.restore()" class="ui basic inverted green button"><translate translate-context="*/*/*">Yes</translate></div>
<div @click="queue.removePrevious()" class="ui basic inverted red button"><translate translate-context="*/*/*">No</translate></div>
</div>
</div>
</div>
2018-12-20 10:49:10 +00:00
<section :class="['ui', 'bottom', 'attached', {active: selectedTab === 'queue'}, 'tab']">
<table class="ui compact inverted very basic fixed single line unstackable table">
<draggable v-model="tracks" tag="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 && track.album.cover.original" :src="$store.getters['instance/absoluteUrl'](track.album.cover.small_square_crop)">
<img class="ui mini image" v-else src="../assets/audio/default-cover.png">
</td>
<td colspan="4">
<button class="title reset ellipsis" :title="track.title" :aria-label="labels.selectTrack">
<strong>{{ track.title }}</strong><br />
<span>
{{ track.artist.name }}
</span>
</button>
</td>
<td>
<template v-if="$store.getters['favorites/isFavorite'](track.id)">
<i class="pink heart icon"></i>
</template>
</td>
<td>
<button :title="labels.removeFromQueue" @click.stop="cleanTrack(index)" :class="['ui', {'inverted': index != queue.currentIndex}, 'really', 'tiny', 'basic', 'circular', 'icon', 'button']">
<i class="trash icon"></i>
</button>
</td>
</tr>
</draggable>
</table>
<div v-if="$store.state.radios.running" class="ui black message">
<div class="content">
<div class="header">
2019-03-08 11:37:02 +00:00
<i class="feed icon"></i> <translate translate-context="Sidebar/Player/Title">You have a radio playing</translate>
</div>
2019-03-08 11:37:02 +00:00
<p><translate translate-context="Sidebar/Player/Paragraph">New tracks will be appended here automatically.</translate></p>
<div @click="$store.dispatch('radios/stop')" class="ui basic inverted red button"><translate translate-context="*/Player/Button.Label/Short, Verb">Stop radio</translate></div>
</div>
</div>
</section>
</div>
2018-04-23 17:05:48 +00:00
<player @next="scrollToCurrent" @previous="scrollToCurrent"></player>
</aside>
</template>
<script>
import { mapState, mapActions, mapGetters } 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 {
selectedTab: "library",
backend: backend,
tracksChangeBuffer: null,
isCollapsed: true,
fetchInterval: null
}
},
destroy() {
if (this.fetchInterval) {
clearInterval(this.fetchInterval)
}
},
computed: {
...mapGetters({
additionalNotifications: "ui/additionalNotifications",
}),
...mapState({
queue: state => state.queue,
url: state => state.route.path
}),
labels() {
let mainMenu = this.$pgettext('Sidebar/*/Hidden text', "Main menu")
let selectTrack = this.$pgettext('Sidebar/Player/Hidden text', "Play this track")
let pendingFollows = this.$pgettext('Sidebar/Notifications/Hidden text', "Pending follow requests")
let pendingReviewEdits = this.$pgettext('Sidebar/Moderation/Hidden text', "Pending review edits")
2018-07-01 19:50:50 +00:00
return {
pendingFollows,
mainMenu,
selectTrack,
pendingReviewEdits
2018-07-01 19:50:50 +00:00
}
},
tracks: {
get() {
return this.$store.state.queue.tracks
},
set(value) {
this.tracksChangeBuffer = value
}
2018-07-17 11:09:13 +00:00
},
logoUrl() {
2018-07-17 11:09:13 +00:00
if (this.$store.state.auth.authenticated) {
return "library.index"
2018-07-17 11:09:13 +00:00
} else {
return "index"
2018-07-17 11:09:13 +00:00
}
}
},
methods: {
...mapActions({
cleanTrack: "queue/cleanTrack"
}),
reorder: function(event) {
this.$store.commit("queue/reorder", {
tracks: this.tracksChangeBuffer,
oldIndex: event.oldIndex,
newIndex: event.newIndex
})
},
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
2019-02-14 09:49:06 +00:00
},
applyContentFilters () {
let artistIds = this.$store.getters['moderation/artistFilters']().map((f) => {
return f.target.id
})
if (artistIds.length === 0) {
return
}
let self = this
let tracks = this.tracks.slice().reverse()
tracks.forEach(async (t, i) => {
// we loop from the end because removing index from the start can lead to removing the wrong tracks
let realIndex = tracks.length - i - 1
let matchArtist = artistIds.indexOf(t.artist.id) > -1
if (matchArtist) {
return await self.cleanTrack(realIndex)
}
if (t.album && artistIds.indexOf(t.album.artist.id) > -1) {
return await self.cleanTrack(realIndex)
}
})
}
},
watch: {
url: function() {
this.isCollapsed = true
},
selectedTab: function(newValue) {
if (newValue === "queue") {
this.scrollToCurrent()
}
},
"$store.state.queue.currentIndex": function() {
if (this.selectedTab !== "queue") {
this.scrollToCurrent()
}
2019-02-14 09:49:06 +00:00
},
"$store.state.moderation.lastUpdate": function () {
this.applyContentFilters()
}
}
}
</script>
<!-- Add "scoped" attribute to limit CSS to this component only -->
<style scoped lang="scss">
@import "../style/vendor/media";
$sidebar-color: #3d3e3f;
.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;
}
}
}
> div {
margin: 0;
background-color: $sidebar-color;
}
.menu.vertical {
background: $sidebar-color;
}
}
.menu-area {
.menu .item:not(.active):not(:hover) {
2018-03-17 11:59:50 +00:00
opacity: 0.75;
}
.menu .item {
border-radius: 0;
}
.menu .item.active {
background-color: $sidebar-color;
&:hover {
background-color: rgba(255, 255, 255, 0.06);
}
}
}
2018-04-28 12:58:25 +00:00
.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);
}
2018-04-28 12:58:25 +00:00
}
}
.tabs {
flex: 1;
display: flex;
flex-direction: column;
overflow-y: auto;
justify-content: space-between;
@include media("<=desktop") {
max-height: 500px;
}
}
.ui.tab.active {
display: flex;
}
.tab[data-tab="queue"] {
flex-direction: column;
tr {
cursor: pointer;
}
2018-07-11 13:58:02 +00:00
td:nth-child(2) {
width: 55px;
}
}
.tab[data-tab="library"] {
flex-direction: column;
flex: 1 1 auto;
> .menu {
flex: 1;
flex-grow: 1;
}
> .player-wrapper {
width: 100%;
}
}
.sidebar .segment {
margin: 0;
border-radius: 0;
}
.ui.inverted.segment.header-wrapper {
padding: 0;
}
.logo {
cursor: pointer;
display: inline-block;
2018-03-17 11:59:50 +00:00
margin: 0px;
}
.ui.search {
2018-03-17 11:59:50 +00:00
display: flex;
.collapse.button,
.collapse.button:hover,
.collapse.button:active {
box-shadow: none !important;
margin: 0px;
display: flex;
flex-direction: column;
justify-content: center;
}
}
2018-03-17 11:59:50 +00:00
.ui.message.black {
background: $sidebar-color;
}
2018-07-11 13:58:02 +00:00
.ui.mini.image {
width: 100%;
}
2018-03-17 11:59:50 +00:00
</style>
<style lang="scss">
.sidebar {
.ui.search .input {
flex: 1;
.prompt {
border-radius: 0;
}
}
.ui.search .results {
vertical-align: middle;
}
.ui.search .name {
vertical-align: middle;
}
2018-03-17 11:59:50 +00:00
}
2018-07-17 11:09:13 +00:00
.ui.tiny.avatar.image {
2018-07-13 12:10:39 +00:00
position: relative;
top: -0.5em;
2018-07-17 11:09:13 +00:00
width: 3em;
2018-07-13 12:10:39 +00:00
}
:not(.active) button.title {
outline-color: white;
}
</style>