| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303 | <template>	<div class="sidebar" transition="slide">		<div class="inner-wrapper">			<div v-if="station.type === 'community'" class="title">				Queue			</div>			<div v-else class="title">				Playlist			</div>			<article v-if="!noSong" class="media">				<figure v-if="currentSong.thumbnail" class="media-left">					<p class="image is-64x64">						<img							:src="currentSong.thumbnail"							onerror="this.src='/assets/notes-transparent.png'"						/>					</p>				</figure>				<div class="media-content">					<div class="content">						<p>							Current Song:							<strong>{{ currentSong.title }}</strong>							<br />							<small>{{ currentSong.artists }}</small>						</p>					</div>				</div>				<div class="media-right">					{{ utils.formatTime(currentSong.duration) }}				</div>			</article>			<p v-if="noSong" class="center">				There is currently no song playing.			</p>			<article				v-else				v-for="song in songsList"				:key="song.songId"				class="media"			>				<div class="media-content">					<div						class="content"						style="display: block;padding-top: 10px;"					>						<strong class="songTitle">{{ song.title }}</strong>						<small>{{ song.artists.join(", ") }}</small>						<div							v-if="								station.type === 'community' &&									station.partyMode === true							"						>							<small>								Requested by								<b>									<user-id-to-username										:userId="song.requestedBy"										:link="true"									/>								</b>							</small>							<i								v-if="isOwnerOnly() || isAdminOnly()"								class="material-icons"								style="vertical-align: middle;"								@click="removeFromQueue(song.songId)"								>delete_forever</i							>						</div>					</div>				</div>				<div class="media-right">					{{ utils.formatTime(song.duration) }}				</div>			</article>			<div				v-if="					station.type === 'community' &&						loggedIn &&						station.partyMode === true				"			>				<button					v-if="						(station.locked && isOwnerOnly()) ||							!station.locked ||							(station.locked &&								isAdminOnly() &&								dismissedWarning)					"					class="button add-to-queue"					@click="						openModal({							sector: 'station',							modal: 'addSongToQueue'						})					"				>					Add Song to Queue				</button>				<button					v-if="						station.locked &&							isAdminOnly() &&							!isOwnerOnly() &&							!dismissedWarning					"					class="button add-to-queue add-to-queue-warning"					@click="dismissedWarning = true"				>					THIS STATION'S QUEUE IS LOCKED.				</button>				<button					v-if="station.locked && !isAdminOnly() && !isOwnerOnly()"					class="button add-to-queue add-to-queue-disabled"				>					THIS STATION'S QUEUE IS LOCKED.				</button>			</div>		</div>	</div></template><script>import { mapState, mapActions } from "vuex";import Toast from "toasters";import utils from "../../js/utils";import UserIdToUsername from "../UserIdToUsername.vue";export default {	data() {		return {			utils,			dismissedWarning: false		};	},	computed: mapState({		loggedIn: state => state.user.auth.loggedIn,		userId: state => state.user.auth.userId,		role: state => state.user.auth.role,		station: state => state.station.station,		currentSong: state => state.station.currentSong,		songsList: state => state.station.songsList,		noSong: state => state.station.noSong	}),	methods: {		isOwnerOnly() {			return this.loggedIn && this.userId === this.station.owner;		},		isAdminOnly() {			return this.loggedIn && this.role === "admin";		},		removeFromQueue(songId) {			window.socket.emit(				"stations.removeFromQueue",				this.station._id,				songId,				res => {					if (res.status === "success") {						new Toast({							content:								"Successfully removed song from the queue.",							timeout: 4000						});					} else new Toast({ content: res.message, timeout: 8000 });				}			);		},		...mapActions("modals", ["openModal"])	},	mounted() {		/*			io.getSocket((socket) => {				this.socket = socket;			}); */	},	components: { UserIdToUsername }};</script><style lang="scss" scoped>@import "styles/global.scss";.night-mode {	.sidebar {		background-color: $night-mode-secondary;		.title {			color: #fff;		}		* {			color: #ddd;		}	}}.sidebar {	position: fixed;	z-index: 1;	top: 0;	right: 0;	width: 300px;	height: 100vh;	background-color: $white;	box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.16),		0 2px 10px 0 rgba(0, 0, 0, 0.12);}.inner-wrapper {	top: 60px;	position: relative;	overflow: auto;	height: 100%;}.slide-transition {	transition: transform 0.6s ease-in-out;	transform: translateX(0);}.slide-enter,.slide-leave {	transform: translateX(100%);}.title {	background-color: rgb(3, 169, 244);	text-align: center;	padding: 10px;	color: $white;	font-weight: 600;}.media {	padding: 0 25px;}.media-content .content {	min-height: 64px;	display: flex;	align-items: center;}.content p strong {	word-break: break-word;}.content p small {	word-break: break-word;}.add-to-queue {	width: 100%;	margin-top: 25px;	height: 40px;	border-radius: 0;	background: rgb(3, 169, 244);	color: $white !important;	border: 0;	&:active,	&:focus {		border: 0;	}}.add-to-queue.add-to-queue-warning {	background-color: red;}.add-to-queue.add-to-queue-disabled {	background-color: gray;}.add-to-queue.add-to-queue-disabled:focus {	background-color: gray;}.add-to-queue:focus {	background: $primary-color;}.media-right {	line-height: 64px;}.songTitle {	word-wrap: break-word;	overflow: hidden;	text-overflow: ellipsis;	display: -webkit-box;	-webkit-box-orient: vertical;	-webkit-line-clamp: 2;	line-height: 20px;	max-height: 40px;}</style>
 |