Skip to content
Snippets Groups Projects
Sidebar.vue 5.7 KiB
Newer Older
  • Learn to ignore specific revisions
  • <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 &nbsp;
             <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="auth.user.authenticated" :to="{name: 'profile', params: {username: auth.user.username}}"><i class="user icon"></i> Logged in as {{ auth.user.username }}</router-link>
            <router-link class="item" v-if="auth.user.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="queue.play(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="favoriteTracks.objects[track.id]">
                      <i @click.stop="queue.cleanTrack(index)" class="pink heart icon"></i>
                      </template
                  </td>
                  <td>
                      <i @click.stop="queue.cleanTrack(index)" class="circular trash icon"></i>
                  </td>
                </tr>
              </draggable>
    
          </table>
          <div v-if="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="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>
    
      <GlobalEvents
        @keydown.r.stop="queue.restore"
        />
    
    import GlobalEvents from '@/components/utils/global-events'
    
    import Player from '@/components/audio/Player'
    import favoriteTracks from '@/favorites/tracks'
    import Logo from '@/components/Logo'
    import SearchBar from '@/components/audio/SearchBar'
    import auth from '@/auth'
    import queue from '@/audio/queue'
    import backend from '@/audio/backend'
    
    import draggable from 'vuedraggable'
    
    import radios from '@/radios'
    
    import $ from 'jquery'
    
    export default {
      name: 'sidebar',
      components: {
        Player,
        SearchBar,
    
        draggable,
        GlobalEvents
    
      },
      data () {
        return {
          auth: auth,
          backend: backend,
          queue: queue,
          radios,
          favoriteTracks
        }
      },
      mounted () {
        $(this.$el).find('.menu .item').tab()
    
      },
      methods: {
        reorder (e) {
          this.queue.reorder(e.oldIndex, e.newIndex)
        }
    
      }
    }
    </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>