Playlists.vue 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038
  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="search-tab"
  8. :class="{ selected: tab === 'search' }"
  9. @click="showTab('search')"
  10. >
  11. Search
  12. </button>
  13. <button
  14. v-if="station.type === 'community'"
  15. class="button is-default"
  16. ref="my-playlists-tab"
  17. :class="{ selected: tab === 'my-playlists' }"
  18. @click="showTab('my-playlists')"
  19. >
  20. My Playlists
  21. </button>
  22. <button
  23. class="button is-default"
  24. ref="party-tab"
  25. :class="{ selected: tab === 'party' }"
  26. v-if="isPartyMode()"
  27. @click="showTab('party')"
  28. >
  29. Party
  30. </button>
  31. <button
  32. class="button is-default"
  33. ref="included-tab"
  34. :class="{ selected: tab === 'included' }"
  35. v-if="isPlaylistMode()"
  36. @click="showTab('included')"
  37. >
  38. Included
  39. </button>
  40. <button
  41. class="button is-default"
  42. ref="excluded-tab"
  43. :class="{ selected: tab === 'excluded' }"
  44. @click="showTab('excluded')"
  45. >
  46. Excluded
  47. </button>
  48. </div>
  49. <div class="tab" v-show="tab === 'search'">
  50. <label class="label"> Search for a public playlist </label>
  51. <div class="control is-grouped input-with-button">
  52. <p class="control is-expanded">
  53. <input
  54. class="input"
  55. type="text"
  56. placeholder="Enter your playlist query here..."
  57. v-model="search.query"
  58. @keyup.enter="searchForPlaylists(1)"
  59. />
  60. </p>
  61. <p class="control">
  62. <a class="button is-info" @click="searchForPlaylists(1)"
  63. ><i class="material-icons icon-with-button"
  64. >search</i
  65. >Search</a
  66. >
  67. </p>
  68. </div>
  69. <div v-if="search.results.length > 0">
  70. <playlist-item
  71. v-for="playlist in search.results"
  72. :key="`searchKey-${playlist._id}`"
  73. :playlist="playlist"
  74. :show-owner="true"
  75. >
  76. <template #item-icon>
  77. <i
  78. class="material-icons"
  79. v-if="
  80. isAllowedToParty() &&
  81. isSelected(playlist._id)
  82. "
  83. content="This playlist is currently selected"
  84. v-tippy
  85. >
  86. radio
  87. </i>
  88. <i
  89. class="material-icons"
  90. v-else-if="
  91. isOwnerOrAdmin() &&
  92. isPlaylistMode() &&
  93. isIncluded(playlist._id)
  94. "
  95. content="This playlist is currently included"
  96. v-tippy
  97. >
  98. play_arrow
  99. </i>
  100. <i
  101. class="material-icons excluded-icon"
  102. v-else-if="
  103. isOwnerOrAdmin() && isExcluded(playlist._id)
  104. "
  105. content="This playlist is currently excluded"
  106. v-tippy
  107. >
  108. block
  109. </i>
  110. <i
  111. class="material-icons"
  112. v-else
  113. :content="
  114. isPartyMode()
  115. ? 'This playlist is currently not selected or excluded'
  116. : 'This playlist is currently not included or excluded'
  117. "
  118. v-tippy
  119. >
  120. play_disabled
  121. </i>
  122. </template>
  123. <template #actions>
  124. <i
  125. v-if="isExcluded(playlist._id)"
  126. class="material-icons stop-icon"
  127. content="This playlist is blacklisted in this station"
  128. v-tippy="{ theme: 'info' }"
  129. >play_disabled</i
  130. >
  131. <confirm
  132. v-if="isPartyMode() && isSelected(playlist._id)"
  133. @confirm="deselectPartyPlaylist(playlist._id)"
  134. >
  135. <i
  136. class="material-icons stop-icon"
  137. content="Stop playing songs from this playlist"
  138. v-tippy
  139. >
  140. stop
  141. </i>
  142. </confirm>
  143. <confirm
  144. v-if="
  145. isOwnerOrAdmin() &&
  146. isPlaylistMode() &&
  147. isIncluded(playlist._id)
  148. "
  149. @confirm="removeIncludedPlaylist(playlist._id)"
  150. >
  151. <i
  152. class="material-icons stop-icon"
  153. content="Stop playing songs from this playlist"
  154. v-tippy
  155. >
  156. stop
  157. </i>
  158. </confirm>
  159. <i
  160. v-if="
  161. isPartyMode() &&
  162. !isSelected(playlist._id) &&
  163. !isExcluded(playlist._id)
  164. "
  165. @click="selectPartyPlaylist(playlist)"
  166. class="material-icons play-icon"
  167. content="Request songs from this playlist"
  168. v-tippy
  169. >play_arrow</i
  170. >
  171. <i
  172. v-if="
  173. isOwnerOrAdmin() &&
  174. isPlaylistMode() &&
  175. !isIncluded(playlist._id) &&
  176. !isExcluded(playlist._id)
  177. "
  178. @click="includePlaylist(playlist)"
  179. class="material-icons play-icon"
  180. :content="'Play songs from this playlist'"
  181. v-tippy
  182. >play_arrow</i
  183. >
  184. <confirm
  185. v-if="
  186. isOwnerOrAdmin() &&
  187. !isExcluded(playlist._id)
  188. "
  189. @confirm="blacklistPlaylist(playlist._id)"
  190. >
  191. <i
  192. class="material-icons stop-icon"
  193. content="Blacklist Playlist"
  194. v-tippy
  195. >block</i
  196. >
  197. </confirm>
  198. <confirm
  199. v-if="
  200. isOwnerOrAdmin() && isExcluded(playlist._id)
  201. "
  202. @confirm="removeExcludedPlaylist(playlist._id)"
  203. >
  204. <i
  205. class="material-icons stop-icon"
  206. content="Stop blacklisting songs from this playlist"
  207. v-tippy
  208. >
  209. stop
  210. </i>
  211. </confirm>
  212. <i
  213. v-if="playlist.createdBy === myUserId"
  214. @click="showPlaylist(playlist._id)"
  215. class="material-icons edit-icon"
  216. content="Edit Playlist"
  217. v-tippy
  218. >edit</i
  219. >
  220. <i
  221. v-if="
  222. playlist.createdBy !== myUserId &&
  223. (playlist.privacy === 'public' || isAdmin())
  224. "
  225. @click="showPlaylist(playlist._id)"
  226. class="material-icons edit-icon"
  227. content="View Playlist"
  228. v-tippy
  229. >visibility</i
  230. >
  231. </template>
  232. </playlist-item>
  233. <button
  234. v-if="resultsLeftCount > 0"
  235. class="button is-primary load-more-button"
  236. @click="searchForPlaylists(search.page + 1)"
  237. >
  238. Load {{ nextPageResultsCount }} more results
  239. </button>
  240. </div>
  241. </div>
  242. <div
  243. v-if="station.type === 'community'"
  244. class="tab"
  245. v-show="tab === 'my-playlists'"
  246. >
  247. <button
  248. class="button is-primary"
  249. id="create-new-playlist-button"
  250. @click="openModal('createPlaylist')"
  251. >
  252. Create new playlist
  253. </button>
  254. <div
  255. class="menu-list scrollable-list"
  256. v-if="playlists.length > 0"
  257. >
  258. <draggable
  259. tag="transition-group"
  260. :component-data="{
  261. name: !drag ? 'draggable-list-transition' : null
  262. }"
  263. item-key="_id"
  264. v-model="playlists"
  265. v-bind="dragOptions"
  266. @start="drag = true"
  267. @end="drag = false"
  268. @change="savePlaylistOrder"
  269. >
  270. <template #item="{ element }">
  271. <playlist-item
  272. class="item-draggable"
  273. :playlist="element"
  274. >
  275. <template #item-icon>
  276. <i
  277. class="material-icons"
  278. v-if="
  279. isAllowedToParty() &&
  280. isSelected(element._id)
  281. "
  282. content="This playlist is currently selected"
  283. v-tippy
  284. >
  285. radio
  286. </i>
  287. <i
  288. class="material-icons"
  289. v-else-if="
  290. isOwnerOrAdmin() &&
  291. isPlaylistMode() &&
  292. isIncluded(element._id)
  293. "
  294. content="This playlist is currently included"
  295. v-tippy
  296. >
  297. play_arrow
  298. </i>
  299. <i
  300. class="material-icons excluded-icon"
  301. v-else-if="
  302. isOwnerOrAdmin() &&
  303. isExcluded(element._id)
  304. "
  305. content="This playlist is currently excluded"
  306. v-tippy
  307. >
  308. block
  309. </i>
  310. <i
  311. class="material-icons"
  312. v-else
  313. :content="
  314. isPartyMode()
  315. ? 'This playlist is currently not selected or excluded'
  316. : 'This playlist is currently not included or excluded'
  317. "
  318. v-tippy
  319. >
  320. play_disabled
  321. </i>
  322. </template>
  323. <template #actions>
  324. <!-- <i
  325. v-if="isExcluded(playlist._id)"
  326. class="material-icons stop-icon"
  327. content="This playlist is blacklisted in this station"
  328. v-tippy="{ theme: 'info' }"
  329. >play_disabled</i
  330. > -->
  331. <i
  332. v-if="
  333. isPartyMode() &&
  334. !isSelected(element._id)
  335. "
  336. @click="selectPartyPlaylist(element)"
  337. class="material-icons play-icon"
  338. content="Request songs from this playlist"
  339. v-tippy
  340. >play_arrow</i
  341. >
  342. <i
  343. v-if="
  344. isPlaylistMode() &&
  345. isOwnerOrAdmin() &&
  346. !isSelected(element._id)
  347. "
  348. @click="includePlaylist(element)"
  349. class="material-icons play-icon"
  350. content="Play songs from this playlist"
  351. v-tippy
  352. >play_arrow</i
  353. >
  354. <confirm
  355. v-if="
  356. isPartyMode() &&
  357. isSelected(element._id)
  358. "
  359. @confirm="
  360. deselectPartyPlaylist(element._id)
  361. "
  362. >
  363. <i
  364. class="material-icons stop-icon"
  365. content="Stop requesting songs from this playlist"
  366. v-tippy
  367. >stop</i
  368. >
  369. </confirm>
  370. <confirm
  371. v-if="
  372. isPlaylistMode() &&
  373. isOwnerOrAdmin() &&
  374. isIncluded(element._id)
  375. "
  376. @confirm="
  377. removeIncludedPlaylist(element._id)
  378. "
  379. >
  380. <i
  381. class="material-icons stop-icon"
  382. content="Stop playing songs from this playlist"
  383. v-tippy
  384. >stop</i
  385. >
  386. </confirm>
  387. <confirm
  388. v-if="
  389. isOwnerOrAdmin() &&
  390. !isExcluded(element._id)
  391. "
  392. @confirm="
  393. blacklistPlaylist(element._id)
  394. "
  395. >
  396. <i
  397. class="material-icons stop-icon"
  398. content="Blacklist Playlist"
  399. v-tippy
  400. >block</i
  401. >
  402. </confirm>
  403. <confirm
  404. v-if="
  405. isOwnerOrAdmin() &&
  406. isExcluded(element._id)
  407. "
  408. @confirm="
  409. removeExcludedPlaylist(element._id)
  410. "
  411. >
  412. <i
  413. class="material-icons stop-icon"
  414. content="Stop blacklisting songs from this playlist"
  415. v-tippy
  416. >
  417. stop
  418. </i>
  419. </confirm>
  420. <i
  421. @click="showPlaylist(element._id)"
  422. class="material-icons edit-icon"
  423. content="Edit Playlist"
  424. v-tippy
  425. >edit</i
  426. >
  427. </template>
  428. </playlist-item>
  429. </template>
  430. </draggable>
  431. </div>
  432. <p v-else class="has-text-centered scrollable-list">
  433. You don't have any playlists!
  434. </p>
  435. </div>
  436. <div class="tab" v-show="tab === 'party'" v-if="isPartyMode()">
  437. <div v-if="partyPlaylists.length > 0">
  438. <playlist-item
  439. v-for="playlist in partyPlaylists"
  440. :key="`key-${playlist._id}`"
  441. :playlist="playlist"
  442. :show-owner="true"
  443. >
  444. <template #item-icon>
  445. <i
  446. class="material-icons"
  447. content="This playlist is currently selected"
  448. v-tippy
  449. >
  450. radio
  451. </i>
  452. </template>
  453. <template #actions>
  454. <confirm
  455. v-if="isOwnerOrAdmin()"
  456. @confirm="deselectPartyPlaylist(playlist._id)"
  457. >
  458. <i
  459. class="material-icons stop-icon"
  460. content="Stop playing songs from this playlist"
  461. v-tippy
  462. >
  463. stop
  464. </i>
  465. </confirm>
  466. <confirm
  467. v-if="isOwnerOrAdmin()"
  468. @confirm="blacklistPlaylist(playlist._id)"
  469. >
  470. <i
  471. class="material-icons stop-icon"
  472. content="Blacklist Playlist"
  473. v-tippy
  474. >block</i
  475. >
  476. </confirm>
  477. <i
  478. v-if="playlist.createdBy === myUserId"
  479. @click="showPlaylist(playlist._id)"
  480. class="material-icons edit-icon"
  481. content="Edit Playlist"
  482. v-tippy
  483. >edit</i
  484. >
  485. <i
  486. v-if="
  487. playlist.createdBy !== myUserId &&
  488. (playlist.privacy === 'public' || isAdmin())
  489. "
  490. @click="showPlaylist(playlist._id)"
  491. class="material-icons edit-icon"
  492. content="View Playlist"
  493. v-tippy
  494. >visibility</i
  495. >
  496. </template>
  497. </playlist-item>
  498. </div>
  499. <p v-else class="has-text-centered scrollable-list">
  500. No playlists currently being played.
  501. </p>
  502. </div>
  503. <div
  504. class="tab"
  505. v-show="tab === 'included'"
  506. v-if="isPlaylistMode()"
  507. >
  508. <div v-if="includedPlaylists.length > 0">
  509. <playlist-item
  510. v-for="playlist in includedPlaylists"
  511. :key="`key-${playlist._id}`"
  512. :playlist="playlist"
  513. :show-owner="true"
  514. >
  515. <template #item-icon>
  516. <i
  517. class="material-icons"
  518. content="This playlist is currently included"
  519. v-tippy
  520. >
  521. play_arrow
  522. </i>
  523. </template>
  524. <template #actions>
  525. <confirm
  526. v-if="isOwnerOrAdmin()"
  527. @confirm="removeIncludedPlaylist(playlist._id)"
  528. >
  529. <i
  530. class="material-icons stop-icon"
  531. content="Stop playing songs from this playlist"
  532. v-tippy
  533. >
  534. stop
  535. </i>
  536. </confirm>
  537. <confirm
  538. v-if="isOwnerOrAdmin()"
  539. @confirm="blacklistPlaylist(playlist._id)"
  540. >
  541. <i
  542. class="material-icons stop-icon"
  543. content="Blacklist Playlist"
  544. v-tippy
  545. >block</i
  546. >
  547. </confirm>
  548. <i
  549. v-if="playlist.createdBy === myUserId"
  550. @click="showPlaylist(playlist._id)"
  551. class="material-icons edit-icon"
  552. content="Edit Playlist"
  553. v-tippy
  554. >edit</i
  555. >
  556. <i
  557. v-if="
  558. playlist.createdBy !== myUserId &&
  559. (playlist.privacy === 'public' || isAdmin())
  560. "
  561. @click="showPlaylist(playlist._id)"
  562. class="material-icons edit-icon"
  563. content="View Playlist"
  564. v-tippy
  565. >visibility</i
  566. >
  567. </template>
  568. </playlist-item>
  569. </div>
  570. <p v-else class="has-text-centered scrollable-list">
  571. No playlists currently included.
  572. </p>
  573. </div>
  574. <div
  575. class="tab"
  576. v-show="tab === 'excluded'"
  577. v-if="isOwnerOrAdmin()"
  578. >
  579. <div v-if="excludedPlaylists.length > 0">
  580. <playlist-item
  581. :playlist="playlist"
  582. v-for="playlist in excludedPlaylists"
  583. :key="`key-${playlist._id}`"
  584. >
  585. <template #item-icon>
  586. <i
  587. class="material-icons excluded-icon"
  588. content="This playlist is currently excluded"
  589. v-tippy
  590. >
  591. block
  592. </i>
  593. </template>
  594. <template #actions>
  595. <confirm
  596. @confirm="removeExcludedPlaylist(playlist._id)"
  597. >
  598. <i
  599. class="material-icons stop-icon"
  600. content="Stop blacklisting songs from this playlist
  601. "
  602. v-tippy
  603. >stop</i
  604. >
  605. </confirm>
  606. <i
  607. v-if="playlist.createdBy === userId"
  608. @click="showPlaylist(playlist._id)"
  609. class="material-icons edit-icon"
  610. content="Edit Playlist"
  611. v-tippy
  612. >edit</i
  613. >
  614. <i
  615. v-else
  616. @click="showPlaylist(playlist._id)"
  617. class="material-icons edit-icon"
  618. content="View Playlist"
  619. v-tippy
  620. >visibility</i
  621. >
  622. </template>
  623. </playlist-item>
  624. </div>
  625. <p v-else class="has-text-centered scrollable-list">
  626. No playlists currently excluded.
  627. </p>
  628. </div>
  629. </div>
  630. </div>
  631. </template>
  632. <script>
  633. import { mapActions, mapState, mapGetters } from "vuex";
  634. import Toast from "toasters";
  635. import ws from "@/ws";
  636. import PlaylistItem from "@/components/PlaylistItem.vue";
  637. import Confirm from "@/components/Confirm.vue";
  638. import SortablePlaylists from "@/mixins/SortablePlaylists.vue";
  639. export default {
  640. components: {
  641. PlaylistItem,
  642. Confirm
  643. },
  644. mixins: [SortablePlaylists],
  645. data() {
  646. return {
  647. tab: "included",
  648. search: {
  649. query: "",
  650. searchedQuery: "",
  651. page: 0,
  652. count: 0,
  653. resultsLeft: 0,
  654. results: []
  655. }
  656. };
  657. },
  658. computed: {
  659. resultsLeftCount() {
  660. return this.search.count - this.search.results.length;
  661. },
  662. nextPageResultsCount() {
  663. return Math.min(this.search.pageSize, this.resultsLeftCount);
  664. },
  665. ...mapState({
  666. loggedIn: state => state.user.auth.loggedIn,
  667. role: state => state.user.auth.role,
  668. userId: state => state.user.auth.userId,
  669. partyPlaylists: state => state.station.partyPlaylists
  670. }),
  671. ...mapState("modals/manageStation", {
  672. parentTab: state => state.tab,
  673. originalStation: state => state.originalStation,
  674. station: state => state.station,
  675. includedPlaylists: state => state.includedPlaylists,
  676. excludedPlaylists: state => state.excludedPlaylists,
  677. songsList: state => state.songsList
  678. }),
  679. ...mapGetters({
  680. socket: "websockets/getSocket"
  681. })
  682. },
  683. watch: {
  684. // eslint-disable-next-line func-names
  685. parentTab(value) {
  686. if (value === "playlists") {
  687. if (this.tab === "included" && this.isPartyMode()) {
  688. this.showTab("party");
  689. } else if (this.tab === "party" && this.isPlaylistMode()) {
  690. this.showTab("included");
  691. }
  692. }
  693. }
  694. },
  695. mounted() {
  696. if (this.station.type === "community" && this.station.partyMode)
  697. this.showTab("search");
  698. ws.onConnect(this.init);
  699. },
  700. methods: {
  701. init() {
  702. this.socket.dispatch("playlists.indexMyPlaylists", true, res => {
  703. if (res.status === "success")
  704. this.setPlaylists(res.data.playlists);
  705. this.orderOfPlaylists = this.calculatePlaylistOrder(); // order in regards to the database
  706. });
  707. this.socket.dispatch(
  708. `stations.getStationIncludedPlaylistsById`,
  709. this.station._id,
  710. res => {
  711. if (res.status === "success") {
  712. this.station.includedPlaylists = res.data.playlists;
  713. this.originalStation.includedPlaylists =
  714. res.data.playlists;
  715. }
  716. }
  717. );
  718. this.socket.dispatch(
  719. `stations.getStationExcludedPlaylistsById`,
  720. this.station._id,
  721. res => {
  722. if (res.status === "success") {
  723. this.station.excludedPlaylists = res.data.playlists;
  724. this.originalStation.excludedPlaylists =
  725. res.data.playlists;
  726. }
  727. }
  728. );
  729. },
  730. showTab(tab) {
  731. this.$refs[`${tab}-tab`].scrollIntoView({ block: "nearest" });
  732. this.tab = tab;
  733. },
  734. isOwner() {
  735. return (
  736. this.loggedIn &&
  737. this.station &&
  738. this.userId === this.station.owner
  739. );
  740. },
  741. isAdmin() {
  742. return this.loggedIn && this.role === "admin";
  743. },
  744. isOwnerOrAdmin() {
  745. return this.isOwner() || this.isAdmin();
  746. },
  747. isPartyMode() {
  748. return (
  749. this.station &&
  750. this.station.type === "community" &&
  751. this.station.partyMode
  752. );
  753. },
  754. isAllowedToParty() {
  755. return (
  756. this.station &&
  757. this.isPartyMode() &&
  758. (!this.station.locked || this.isOwnerOrAdmin()) &&
  759. this.loggedIn
  760. );
  761. },
  762. isPlaylistMode() {
  763. return this.station && !this.isPartyMode();
  764. },
  765. showPlaylist(playlistId) {
  766. this.editPlaylist(playlistId);
  767. this.openModal("editPlaylist");
  768. },
  769. selectPartyPlaylist(playlist) {
  770. if (!this.isSelected(playlist.id)) {
  771. this.partyPlaylists.push(playlist);
  772. this.addPartyPlaylistSongToQueue();
  773. new Toast(
  774. "Successfully selected playlist to auto request songs."
  775. );
  776. } else {
  777. new Toast("Error: Playlist already selected.");
  778. }
  779. },
  780. includePlaylist(playlist) {
  781. this.socket.dispatch(
  782. "stations.includePlaylist",
  783. this.station._id,
  784. playlist._id,
  785. res => {
  786. new Toast(res.message);
  787. }
  788. );
  789. },
  790. deselectPartyPlaylist(id) {
  791. return new Promise(resolve => {
  792. let selected = false;
  793. this.partyPlaylists.forEach((playlist, index) => {
  794. if (playlist._id === id) {
  795. selected = true;
  796. this.partyPlaylists.splice(index, 1);
  797. }
  798. });
  799. if (selected) {
  800. new Toast("Successfully deselected playlist.");
  801. resolve();
  802. } else {
  803. new Toast("Playlist not selected.");
  804. resolve();
  805. }
  806. });
  807. },
  808. removeIncludedPlaylist(id) {
  809. return new Promise(resolve => {
  810. this.socket.dispatch(
  811. "stations.removeIncludedPlaylist",
  812. this.station._id,
  813. id,
  814. res => {
  815. new Toast(res.message);
  816. resolve();
  817. }
  818. );
  819. });
  820. },
  821. removeExcludedPlaylist(id) {
  822. return new Promise(resolve => {
  823. this.socket.dispatch(
  824. "stations.removeExcludedPlaylist",
  825. this.station._id,
  826. id,
  827. res => {
  828. new Toast(res.message);
  829. resolve();
  830. }
  831. );
  832. });
  833. },
  834. isSelected(id) {
  835. let selected = false;
  836. this.partyPlaylists.forEach(playlist => {
  837. if (playlist._id === id) selected = true;
  838. });
  839. return selected;
  840. },
  841. isIncluded(id) {
  842. let included = false;
  843. this.includedPlaylists.forEach(playlist => {
  844. if (playlist._id === id) included = true;
  845. });
  846. return included;
  847. },
  848. isExcluded(id) {
  849. let selected = false;
  850. this.excludedPlaylists.forEach(playlist => {
  851. if (playlist._id === id) selected = true;
  852. });
  853. return selected;
  854. },
  855. searchForPlaylists(page) {
  856. if (
  857. this.search.page >= page ||
  858. this.search.searchedQuery !== this.search.query
  859. ) {
  860. this.search.results = [];
  861. this.search.page = 0;
  862. this.search.count = 0;
  863. this.search.resultsLeft = 0;
  864. this.search.pageSize = 0;
  865. }
  866. const { query } = this.search;
  867. const action =
  868. this.station.type === "official"
  869. ? "playlists.searchOfficial"
  870. : "playlists.searchCommunity";
  871. this.search.searchedQuery = this.search.query;
  872. this.socket.dispatch(action, query, page, res => {
  873. const { data } = res;
  874. if (res.status === "success") {
  875. const { count, pageSize, playlists } = data;
  876. this.search.results = [
  877. ...this.search.results,
  878. ...playlists
  879. ];
  880. this.search.page = page;
  881. this.search.count = count;
  882. this.search.resultsLeft =
  883. count - this.search.results.length;
  884. this.search.pageSize = pageSize;
  885. } else if (res.status === "error") {
  886. this.search.results = [];
  887. this.search.page = 0;
  888. this.search.count = 0;
  889. this.search.resultsLeft = 0;
  890. this.search.pageSize = 0;
  891. new Toast(res.message);
  892. }
  893. });
  894. },
  895. async blacklistPlaylist(id) {
  896. if (this.isIncluded(id)) await this.removeIncludedPlaylist(id);
  897. this.socket.dispatch(
  898. "stations.excludePlaylist",
  899. this.station._id,
  900. id,
  901. res => {
  902. new Toast(res.message);
  903. }
  904. );
  905. },
  906. addPartyPlaylistSongToQueue() {
  907. let isInQueue = false;
  908. if (
  909. this.station.type === "community" &&
  910. this.station.partyMode === true
  911. ) {
  912. this.songsList.forEach(queueSong => {
  913. if (queueSong.requestedBy === this.userId) isInQueue = true;
  914. });
  915. if (!isInQueue && this.partyPlaylists) {
  916. const selectedPlaylist =
  917. this.partyPlaylists[
  918. Math.floor(
  919. Math.random() * this.partyPlaylists.length
  920. )
  921. ];
  922. if (
  923. selectedPlaylist._id &&
  924. selectedPlaylist.songs.length > 0
  925. ) {
  926. const selectedSong =
  927. selectedPlaylist.songs[
  928. Math.floor(
  929. Math.random() *
  930. selectedPlaylist.songs.length
  931. )
  932. ];
  933. if (selectedSong.youtubeId) {
  934. this.socket.dispatch(
  935. "stations.addToQueue",
  936. this.station._id,
  937. selectedSong.youtubeId,
  938. data => {
  939. if (data.status !== "success")
  940. new Toast("Error auto queueing song");
  941. }
  942. );
  943. }
  944. }
  945. }
  946. }
  947. },
  948. ...mapActions("station", ["updatePartyPlaylists"]),
  949. ...mapActions("modalVisibility", ["openModal"]),
  950. ...mapActions("user/playlists", ["editPlaylist", "setPlaylists"])
  951. }
  952. };
  953. </script>
  954. <style lang="scss" scoped>
  955. .night-mode {
  956. .tabs-container .tab-selection .button {
  957. background: var(--dark-grey) !important;
  958. color: var(--white) !important;
  959. }
  960. }
  961. .excluded-icon {
  962. color: var(--red);
  963. }
  964. .included-icon {
  965. color: var(--green);
  966. }
  967. .selected-icon {
  968. color: var(--purple);
  969. }
  970. .station-playlists {
  971. .tabs-container {
  972. .tab-selection {
  973. display: flex;
  974. overflow-x: auto;
  975. .button {
  976. border-radius: 0;
  977. border: 0;
  978. text-transform: uppercase;
  979. font-size: 14px;
  980. color: var(--dark-grey-3);
  981. background-color: var(--light-grey-2);
  982. flex-grow: 1;
  983. height: 32px;
  984. &:not(:first-of-type) {
  985. margin-left: 5px;
  986. }
  987. }
  988. .selected {
  989. background-color: var(--primary-color) !important;
  990. color: var(--white) !important;
  991. font-weight: 600;
  992. }
  993. }
  994. .tab {
  995. padding: 15px 0;
  996. border-radius: 0;
  997. .playlist-item:not(:last-of-type),
  998. .item.item-draggable:not(:last-of-type) {
  999. margin-bottom: 10px;
  1000. }
  1001. .load-more-button {
  1002. width: 100%;
  1003. margin-top: 10px;
  1004. }
  1005. }
  1006. }
  1007. }
  1008. .draggable-list-transition-move {
  1009. transition: transform 0.5s;
  1010. }
  1011. .draggable-list-ghost {
  1012. opacity: 0.5;
  1013. filter: brightness(95%);
  1014. }
  1015. </style>