Playlists.vue 24 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031
  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. if (
  908. this.station.type === "community" &&
  909. this.station.partyMode === true &&
  910. this.songsList.length < 50 &&
  911. this.songsList.filter(
  912. queueSong => queueSong.requestedBy === this.userId
  913. ).length < 3 &&
  914. this.partyPlaylists
  915. ) {
  916. const selectedPlaylist =
  917. this.partyPlaylists[
  918. Math.floor(Math.random() * this.partyPlaylists.length)
  919. ];
  920. if (selectedPlaylist._id && selectedPlaylist.songs.length > 0) {
  921. const selectedSong =
  922. selectedPlaylist.songs[
  923. Math.floor(
  924. Math.random() * selectedPlaylist.songs.length
  925. )
  926. ];
  927. if (selectedSong.youtubeId) {
  928. this.socket.dispatch(
  929. "stations.addToQueue",
  930. this.station._id,
  931. selectedSong.youtubeId,
  932. data => {
  933. if (data.status !== "success")
  934. this.addPartyPlaylistSongToQueue();
  935. }
  936. );
  937. }
  938. }
  939. }
  940. },
  941. ...mapActions("station", ["updatePartyPlaylists"]),
  942. ...mapActions("modalVisibility", ["openModal"]),
  943. ...mapActions("user/playlists", ["editPlaylist", "setPlaylists"])
  944. }
  945. };
  946. </script>
  947. <style lang="scss" scoped>
  948. .night-mode {
  949. .tabs-container .tab-selection .button {
  950. background: var(--dark-grey) !important;
  951. color: var(--white) !important;
  952. }
  953. }
  954. .excluded-icon {
  955. color: var(--red);
  956. }
  957. .included-icon {
  958. color: var(--green);
  959. }
  960. .selected-icon {
  961. color: var(--purple);
  962. }
  963. .station-playlists {
  964. .tabs-container {
  965. .tab-selection {
  966. display: flex;
  967. overflow-x: auto;
  968. .button {
  969. border-radius: 0;
  970. border: 0;
  971. text-transform: uppercase;
  972. font-size: 14px;
  973. color: var(--dark-grey-3);
  974. background-color: var(--light-grey-2);
  975. flex-grow: 1;
  976. height: 32px;
  977. &:not(:first-of-type) {
  978. margin-left: 5px;
  979. }
  980. }
  981. .selected {
  982. background-color: var(--primary-color) !important;
  983. color: var(--white) !important;
  984. font-weight: 600;
  985. }
  986. }
  987. .tab {
  988. padding: 15px 0;
  989. border-radius: 0;
  990. .playlist-item:not(:last-of-type),
  991. .item.item-draggable:not(:last-of-type) {
  992. margin-bottom: 10px;
  993. }
  994. .load-more-button {
  995. width: 100%;
  996. margin-top: 10px;
  997. }
  998. }
  999. }
  1000. }
  1001. .draggable-list-transition-move {
  1002. transition: transform 0.5s;
  1003. }
  1004. .draggable-list-ghost {
  1005. opacity: 0.5;
  1006. filter: brightness(95%);
  1007. }
  1008. </style>