| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213 | <template>	<div class='modal is-active'>		<div class='modal-background'></div>		<div class='modal-card'>			<header class='modal-card-head'>				<p class='modal-card-title'>Editing: {{ playlist.displayName }}</p>				<button class='delete' @click='$parent.toggleModal("editPlaylist")'></button>			</header>			<section class='modal-card-body'>				<aside class='menu' v-if='playlist.songs.length > 0'>					<ul class='menu-list'>						<li v-for='song in playlist.songs' track-by='$index'>							<a :href='' target='_blank'>{{ song.title }} - {{ song.artists.join(', ') }}</a>							<div class='controls'>								<a href='#'>									<i class='material-icons' v-if='playlist.songs[0] !== song' @click='promoteSong($index)'>keyboard_arrow_up</i>									<i class='material-icons' v-else>error</i>								</a>								<a href='#' @click=''>									<i class='material-icons' v-if='playlist.songs.length - 1 !== $index' @click='demoteSong($index)'>keyboard_arrow_down</i>									<i class='material-icons' v-else>error</i>								</a>								<a href='#' @click='removeSongFromPlaylist(song._id)'><i class='material-icons'>delete</i></a>							</div>						</li>					</ul>					<br />				</aside>				<div class='control is-grouped'>					<p class='control is-expanded'>						<input class='input' type='text' placeholder='Search for Song to add' v-model='songQuery'>					</p>					<p class='control'>						<a class='button is-info' @click='searchForSongs()'>Search</a>					</p>				</div>				<table class='table' v-if='songQueryResults.length > 0'>					<tbody>						<tr v-for='result in songQueryResults'>							<td>								<img :src='result.thumbnail' />							</td>							<td>{{ result.title }}</td>							<td>								<a class='button is-success' @click='addSongToPlaylist(result.id)'>									Add								</a>							</td>						</tr>					</tbody>				</table>				<div class='control is-grouped'>					<p class='control is-expanded'>						<input class='input' type='text' placeholder='YouTube Playlist URL' v-model='importQuery'>					</p>					<p class='control'>						<a class='button is-info' @click='importPlaylist()'>Import</a>					</p>				</div>				<h5>Edit playlist details:</h5>				<div class='control is-grouped'>					<p class='control is-expanded'>						<input class='input' type='text' placeholder='Playlist Display Name' v-model='playlist.displayName'>					</p>					<p class='control'>						<a class='button is-info' @click='renamePlaylist()'>Rename</a>					</p>				</div>				<div class='control is-grouped'>					<p class='control is-expanded'>						<input class='input' type='text' placeholder='Playlist ID' v-model='playlist._id'>					</p>					<p class='control'>						<a class='button is-info' @click='renamePlaylistId()'>Rename</a>					</p>				</div>			</section>			<footer class='modal-card-foot'>				<a class='button is-danger' @click='removePlaylist()'>Remove Playlist</a>			</footer>		</div>	</div></template><script>	import { Toast } from 'vue-roaster';	export default {		data() {			return {				playlist: {},				songQueryResults: []			}		},		methods: {			searchForSongs: function () {				let _this = this;				_this.socket.emit('apis.searchYoutube', _this.songQuery, res => {					if (res.status == 'success') {						_this.songQueryResults = [];						for (let i = 0; i < res.data.items.length; i++) {							_this.songQueryResults.push({								id: res.data.items[i].id.videoId,								url: `https://www.youtube.com/watch?v=${this.id}`,								title: res.data.items[i].snippet.title,								thumbnail: res.data.items[i].snippet.thumbnails.default.url							});						}					} else if (res.status == 'error') Toast.methods.addToast(res.message, 3000);				});			},			addSongToPlaylist: function (id) {				let _this = this;				_this.socket.emit('playlists.addSongToPlaylist', id, _this.playlist._id, res => {					if (res.status == 'success') {						Toast.methods.addToast(res.message, 3000);						_this.playlist.songs = res.data;					}				});			},			importPlaylist: function () {				let _this = this;				this.socket.emit('playlists.addSetToPlaylist', _this.importQuery, _this.playlist._id, res => {					if (res.status == 'success') _this.playlist.songs = res.data;				});			},			removeSongFromPlaylist: function (id) {				let _this = this;				this.socket.emit('playlists.removeSongFromPlaylist', id, _this.playlist._id, res => {					if (res.status == 'success') {						Toast.methods.addToast(res.message, 3000);						_this.playlist.songs = res.data;					}				});			},			renamePlaylist: function () {				this.socket.emit('playlists.updateDisplayName', this.playlist._id, this.playlist.displayName, res => {					if (res.status == 'success') Toast.methods.addToast(res.message, 3000);				});			},			renamePlaylistId: function () {				let _this = this;				_this.socket.emit('playlists.updatePlaylistId', _this.playlist.oldId, _this.playlist._id, res => {					if (res.status == 'success') _this.playlist = res.data;				});			},			removePlaylist: function () {				let _this = this;				_this.socket.emit('playlists.remove', _this.playlist._id, res => {					if (res.status == 'success') {						Toast.methods.addToast(res.message, 3000);						_this.$parent.toggleModal('editPlaylist');					}				});			},			promoteSong: function (fromIndex) {				let _this = this;				_this.socket.emit('playlists.promoteSong', _this.playlist._id, fromIndex, res => {					if (res.status == 'success') _this.$set('playlist.songs', res.data); // bug: v-for is not updating				});			},			demoteSong: function (fromIndex) {				let _this = this;				_this.socket.emit('playlists.demoteSong', _this.playlist._id, fromIndex, res => {					if (res.status == 'success') _this.$set('playlist.songs', res.data); // bug: v-for is not updating				});			}		},		ready: function () {			let _this = this;			let socketInterval = setInterval(() => {				if (!!_this.$parent.$parent.socket) {					_this.socket = _this.$parent.$parent.socket;					_this.socket.emit('playlists.getPlaylist', _this.$parent.playlistBeingEdited, res => {						if (res.status == 'success') _this.playlist = res.data; _this.playlist.oldId = res.data._id;					});					clearInterval(socketInterval);				}			}, 100);		}	}</script><style type='scss' scoped>	.menu { padding: 0 20px; }	.menu-list li {		display: flex;		justify-content: space-between;	}	.menu-list a:hover { color: #000 !important; }	li a {		display: flex;    	align-items: center;	}	.controls {		display: flex;		a {			display: flex;    		align-items: center;		}	}	.table {		margin-bottom: 0;	}	h5 { padding: 20px 0; }</style>
 |