koel/resources/assets/js/components/shared/song-list.vue

579 lines
14 KiB
Vue
Raw Normal View History

2015-12-13 04:42:28 +00:00
<template>
2016-06-25 16:05:24 +00:00
<div class="song-list-wrap main-scroll-wrap" :class="type"
ref="wrapper"
tabindex="1"
@scroll="scrolling"
@keydown.delete.prevent.stop="handleDelete"
@keydown.enter.prevent.stop="handleEnter"
@keydown.a.prevent="handleA"
>
<table v-show="items.length">
<thead>
<tr>
<th @click="sort('track')" class="track-number">#
2016-10-31 04:28:12 +00:00
<i class="fa fa-angle-down" v-show="sortKey === 'track' && order > 0"/>
<i class="fa fa-angle-up" v-show="sortKey === 'track' && order < 0"/>
2016-06-25 16:05:24 +00:00
</th>
<th @click="sort('title')" class="title">Title
2016-10-31 04:28:12 +00:00
<i class="fa fa-angle-down" v-show="sortKey === 'title' && order > 0"/>
<i class="fa fa-angle-up" v-show="sortKey === 'title' && order < 0"/>
2016-06-25 16:05:24 +00:00
</th>
<th @click="sort(['album.artist.name', 'album.name', 'track'])" class="artist">Artist
2016-10-31 04:28:12 +00:00
<i class="fa fa-angle-down" v-show="sortingByArtist && order > 0"/>
<i class="fa fa-angle-up" v-show="sortingByArtist && order < 0"/>
2016-06-25 16:05:24 +00:00
</th>
<th @click="sort(['album.name', 'track'])" class="album">Album
2016-10-31 04:28:12 +00:00
<i class="fa fa-angle-down" v-show="sortingByAlbum && order > 0"/>
<i class="fa fa-angle-up" v-show="sortingByAlbum && order < 0"/>
2016-06-25 16:05:24 +00:00
</th>
2016-11-15 07:54:41 +00:00
<th @click="sort('length')" class="time">Time
<i class="fa fa-angle-down" v-show="sortKey === 'length' && order > 0"/>
<i class="fa fa-angle-up" v-show="sortKey === 'length' && order < 0"/>
2016-06-25 16:05:24 +00:00
</th>
<th class="play"></th>
</tr>
</thead>
<tbody>
2016-11-15 07:54:41 +00:00
<tr is="song-item"
v-for="item in displayedItems"
@itemClicked="itemClicked"
:song="item"
:key="item.id"
ref="rows"/>
2016-06-25 16:05:24 +00:00
</tbody>
</table>
2016-10-31 04:28:12 +00:00
<song-menu ref="contextMenu" :songs="selectedSongs"/>
<to-top-button :showing="showBackToTop"/>
2016-06-25 16:05:24 +00:00
</div>
2015-12-13 04:42:28 +00:00
</template>
<script>
2016-11-26 03:25:35 +00:00
import { find, invokeMap, filter, map } from 'lodash'
import isMobile from 'ismobilejs'
import $ from 'jquery'
2016-12-19 05:37:30 +00:00
import { filterBy, orderBy, limitBy, event, pluralize } from '../../utils'
2016-11-26 03:25:35 +00:00
import { playlistStore, queueStore, songStore, favoriteStore } from '../../stores'
import { playback } from '../../services'
import router from '../../router'
import songItem from './song-item.vue'
import songMenu from './song-menu.vue'
import infiniteScroll from '../../mixins/infinite-scroll'
2016-06-25 16:05:24 +00:00
export default {
name: 'song-list',
props: ['items', 'type', 'playlist', 'sortable'],
mixins: [infiniteScroll],
components: { songItem, songMenu },
2016-11-26 03:25:35 +00:00
data () {
2016-06-25 16:05:24 +00:00
return {
lastSelectedRow: null,
q: '', // The filter query
sortKey: '',
order: 1,
sortingByAlbum: false,
sortingByArtist: false,
selectedSongs: [],
2016-11-26 03:25:35 +00:00
mutatedItems: []
}
2016-06-25 16:05:24 +00:00
},
watch: {
/**
* Watch the items.
*/
2016-11-26 03:25:35 +00:00
items () {
2016-06-25 16:05:24 +00:00
if (this.sortable === false) {
2016-11-26 03:25:35 +00:00
this.sortKey = ''
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
this.mutatedItems = this.items
2016-06-26 16:56:55 +00:00
2016-06-25 16:05:24 +00:00
// Update the song count and duration status on parent.
this.$parent.updateMeta({
songCount: this.items.length,
2016-11-26 03:25:35 +00:00
totalLength: songStore.getLength(this.items, true)
})
2016-06-25 16:05:24 +00:00
},
2016-11-26 03:25:35 +00:00
selectedSongs (val) {
this.$parent.setSelectedSongs(val)
}
2016-06-25 16:05:24 +00:00
},
computed: {
2016-11-26 03:25:35 +00:00
displayedItems () {
2016-06-25 16:05:24 +00:00
return limitBy(
filterBy(
2016-06-26 16:56:55 +00:00
this.mutatedItems,
2016-06-25 16:05:24 +00:00
this.q,
'title', 'album.name', 'artist.name'
),
this.numOfItems
2016-11-26 03:25:35 +00:00
)
}
2016-06-25 16:05:24 +00:00
},
methods: {
/**
* Handle sorting the song list.
*
2016-11-15 07:54:41 +00:00
* @param {String} key The sort key. Can be 'title', 'album', 'artist', or 'length'
2016-06-25 16:05:24 +00:00
*/
2016-11-26 03:25:35 +00:00
sort (key) {
2016-06-25 16:05:24 +00:00
if (this.sortable === false) {
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
this.sortKey = key
this.order = 0 - this.order
this.sortingByAlbum = Array.isArray(this.sortKey) && this.sortKey[0] === 'album.name'
this.sortingByArtist = Array.isArray(this.sortKey) && this.sortKey[0] === 'album.artist.name'
this.mutatedItems = orderBy(this.items, this.sortKey, this.order)
2016-06-25 16:05:24 +00:00
},
/**
* Execute the corresponding reaction(s) when the user presses Delete.
*/
2016-11-26 03:25:35 +00:00
handleDelete () {
const songs = this.selectedSongs
2016-06-25 16:05:24 +00:00
if (!songs.length) {
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
switch (this.type) {
case 'queue':
2016-11-26 03:25:35 +00:00
queueStore.unqueue(songs)
break
2016-06-25 16:05:24 +00:00
case 'favorites':
2016-11-26 03:25:35 +00:00
favoriteStore.unlike(songs)
break
2016-06-25 16:05:24 +00:00
case 'playlist':
2016-11-26 03:25:35 +00:00
playlistStore.removeSongs(this.playlist, songs)
break
2016-06-25 16:05:24 +00:00
default:
2016-11-26 03:25:35 +00:00
break
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
this.clearSelection()
2016-06-25 16:05:24 +00:00
},
/**
* Execute the corresponding reaction(s) when the user presses Enter.
*
* @param {Object} e The keydown event.
*/
2016-11-26 03:25:35 +00:00
handleEnter (e) {
const songs = this.selectedSongs
2016-06-25 16:05:24 +00:00
if (!songs.length) {
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
if (songs.length === 1) {
// Just play the song
2016-11-26 03:25:35 +00:00
playback.play(songs[0])
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
switch (this.type) {
case 'queue':
// Play the first song selected if we're in Queue screen.
2016-11-26 03:25:35 +00:00
playback.play(songs[0])
break
2016-06-25 16:05:24 +00:00
case 'favorites':
case 'playlist':
default:
//
// --------------------------------------------------------------------
// For other screens, follow this map:
//
// • Enter: Queue songs to bottom
// • Shift+Enter: Queues song to top
// • Cmd/Ctrl+Enter: Queues song to bottom and play the first selected song
// • Cmd/Ctrl+Shift+Enter: Queue songs to top and play the first queued song
//
// Also, if there's only one song selected, play it right away.
// --------------------------------------------------------------------
//
2016-11-26 03:25:35 +00:00
queueStore.queue(songs, false, e.shiftKey)
2016-06-25 16:05:24 +00:00
this.$nextTick(() => {
2016-11-26 03:25:35 +00:00
router.go('queue')
2016-06-25 16:05:24 +00:00
if (e.ctrlKey || e.metaKey || songs.length === 1) {
2016-11-26 03:25:35 +00:00
playback.play(songs[0])
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
})
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
break
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
this.clearSelection()
2016-06-25 16:05:24 +00:00
},
/**
* Get the song-item component that's associated with a song ID.
*
* @param {String} id The song ID.
*
* @return {Object} The Vue compoenent
*/
2016-11-26 03:25:35 +00:00
getComponentBySongId (id) {
return find(this.$refs.rows, { song: { id }})
2016-06-25 16:05:24 +00:00
},
/**
* Capture A keydown event and select all if applicable.
*
* @param {Object} e The keydown event.
*/
2016-11-26 03:25:35 +00:00
handleA (e) {
2016-06-25 16:05:24 +00:00
if (!e.metaKey && !e.ctrlKey) {
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
invokeMap(this.$refs.rows, 'select')
this.gatherSelected()
2016-06-25 16:05:24 +00:00
},
/**
* Gather all selected songs.
*
* @return {Array.<Object>} An array of Song objects
*/
2016-11-26 03:25:35 +00:00
gatherSelected () {
const selectedRows = filter(this.$refs.rows, { selected: true })
const ids = map(selectedRows, row => row.song.id)
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
this.selectedSongs = songStore.byIds(ids)
2016-06-25 16:05:24 +00:00
},
/**
* -----------------------------------------------------------
* The next four methods are to deal with selection.
*
* Credits: http://stackoverflow.com/a/17966381/794641 by andyb
* -----------------------------------------------------------
*/
/**
* Handle the click event on a row to perform selection.
*
* @param {String} songId
* @param {Object} e
*/
2016-11-26 03:25:35 +00:00
itemClicked (songId, e) {
const row = this.getComponentBySongId(songId)
2016-06-25 16:05:24 +00:00
// If we're on a touch device, or if Ctrl/Cmd key is pressed, just toggle selection.
if (isMobile.any) {
2016-11-26 03:25:35 +00:00
this.toggleRow(row)
this.gatherSelected()
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
if (e.ctrlKey || e.metaKey) {
2016-11-26 03:25:35 +00:00
this.toggleRow(row)
2016-06-25 16:05:24 +00:00
}
if (e.button === 0) {
if (!e.ctrlKey && !e.metaKey && !e.shiftKey) {
2016-11-26 03:25:35 +00:00
this.clearSelection()
this.toggleRow(row)
2016-06-25 16:05:24 +00:00
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
if (e.shiftKey && this.lastSelectedRow && this.lastSelectedRow.$el) {
2016-11-26 03:25:35 +00:00
this.selectRowsBetweenIndexes([this.lastSelectedRow.$el.rowIndex, row.$el.rowIndex])
2016-06-25 16:05:24 +00:00
}
}
2016-11-26 03:25:35 +00:00
this.gatherSelected()
2016-06-25 16:05:24 +00:00
},
/**
* Toggle select/unslect a row.
*
* @param {Object} row The song-item component
*/
2016-11-26 03:25:35 +00:00
toggleRow (row) {
row.toggleSelectedState()
this.lastSelectedRow = row
2016-06-25 16:05:24 +00:00
},
2016-11-26 03:25:35 +00:00
selectRowsBetweenIndexes (indexes) {
indexes.sort((a, b) => a - b)
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
const rows = $(this.$refs.wrapper).find('tbody tr')
2016-06-25 16:05:24 +00:00
for (let i = indexes[0]; i <= indexes[1]; ++i) {
2016-11-26 03:25:35 +00:00
this.getComponentBySongId($(rows[i - 1]).data('song-id')).select()
2016-06-25 16:05:24 +00:00
}
},
/**
* Clear the current selection on this song list.
*/
2016-11-26 03:25:35 +00:00
clearSelection () {
invokeMap(this.$refs.rows, 'deselect')
this.gatherSelected()
2016-06-25 16:05:24 +00:00
},
/**
* Enable dragging songs by capturing the dragstart event on a table row.
* Even though the event is triggered on one row only, we'll collect other
* selected rows, if any, as well.
*
* @param {Object} e The event.
*/
2016-11-26 03:25:35 +00:00
dragStart (songId, e) {
2016-06-25 16:05:24 +00:00
// If the user is dragging an unselected row, clear the current selection.
2016-11-26 03:25:35 +00:00
const currentRow = this.getComponentBySongId(songId)
2016-06-25 16:05:24 +00:00
if (!currentRow.selected) {
2016-11-26 03:25:35 +00:00
this.clearSelection()
currentRow.select()
this.gatherSelected()
2016-06-25 16:05:24 +00:00
}
this.$nextTick(() => {
2016-11-26 03:25:35 +00:00
const songIds = map(this.selectedSongs, 'id')
e.dataTransfer.setData('application/x-koel.text+plain', songIds)
e.dataTransfer.effectAllowed = 'move'
2016-06-25 16:05:24 +00:00
// Set a fancy drop image using our ghost element.
2016-12-19 05:37:30 +00:00
const $ghost = $('#dragGhost').text(`${pluralize(songIds.length, 'song')}`)
2016-11-26 03:25:35 +00:00
e.dataTransfer.setDragImage($ghost[0], 0, 0)
})
2016-06-25 16:05:24 +00:00
},
/**
* Add a "droppable" class and set the drop effect when other songs are dragged over a row.
*
* @param {String} songId
* @param {Object} e The dragover event.
*/
2016-11-26 03:25:35 +00:00
allowDrop (songId, e) {
2016-06-25 16:05:24 +00:00
if (this.type !== 'queue') {
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
$(e.target).parents('tr').addClass('droppable')
e.dataTransfer.dropEffect = 'move'
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
return false
2016-06-25 16:05:24 +00:00
},
/**
* Perform reordering songs upon dropping if the current song list is of type Queue.
*
* @param {String} songId
* @param {Object} e
*/
2016-11-26 03:25:35 +00:00
handleDrop (songId, e) {
2016-06-25 16:05:24 +00:00
if (this.type !== 'queue') {
2016-11-26 03:25:35 +00:00
return this.removeDroppableState(e) && false
2016-06-25 16:05:24 +00:00
}
if (!e.dataTransfer.getData('application/x-koel.text+plain')) {
2016-11-26 03:25:35 +00:00
return this.removeDroppableState(e) && false
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
const songs = this.selectedSongs
2016-06-25 16:05:24 +00:00
if (!songs.length) {
2016-11-26 03:25:35 +00:00
return this.removeDroppableState(e) && false
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
queueStore.move(songs, songStore.byId(songId))
2016-06-25 16:05:24 +00:00
2016-11-26 03:25:35 +00:00
return this.removeDroppableState(e) && false
2016-06-25 16:05:24 +00:00
},
/**
* Remove the droppable state (and the styles) from a row.
*
* @param {Object} e
*/
2016-11-26 03:25:35 +00:00
removeDroppableState (e) {
return $(e.target).parents('tr').removeClass('droppable')
2016-06-25 16:05:24 +00:00
},
2016-11-26 03:25:35 +00:00
openContextMenu (songId, e) {
2016-06-25 16:05:24 +00:00
// If the user is right-clicking an unselected row,
// clear the current selection and select it instead.
2016-11-26 03:25:35 +00:00
const currentRow = this.getComponentBySongId(songId)
2016-06-25 16:05:24 +00:00
if (!currentRow.selected) {
2016-11-26 03:25:35 +00:00
this.clearSelection()
currentRow.select()
this.gatherSelected()
2016-06-25 16:05:24 +00:00
}
2016-11-26 03:25:35 +00:00
this.$nextTick(() => this.$refs.contextMenu.open(e.pageY, e.pageX))
}
2016-06-25 16:05:24 +00:00
},
2016-11-26 03:25:35 +00:00
created () {
2016-06-25 16:05:24 +00:00
event.on({
/**
* Listen to song:played event to do some logic.
*
* @param {Object} song The current playing song.
*/
'song:played': song => {
// If the song is at the end of the current displayed items, load more.
if (this.type === 'queue' && this.items.indexOf(song) >= this.numOfItems) {
2016-11-26 03:25:35 +00:00
this.displayMore()
2016-06-25 16:05:24 +00:00
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
// Scroll the item into view if it's lost into oblivion.
if (this.type === 'queue') {
2016-11-26 03:25:35 +00:00
const $wrapper = $(this.$refs.wrapper)
const $row = $wrapper.find(`.song-item[data-song-id="${song.id}"]`)
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
if (!$row.length) {
2016-11-26 03:25:35 +00:00
return
2016-06-25 16:05:24 +00:00
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
if ($wrapper[0].getBoundingClientRect().top + $wrapper[0].getBoundingClientRect().height <
$row[0].getBoundingClientRect().top) {
2016-11-26 03:25:35 +00:00
$wrapper.scrollTop($wrapper.scrollTop() + $row.position().top)
2016-06-25 16:05:24 +00:00
}
}
2016-06-25 16:05:24 +00:00
},
/**
* Listen to 'filter:changed' event to filter the current list.
*/
2016-11-26 03:25:35 +00:00
'filter:changed': q => {
this.q = q
},
2016-06-25 16:05:24 +00:00
/**
* Clears the current list's selection if the user has switched to another view.
*/
'main-content-view:load': () => this.clearSelection(),
/**
* Listen to 'song:selection-clear' (often broadcasted from the direct parent)
* to clear the selected songs.
*/
2016-11-26 03:25:35 +00:00
'song:selection-clear': () => this.clearSelection()
})
}
}
2016-06-25 16:05:24 +00:00
</script>
2016-06-25 16:05:24 +00:00
<style lang="sass">
@import "../../../sass/partials/_vars.scss";
@import "../../../sass/partials/_mixins.scss";
.song-list-wrap {
position: relative;
table {
width: 100%;
table-layout: fixed;
}
tr.droppable {
border-bottom-width: 3px;
border-bottom-color: $colorGreen;
}
td, th {
text-align: left;
padding: 8px;
vertical-align: middle;
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
&.time {
width: 72px;
text-align: right;
}
2016-01-06 16:41:59 +00:00
2016-06-25 16:05:24 +00:00
&.track-number {
width: 42px;
}
2016-06-17 06:09:21 +00:00
2016-06-25 16:05:24 +00:00
&.artist {
width: 23%;
}
2016-06-17 06:09:21 +00:00
2016-06-25 16:05:24 +00:00
&.album {
width: 27%;
}
2016-03-24 04:37:39 +00:00
2016-06-25 16:05:24 +00:00
&.play {
display: none;
2016-01-07 01:39:38 +00:00
2016-06-25 16:05:24 +00:00
html.touchevents & {
display: block;
position: absolute;
top: 8px;
right: 4px;
}
}
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
th {
color: $color2ndText;
letter-spacing: 1px;
text-transform: uppercase;
cursor: pointer;
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
i {
color: $colorHighlight;
font-size: 1.2rem;
}
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
/**
* Since the Queue screen doesn't allow sorting, we reset the cursor style.
*/
&.queue th {
cursor: default;
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
@media only screen and (max-width: 768px) {
table, tbody, tr {
display: block;
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
thead, tfoot {
display: none;
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
tr {
padding: 8px 32px 8px 4px;
position: relative;
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
td {
display: inline;
padding: 0;
vertical-align: bottom;
2016-06-26 11:06:37 +00:00
white-space: normal;
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
&.album, &.time, &.track-number {
display: none;
}
2015-12-13 04:42:28 +00:00
2016-06-25 16:05:24 +00:00
&.artist {
opacity: .5;
font-size: .9rem;
padding: 0 4px;
}
2015-12-13 04:42:28 +00:00
}
2016-06-25 16:05:24 +00:00
}
}
2015-12-13 04:42:28 +00:00
</style>