Playlists.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. <template>
  2. <div class="station-playlists">
  3. <div class="tabs-container">
  4. <div class="tab-selection">
  5. <button
  6. class="button is-default"
  7. ref="current-tab"
  8. :class="{ selected: tab === 'current' }"
  9. @click="showTab('current')"
  10. >
  11. Current
  12. </button>
  13. <button
  14. class="button is-default"
  15. ref="search-tab"
  16. :class="{ selected: tab === 'search' }"
  17. @click="showTab('search')"
  18. >
  19. Search
  20. </button>
  21. <button
  22. v-if="station.type === 'community'"
  23. class="button is-default"
  24. ref="my-playlists-tab"
  25. :class="{ selected: tab === 'my-playlists' }"
  26. @click="showTab('my-playlists')"
  27. >
  28. My Playlists
  29. </button>
  30. </div>
  31. <div class="tab" v-show="tab === 'current'">
  32. <div v-if="currentPlaylists.length > 0">
  33. <playlist-item
  34. v-for="playlist in currentPlaylists"
  35. :key="`key-${playlist._id}`"
  36. :playlist="playlist"
  37. :show-owner="true"
  38. >
  39. <div class="icons-group" slot="actions">
  40. <confirm
  41. v-if="isOwnerOrAdmin()"
  42. @confirm="deselectPlaylist(playlist._id)"
  43. >
  44. <i
  45. class="material-icons stop-icon"
  46. content="Stop playing songs from this playlist"
  47. v-tippy
  48. >
  49. stop
  50. </i>
  51. </confirm>
  52. <confirm
  53. v-if="isOwnerOrAdmin()"
  54. @confirm="blacklistPlaylist(playlist._id)"
  55. >
  56. <i
  57. class="material-icons stop-icon"
  58. content="Blacklist Playlist"
  59. v-tippy
  60. >block</i
  61. >
  62. </confirm>
  63. <i
  64. v-if="playlist.createdBy === myUserId"
  65. @click="showPlaylist(playlist._id)"
  66. class="material-icons edit-icon"
  67. content="Edit Playlist"
  68. v-tippy
  69. >edit</i
  70. >
  71. <i
  72. v-if="
  73. playlist.createdBy !== myUserId &&
  74. (playlist.privacy === 'public' ||
  75. isAdmin())
  76. "
  77. @click="showPlaylist(playlist._id)"
  78. class="material-icons edit-icon"
  79. content="View Playlist"
  80. v-tippy
  81. >visibility</i
  82. >
  83. </div>
  84. </playlist-item>
  85. </div>
  86. <p v-else class="has-text-centered scrollable-list">
  87. No playlists currently selected.
  88. </p>
  89. </div>
  90. <div class="tab" v-show="tab === 'search'">
  91. <label class="label"> Search for a public playlist </label>
  92. <div class="control is-grouped input-with-button">
  93. <p class="control is-expanded">
  94. <input
  95. class="input"
  96. type="text"
  97. placeholder="Enter your playlist query here..."
  98. v-model="search.query"
  99. @keyup.enter="searchForPlaylists(1)"
  100. />
  101. </p>
  102. <p class="control">
  103. <a class="button is-info" @click="searchForPlaylists(1)"
  104. ><i class="material-icons icon-with-button"
  105. >search</i
  106. >Search</a
  107. >
  108. </p>
  109. </div>
  110. <div v-if="search.results.length > 0">
  111. <playlist-item
  112. v-for="playlist in search.results"
  113. :key="`searchKey-${playlist._id}`"
  114. :playlist="playlist"
  115. :show-owner="true"
  116. >
  117. <div class="icons-group" slot="actions">
  118. <i
  119. v-if="isExcluded(playlist._id)"
  120. class="material-icons stop-icon"
  121. content="This playlist is blacklisted in this station"
  122. v-tippy="{ theme: 'info' }"
  123. >play_disabled</i
  124. >
  125. <confirm
  126. v-if="
  127. (isOwnerOrAdmin() ||
  128. (station.type === 'community' &&
  129. station.partyMode)) &&
  130. isSelected(playlist._id)
  131. "
  132. @confirm="deselectPlaylist(playlist._id)"
  133. >
  134. <i
  135. class="material-icons stop-icon"
  136. content="Stop playing songs from this playlist"
  137. v-tippy
  138. >
  139. stop
  140. </i>
  141. </confirm>
  142. <i
  143. v-if="
  144. (isOwnerOrAdmin() ||
  145. (station.type === 'community' &&
  146. station.partyMode)) &&
  147. !isSelected(playlist._id) &&
  148. !isExcluded(playlist._id)
  149. "
  150. @click="selectPlaylist(playlist)"
  151. class="material-icons play-icon"
  152. :content="
  153. station.partyMode
  154. ? 'Request songs from this playlist'
  155. : 'Play songs from this playlist'
  156. "
  157. v-tippy
  158. >play_arrow</i
  159. >
  160. <confirm
  161. v-if="
  162. isOwnerOrAdmin() &&
  163. !isExcluded(playlist._id)
  164. "
  165. @confirm="blacklistPlaylist(playlist._id)"
  166. >
  167. <i
  168. class="material-icons stop-icon"
  169. content="Blacklist Playlist"
  170. v-tippy
  171. >block</i
  172. >
  173. </confirm>
  174. <i
  175. v-if="playlist.createdBy === myUserId"
  176. @click="showPlaylist(playlist._id)"
  177. class="material-icons edit-icon"
  178. content="Edit Playlist"
  179. v-tippy
  180. >edit</i
  181. >
  182. <i
  183. v-if="
  184. playlist.createdBy !== myUserId &&
  185. (playlist.privacy === 'public' ||
  186. isAdmin())
  187. "
  188. @click="showPlaylist(playlist._id)"
  189. class="material-icons edit-icon"
  190. content="View Playlist"
  191. v-tippy
  192. >visibility</i
  193. >
  194. </div>
  195. </playlist-item>
  196. <button
  197. v-if="resultsLeftCount > 0"
  198. class="button is-primary load-more-button"
  199. @click="searchForPlaylists(search.page + 1)"
  200. >
  201. Load {{ nextPageResultsCount }} more results
  202. </button>
  203. </div>
  204. </div>
  205. <div
  206. v-if="station.type === 'community'"
  207. class="tab"
  208. v-show="tab === 'my-playlists'"
  209. >
  210. <button
  211. class="button is-primary"
  212. id="create-new-playlist-button"
  213. @click="openModal('createPlaylist')"
  214. >
  215. Create new playlist
  216. </button>
  217. <draggable
  218. class="menu-list scrollable-list"
  219. v-if="playlists.length > 0"
  220. v-model="playlists"
  221. v-bind="dragOptions"
  222. @start="drag = true"
  223. @end="drag = false"
  224. @change="savePlaylistOrder"
  225. >
  226. <transition-group
  227. type="transition"
  228. :name="!drag ? 'draggable-list-transition' : null"
  229. >
  230. <playlist-item
  231. class="item-draggable"
  232. v-for="playlist in playlists"
  233. :key="playlist._id"
  234. :playlist="playlist"
  235. >
  236. <div slot="actions">
  237. <i
  238. v-if="isExcluded(playlist._id)"
  239. class="material-icons stop-icon"
  240. content="This playlist is blacklisted in this station"
  241. v-tippy="{ theme: 'info' }"
  242. >play_disabled</i
  243. >
  244. <i
  245. v-if="
  246. station.type === 'community' &&
  247. (isOwnerOrAdmin() ||
  248. station.partyMode) &&
  249. !isSelected(playlist._id) &&
  250. !isExcluded(playlist._id)
  251. "
  252. @click="selectPlaylist(playlist)"
  253. class="material-icons play-icon"
  254. :content="
  255. station.partyMode
  256. ? 'Request songs from this playlist'
  257. : 'Play songs from this playlist'
  258. "
  259. v-tippy
  260. >play_arrow</i
  261. >
  262. <confirm
  263. v-if="
  264. station.type === 'community' &&
  265. (isOwnerOrAdmin() ||
  266. station.partyMode) &&
  267. isSelected(playlist._id)
  268. "
  269. @confirm="deselectPlaylist(playlist._id)"
  270. >
  271. <i
  272. class="material-icons stop-icon"
  273. :content="
  274. station.partyMode
  275. ? 'Stop requesting songs from this playlist'
  276. : 'Stop playing songs from this playlist'
  277. "
  278. v-tippy
  279. >stop</i
  280. >
  281. </confirm>
  282. <confirm
  283. v-if="
  284. isOwnerOrAdmin() &&
  285. !isExcluded(playlist._id)
  286. "
  287. @confirm="blacklistPlaylist(playlist._id)"
  288. >
  289. <i
  290. class="material-icons stop-icon"
  291. content="Blacklist Playlist"
  292. v-tippy
  293. >block</i
  294. >
  295. </confirm>
  296. <i
  297. @click="showPlaylist(playlist._id)"
  298. class="material-icons edit-icon"
  299. content="Edit Playlist"
  300. v-tippy
  301. >edit</i
  302. >
  303. </div>
  304. </playlist-item>
  305. </transition-group>
  306. </draggable>
  307. <p v-else class="has-text-centered scrollable-list">
  308. You don't have any playlists!
  309. </p>
  310. </div>
  311. </div>
  312. </div>
  313. </template>
  314. <script>
  315. import { mapActions, mapState, mapGetters } from "vuex";
  316. import Toast from "toasters";
  317. import PlaylistItem from "@/components/PlaylistItem.vue";
  318. import Confirm from "@/components/Confirm.vue";
  319. import SortablePlaylists from "@/mixins/SortablePlaylists.vue";
  320. export default {
  321. components: {
  322. PlaylistItem,
  323. Confirm
  324. },
  325. mixins: [SortablePlaylists],
  326. data() {
  327. return {
  328. tab: "current",
  329. search: {
  330. query: "",
  331. searchedQuery: "",
  332. page: 0,
  333. count: 0,
  334. resultsLeft: 0,
  335. results: []
  336. }
  337. };
  338. },
  339. computed: {
  340. currentPlaylists() {
  341. if (this.station.type === "community" && this.station.partyMode) {
  342. return this.partyPlaylists;
  343. }
  344. return this.includedPlaylists;
  345. },
  346. resultsLeftCount() {
  347. return this.search.count - this.search.results.length;
  348. },
  349. nextPageResultsCount() {
  350. return Math.min(this.search.pageSize, this.resultsLeftCount);
  351. },
  352. ...mapState({
  353. loggedIn: state => state.user.auth.loggedIn,
  354. role: state => state.user.auth.role,
  355. userId: state => state.user.auth.userId,
  356. partyPlaylists: state => state.station.partyPlaylists
  357. }),
  358. ...mapState("modals/manageStation", {
  359. originalStation: state => state.originalStation,
  360. station: state => state.station,
  361. includedPlaylists: state => state.includedPlaylists,
  362. excludedPlaylists: state => state.excludedPlaylists,
  363. songsList: state => state.songsList
  364. }),
  365. ...mapGetters({
  366. socket: "websockets/getSocket"
  367. })
  368. },
  369. mounted() {
  370. this.socket.dispatch("playlists.indexMyPlaylists", true, res => {
  371. if (res.status === "success") this.setPlaylists(res.data.playlists);
  372. this.orderOfPlaylists = this.calculatePlaylistOrder(); // order in regards to the database
  373. });
  374. this.socket.dispatch(
  375. `stations.getStationIncludedPlaylistsById`,
  376. this.station._id,
  377. res => {
  378. if (res.status === "success") {
  379. this.station.includedPlaylists = res.data.playlists;
  380. this.originalStation.includedPlaylists = res.data.playlists;
  381. }
  382. }
  383. );
  384. this.socket.dispatch(
  385. `stations.getStationExcludedPlaylistsById`,
  386. this.station._id,
  387. res => {
  388. if (res.status === "success") {
  389. this.station.excludedPlaylists = res.data.playlists;
  390. this.originalStation.excludedPlaylists = res.data.playlists;
  391. }
  392. }
  393. );
  394. },
  395. methods: {
  396. showTab(tab) {
  397. this.$refs[`${tab}-tab`].scrollIntoView();
  398. this.tab = tab;
  399. },
  400. isOwner() {
  401. return this.loggedIn && this.userId === this.station.owner;
  402. },
  403. isAdmin() {
  404. return this.loggedIn && this.role === "admin";
  405. },
  406. isOwnerOrAdmin() {
  407. return this.isOwner() || this.isAdmin();
  408. },
  409. showPlaylist(playlistId) {
  410. this.editPlaylist(playlistId);
  411. this.openModal("editPlaylist");
  412. },
  413. selectPlaylist(playlist) {
  414. if (this.station.type === "community" && this.station.partyMode) {
  415. if (!this.isSelected(playlist.id)) {
  416. this.partyPlaylists.push(playlist);
  417. this.addPartyPlaylistSongToQueue();
  418. new Toast(
  419. "Successfully selected playlist to auto request songs."
  420. );
  421. } else {
  422. new Toast("Error: Playlist already selected.");
  423. }
  424. } else {
  425. this.socket.dispatch(
  426. "stations.includePlaylist",
  427. this.station._id,
  428. playlist._id,
  429. res => {
  430. new Toast(res.message);
  431. }
  432. );
  433. }
  434. },
  435. deselectPlaylist(id) {
  436. return new Promise(resolve => {
  437. if (
  438. this.station.type === "community" &&
  439. this.station.partyMode
  440. ) {
  441. let selected = false;
  442. this.currentPlaylists.forEach((playlist, index) => {
  443. if (playlist._id === id) {
  444. selected = true;
  445. this.partyPlaylists.splice(index, 1);
  446. }
  447. });
  448. if (selected) {
  449. new Toast("Successfully deselected playlist.");
  450. resolve();
  451. } else {
  452. new Toast("Playlist not selected.");
  453. resolve();
  454. }
  455. } else {
  456. this.socket.dispatch(
  457. "stations.removeIncludedPlaylist",
  458. this.station._id,
  459. id,
  460. res => {
  461. new Toast(res.message);
  462. resolve();
  463. }
  464. );
  465. }
  466. });
  467. },
  468. isSelected(id) {
  469. let selected = false;
  470. this.currentPlaylists.forEach(playlist => {
  471. if (playlist._id === id) selected = true;
  472. });
  473. return selected;
  474. },
  475. isExcluded(id) {
  476. let selected = false;
  477. this.excludedPlaylists.forEach(playlist => {
  478. if (playlist._id === id) selected = true;
  479. });
  480. return selected;
  481. },
  482. searchForPlaylists(page) {
  483. if (
  484. this.search.page >= page ||
  485. this.search.searchedQuery !== this.search.query
  486. ) {
  487. this.search.results = [];
  488. this.search.page = 0;
  489. this.search.count = 0;
  490. this.search.resultsLeft = 0;
  491. this.search.pageSize = 0;
  492. }
  493. const { query } = this.search;
  494. const action =
  495. this.station.type === "official"
  496. ? "playlists.searchOfficial"
  497. : "playlists.searchCommunity";
  498. this.search.searchedQuery = this.search.query;
  499. this.socket.dispatch(action, query, page, res => {
  500. const { data } = res;
  501. const { count, pageSize, playlists } = data;
  502. if (res.status === "success") {
  503. this.search.results = [
  504. ...this.search.results,
  505. ...playlists
  506. ];
  507. this.search.page = page;
  508. this.search.count = count;
  509. this.search.resultsLeft =
  510. count - this.search.results.length;
  511. this.search.pageSize = pageSize;
  512. } else if (res.status === "error") {
  513. this.search.results = [];
  514. this.search.page = 0;
  515. this.search.count = 0;
  516. this.search.resultsLeft = 0;
  517. this.search.pageSize = 0;
  518. new Toast(res.message);
  519. }
  520. });
  521. },
  522. async blacklistPlaylist(id) {
  523. if (this.isSelected(id)) await this.deselectPlaylist(id);
  524. this.socket.dispatch(
  525. "stations.excludePlaylist",
  526. this.station._id,
  527. id,
  528. res => {
  529. new Toast(res.message);
  530. }
  531. );
  532. },
  533. addPartyPlaylistSongToQueue() {
  534. let isInQueue = false;
  535. if (
  536. this.station.type === "community" &&
  537. this.station.partyMode === true
  538. ) {
  539. this.songsList.forEach(queueSong => {
  540. if (queueSong.requestedBy === this.userId) isInQueue = true;
  541. });
  542. if (!isInQueue && this.partyPlaylists) {
  543. const selectedPlaylist = this.partyPlaylists[
  544. Math.floor(Math.random() * this.partyPlaylists.length)
  545. ];
  546. if (
  547. selectedPlaylist._id &&
  548. selectedPlaylist.songs.length > 0
  549. ) {
  550. const selectedSong =
  551. selectedPlaylist.songs[
  552. Math.floor(
  553. Math.random() *
  554. selectedPlaylist.songs.length
  555. )
  556. ];
  557. if (selectedSong.youtubeId) {
  558. this.socket.dispatch(
  559. "stations.addToQueue",
  560. this.station._id,
  561. selectedSong.youtubeId,
  562. data => {
  563. if (data.status !== "success")
  564. new Toast("Error auto queueing song");
  565. }
  566. );
  567. }
  568. }
  569. }
  570. }
  571. },
  572. ...mapActions("station", ["updatePartyPlaylists"]),
  573. ...mapActions("modalVisibility", ["openModal"]),
  574. ...mapActions("user/playlists", ["editPlaylist", "setPlaylists"])
  575. }
  576. };
  577. </script>
  578. <style lang="scss" scoped>
  579. .station-playlists {
  580. .tabs-container {
  581. .tab-selection {
  582. display: flex;
  583. overflow-x: auto;
  584. .button {
  585. border-radius: 0;
  586. border: 0;
  587. text-transform: uppercase;
  588. font-size: 14px;
  589. color: var(--dark-grey-3);
  590. background-color: var(--light-grey-2);
  591. flex-grow: 1;
  592. height: 32px;
  593. &:not(:first-of-type) {
  594. margin-left: 5px;
  595. }
  596. }
  597. .selected {
  598. background-color: var(--primary-color) !important;
  599. color: var(--white) !important;
  600. font-weight: 600;
  601. }
  602. }
  603. .tab {
  604. padding: 15px 0;
  605. border-radius: 0;
  606. .playlist-item:not(:last-of-type),
  607. .item.item-draggable:not(:last-of-type) {
  608. margin-bottom: 10px;
  609. }
  610. .load-more-button {
  611. width: 100%;
  612. margin-top: 10px;
  613. }
  614. }
  615. }
  616. }
  617. .draggable-list-transition-move {
  618. transition: transform 0.5s;
  619. }
  620. .draggable-list-ghost {
  621. opacity: 0.5;
  622. filter: brightness(95%);
  623. }
  624. </style>