index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727
  1. <template>
  2. <modal
  3. :title="
  4. userId === playlist.createdBy ? 'Edit Playlist' : 'View Playlist'
  5. "
  6. class="edit-playlist-modal"
  7. >
  8. <template #body>
  9. <div
  10. :class="{
  11. 'view-only': !isEditable(),
  12. 'edit-playlist-modal-inner-container': true
  13. }"
  14. >
  15. <div id="first-column">
  16. <div id="playlist-info-section" class="section">
  17. <h3>{{ playlist.displayName }}</h3>
  18. <h5>Song Count: {{ playlist.songs.length }}</h5>
  19. <h5>Duration: {{ totalLength() }}</h5>
  20. </div>
  21. <div id="tabs-container">
  22. <div id="tab-selection">
  23. <button
  24. class="button is-default"
  25. :class="{ selected: tab === 'settings' }"
  26. ref="settings-tab"
  27. @click="showTab('settings')"
  28. v-if="
  29. userId === playlist.createdBy ||
  30. isEditable() ||
  31. (playlist.type === 'genre' && isAdmin())
  32. "
  33. >
  34. Settings
  35. </button>
  36. <button
  37. class="button is-default"
  38. :class="{ selected: tab === 'youtube' }"
  39. ref="youtube-tab"
  40. @click="showTab('youtube')"
  41. v-if="isEditable()"
  42. >
  43. YouTube
  44. </button>
  45. </div>
  46. <settings
  47. class="tab"
  48. v-show="tab === 'settings'"
  49. v-if="
  50. userId === playlist.createdBy ||
  51. isEditable() ||
  52. (playlist.type === 'genre' && isAdmin())
  53. "
  54. />
  55. <youtube
  56. class="tab"
  57. v-show="tab === 'youtube'"
  58. v-if="isEditable()"
  59. />
  60. </div>
  61. <!--
  62. <div
  63. id="import-from-youtube-section"
  64. -->
  65. </div>
  66. <div id="second-column">
  67. <div id="rearrange-songs-section" class="section">
  68. <div v-if="isEditable()">
  69. <h4 class="section-title">Rearrange Songs</h4>
  70. <p class="section-description">
  71. Drag and drop songs to change their order
  72. </p>
  73. <hr class="section-horizontal-rule" />
  74. </div>
  75. <aside class="menu">
  76. <draggable
  77. tag="transition-group"
  78. :component-data="{
  79. name: !drag
  80. ? 'draggable-list-transition'
  81. : null
  82. }"
  83. v-if="playlistSongs.length > 0"
  84. v-model="playlistSongs"
  85. item-key="_id"
  86. v-bind="dragOptions"
  87. @start="drag = true"
  88. @end="drag = false"
  89. @change="repositionSong"
  90. >
  91. <template #item="{element, index}">
  92. <div class="menu-list scrollable-list">
  93. <song-item
  94. :song="element"
  95. :class="{
  96. 'item-draggable': isEditable()
  97. }"
  98. >
  99. <template #actions>
  100. <i
  101. class="material-icons add-to-queue-icon"
  102. v-if="
  103. station.partyMode &&
  104. !station.locked
  105. "
  106. @click="
  107. addSongToQueue(
  108. element.youtubeId
  109. )
  110. "
  111. content="Add Song to Queue"
  112. v-tippy
  113. >queue</i
  114. >
  115. <confirm
  116. v-if="
  117. userId ===
  118. playlist.createdBy ||
  119. isEditable()
  120. "
  121. placement="left"
  122. @confirm="
  123. removeSongFromPlaylist(
  124. element.youtubeId
  125. )
  126. "
  127. >
  128. <i
  129. class="material-icons delete-icon"
  130. content="Remove Song from Playlist"
  131. v-tippy
  132. >delete_forever</i
  133. >
  134. </confirm>
  135. <i
  136. class="material-icons"
  137. v-if="
  138. isEditable() &&
  139. index > 0
  140. "
  141. @click="
  142. moveSongToTop(
  143. element,
  144. index
  145. )
  146. "
  147. content="Move to top of Playlist"
  148. v-tippy
  149. >vertical_align_top</i
  150. >
  151. <i
  152. v-if="
  153. isEditable() &&
  154. playlistSongs.length -
  155. 1 !==
  156. index
  157. "
  158. @click="
  159. moveSongToBottom(
  160. element,
  161. index
  162. )
  163. "
  164. class="material-icons"
  165. content="Move to bottom of Playlist"
  166. v-tippy
  167. >vertical_align_bottom</i
  168. >
  169. </template>
  170. </song-item>
  171. </div>
  172. </template>
  173. </draggable>
  174. <p v-else class="nothing-here-text">
  175. This playlist doesn't have any songs.
  176. </p>
  177. </aside>
  178. </div>
  179. </div>
  180. <!--
  181. <button
  182. class="button is-info"
  183. @click="shuffle()"
  184. v-if="playlist.isUserModifiable"
  185. >
  186. Shuffle
  187. </button>
  188. <h5>Edit playlist details:</h5>
  189. -->
  190. </div>
  191. </template>
  192. <template #footer>
  193. <a
  194. class="button is-default"
  195. v-if="
  196. userId === playlist.createdBy ||
  197. isEditable() ||
  198. playlist.privacy === 'public'
  199. "
  200. @click="downloadPlaylist()"
  201. href="#"
  202. >
  203. Download Playlist
  204. </a>
  205. <div class="right">
  206. <confirm
  207. v-if="playlist.type === 'station'"
  208. @confirm="clearAndRefillStationPlaylist()"
  209. >
  210. <a class="button is-danger">
  211. Clear and refill station playlist
  212. </a>
  213. </confirm>
  214. <confirm
  215. v-if="playlist.type === 'genre'"
  216. @confirm="clearAndRefillGenrePlaylist()"
  217. >
  218. <a class="button is-danger">
  219. Clear and refill genre playlist
  220. </a>
  221. </confirm>
  222. <confirm v-if="isEditable()" @confirm="removePlaylist()">
  223. <a class="button is-danger"> Remove Playlist </a>
  224. </confirm>
  225. </div>
  226. </template>
  227. </modal>
  228. </template>
  229. <script>
  230. import { mapState, mapGetters, mapActions } from "vuex";
  231. import draggable from "vuedraggable";
  232. import Toast from "toasters";
  233. import Confirm from "@/components/Confirm.vue";
  234. import Modal from "../../Modal.vue";
  235. import SongItem from "../../SongItem.vue";
  236. import Settings from "./Tabs/Settings.vue";
  237. import Youtube from "./Tabs/Youtube.vue";
  238. import utils from "../../../../js/utils";
  239. export default {
  240. components: { Modal, draggable, Confirm, SongItem, Settings, Youtube },
  241. data() {
  242. return {
  243. utils,
  244. drag: false,
  245. apiDomain: ""
  246. };
  247. },
  248. computed: {
  249. ...mapState("station", {
  250. station: state => state.station
  251. }),
  252. ...mapState("user/playlists", {
  253. editing: state => state.editing
  254. }),
  255. ...mapState("modals/editPlaylist", {
  256. tab: state => state.tab,
  257. playlist: state => state.playlist
  258. }),
  259. playlistSongs: {
  260. get() {
  261. return this.$store.state.modals.editPlaylist.playlist.songs;
  262. },
  263. set(value) {
  264. this.$store.commit(
  265. "modals/editPlaylist/updatePlaylistSongs",
  266. value
  267. );
  268. }
  269. },
  270. ...mapState({
  271. userId: state => state.user.auth.userId,
  272. userRole: state => state.user.auth.role
  273. }),
  274. dragOptions() {
  275. return {
  276. animation: 200,
  277. group: "songs",
  278. disabled: !this.isEditable(),
  279. ghostClass: "draggable-list-ghost"
  280. };
  281. },
  282. ...mapGetters({
  283. socket: "websockets/getSocket"
  284. })
  285. },
  286. mounted() {
  287. this.socket.dispatch("playlists.getPlaylist", this.editing, res => {
  288. if (res.status === "success") {
  289. // this.playlist = res.data.playlist;
  290. // this.playlist.songs.sort((a, b) => a.position - b.position);
  291. this.setPlaylist(res.data.playlist);
  292. } else new Toast(res.message);
  293. });
  294. this.socket.on(
  295. "event:playlist.song.added",
  296. res => {
  297. if (this.playlist._id === res.data.playlistId)
  298. this.addSong(res.data.song);
  299. },
  300. { modal: "editPlaylist" }
  301. );
  302. this.socket.on(
  303. "event:playlist.song.removed",
  304. res => {
  305. if (this.playlist._id === res.data.playlistId) {
  306. // remove song from array of playlists
  307. this.removeSong(res.data.youtubeId);
  308. // // if this song is in search results, mark it available to add to the playlist again
  309. // this.search.songs.results.forEach((searchItem, index) => {
  310. // if (res.data.youtubeId === searchItem.id) {
  311. // this.search.songs.results[
  312. // index
  313. // ].isAddedToQueue = false;
  314. // }
  315. // });
  316. }
  317. },
  318. { modal: "editPlaylist" }
  319. );
  320. this.socket.on(
  321. "event:playlist.displayName.updated",
  322. res => {
  323. if (this.playlist._id === res.data.playlistId) {
  324. const playlist = {
  325. displayName: res.data.displayName,
  326. ...this.playlist
  327. };
  328. this.setPlaylist(playlist);
  329. }
  330. },
  331. { modal: "editPlaylist" }
  332. );
  333. this.socket.on(
  334. "event:playlist.song.repositioned",
  335. res => {
  336. if (this.playlist._id === res.data.playlistId) {
  337. const { song, playlistId } = res.data;
  338. if (this.playlist._id === playlistId) {
  339. this.repositionedSong(song);
  340. }
  341. }
  342. },
  343. { modal: "editPlaylist" }
  344. );
  345. },
  346. methods: {
  347. isEditable() {
  348. return (
  349. this.playlist.isUserModifiable &&
  350. (this.userId === this.playlist.createdBy ||
  351. this.userRole === "admin")
  352. );
  353. },
  354. isAdmin() {
  355. return this.userRole === "admin";
  356. },
  357. repositionSong({ moved }) {
  358. if (!moved) return; // we only need to update when song is moved
  359. this.socket.dispatch(
  360. "playlists.repositionSong",
  361. this.playlist._id,
  362. {
  363. ...moved.element,
  364. oldIndex: moved.oldIndex,
  365. newIndex: moved.newIndex
  366. },
  367. res => {
  368. if (res.status !== "success")
  369. this.repositionedSong({
  370. ...moved.element,
  371. newIndex: moved.oldIndex,
  372. oldIndex: moved.newIndex
  373. });
  374. }
  375. );
  376. },
  377. moveSongToTop(song, index) {
  378. this.repositionSong({
  379. moved: {
  380. element: song,
  381. oldIndex: index,
  382. newIndex: 0
  383. }
  384. });
  385. },
  386. moveSongToBottom(song, index) {
  387. this.repositionSong({
  388. moved: {
  389. element: song,
  390. oldIndex: index,
  391. newIndex: this.playlistSongs.length
  392. }
  393. });
  394. },
  395. totalLength() {
  396. let length = 0;
  397. this.playlist.songs.forEach(song => {
  398. length += song.duration;
  399. });
  400. return this.utils.formatTimeLong(length);
  401. },
  402. shuffle() {
  403. this.socket.dispatch(
  404. "playlists.shuffle",
  405. this.playlist._id,
  406. res => {
  407. new Toast(res.message);
  408. if (res.status === "success") {
  409. this.updatePlaylistSongs(
  410. res.data.playlist.songs.sort(
  411. (a, b) => a.position - b.position
  412. )
  413. );
  414. }
  415. }
  416. );
  417. },
  418. removeSongFromPlaylist(id) {
  419. if (this.playlist.displayName === "Liked Songs")
  420. return this.socket.dispatch("songs.unlike", id, res => {
  421. new Toast(res.message);
  422. });
  423. if (this.playlist.displayName === "Disliked Songs")
  424. return this.socket.dispatch("songs.undislike", id, res => {
  425. new Toast(res.message);
  426. });
  427. return this.socket.dispatch(
  428. "playlists.removeSongFromPlaylist",
  429. id,
  430. this.playlist._id,
  431. res => {
  432. new Toast(res.message);
  433. }
  434. );
  435. },
  436. removePlaylist() {
  437. this.socket.dispatch("playlists.remove", this.playlist._id, res => {
  438. new Toast(res.message);
  439. if (res.status === "success") this.closeModal("editPlaylist");
  440. });
  441. },
  442. async downloadPlaylist() {
  443. if (this.apiDomain === "")
  444. this.apiDomain = await lofig.get("apiDomain");
  445. fetch(
  446. `${this.apiDomain}/export/privatePlaylist/${this.playlist._id}`,
  447. { credentials: "include" }
  448. )
  449. .then(res => res.blob())
  450. .then(blob => {
  451. const url = window.URL.createObjectURL(blob);
  452. const a = document.createElement("a");
  453. a.style.display = "none";
  454. a.href = url;
  455. a.download = `musare-privateplaylist-${
  456. this.playlist._id
  457. }-${new Date().toISOString()}.json`;
  458. document.body.appendChild(a);
  459. a.click();
  460. window.URL.revokeObjectURL(url);
  461. new Toast("Successfully downloaded playlist.");
  462. })
  463. .catch(
  464. () => new Toast("Failed to export and download playlist.")
  465. );
  466. },
  467. addSongToQueue(youtubeId) {
  468. this.socket.dispatch(
  469. "stations.addToQueue",
  470. this.station._id,
  471. youtubeId,
  472. data => {
  473. if (data.status !== "success")
  474. new Toast({
  475. content: `Error: ${data.message}`,
  476. timeout: 8000
  477. });
  478. else new Toast({ content: data.message, timeout: 4000 });
  479. }
  480. );
  481. },
  482. clearAndRefillStationPlaylist() {
  483. this.socket.dispatch(
  484. "playlists.clearAndRefillStationPlaylist",
  485. this.playlist._id,
  486. data => {
  487. console.log(data.message);
  488. if (data.status !== "success")
  489. new Toast({
  490. content: `Error: ${data.message}`,
  491. timeout: 8000
  492. });
  493. else new Toast({ content: data.message, timeout: 4000 });
  494. }
  495. );
  496. },
  497. clearAndRefillGenrePlaylist() {
  498. this.socket.dispatch(
  499. "playlists.clearAndRefillGenrePlaylist",
  500. this.playlist._id,
  501. data => {
  502. if (data.status !== "success")
  503. new Toast({
  504. content: `Error: ${data.message}`,
  505. timeout: 8000
  506. });
  507. else new Toast({ content: data.message, timeout: 4000 });
  508. }
  509. );
  510. },
  511. ...mapActions({
  512. showTab(dispatch, payload) {
  513. this.$refs[`${payload}-tab`].scrollIntoView();
  514. return dispatch("modals/editPlaylist/showTab", payload);
  515. }
  516. }),
  517. ...mapActions("modals/editPlaylist", [
  518. "setPlaylist",
  519. "addSong",
  520. "removeSong",
  521. "repositionedSong"
  522. ]),
  523. ...mapActions("modalVisibility", ["openModal", "closeModal"])
  524. }
  525. };
  526. </script>
  527. <style lang="scss">
  528. .edit-playlist-modal {
  529. .modal-card {
  530. width: 1300px;
  531. .modal-card-body {
  532. padding: 16px;
  533. }
  534. }
  535. }
  536. </style>
  537. <style lang="scss" scoped>
  538. .night-mode {
  539. .label,
  540. p,
  541. strong {
  542. color: var(--light-grey-2);
  543. }
  544. }
  545. .menu-list li {
  546. display: flex;
  547. justify-content: space-between;
  548. &:not(:last-of-type) {
  549. margin-bottom: 10px;
  550. }
  551. a {
  552. display: flex;
  553. }
  554. }
  555. .controls {
  556. display: flex;
  557. a {
  558. display: flex;
  559. align-items: center;
  560. }
  561. }
  562. @media screen and (max-width: 1300px) {
  563. .edit-playlist-modal .edit-playlist-modal-inner-container {
  564. height: auto !important;
  565. /deep/ .section {
  566. max-width: 100% !important;
  567. }
  568. }
  569. }
  570. #tabs-container {
  571. // padding: 16px;
  572. #tab-selection {
  573. display: flex;
  574. overflow-x: auto;
  575. margin: 24px 10px 0 10px;
  576. .button {
  577. border-radius: 5px 5px 0 0;
  578. border: 0;
  579. text-transform: uppercase;
  580. font-size: 14px;
  581. color: var(--dark-grey-3);
  582. background-color: var(--light-grey-2);
  583. flex-grow: 1;
  584. height: 32px;
  585. &:not(:first-of-type) {
  586. margin-left: 5px;
  587. }
  588. }
  589. .selected {
  590. background-color: var(--primary-color) !important;
  591. color: var(--white) !important;
  592. font-weight: 600;
  593. }
  594. }
  595. .tab {
  596. border: 1px solid var(--light-grey-3);
  597. // padding: 15px;
  598. border-radius: 0 0 5px 5px;
  599. }
  600. }
  601. .edit-playlist-modal {
  602. .edit-playlist-modal-inner-container {
  603. display: flex;
  604. flex-wrap: wrap;
  605. height: 100%;
  606. &.view-only {
  607. height: auto !important;
  608. #first-column {
  609. flex-basis: 100%;
  610. }
  611. /deep/ .section {
  612. max-width: 100% !important;
  613. }
  614. }
  615. }
  616. .nothing-here-text {
  617. display: flex;
  618. align-items: center;
  619. justify-content: center;
  620. }
  621. /deep/ .section {
  622. padding: 15px !important;
  623. margin: 0 10px;
  624. max-width: 600px;
  625. display: flex;
  626. flex-direction: column;
  627. flex-grow: 1;
  628. }
  629. .label {
  630. font-size: 1rem;
  631. font-weight: normal;
  632. }
  633. .input-with-button .button {
  634. width: 150px;
  635. }
  636. #first-column {
  637. max-width: 100%;
  638. height: 100%;
  639. overflow-y: auto;
  640. flex-grow: 1;
  641. /deep/ .section {
  642. width: auto;
  643. }
  644. #playlist-info-section {
  645. border: 1px solid var(--light-grey-3);
  646. border-radius: 3px;
  647. padding: 15px !important;
  648. h3 {
  649. font-weight: 600;
  650. font-size: 30px;
  651. }
  652. h5 {
  653. font-size: 18px;
  654. }
  655. h3,
  656. h5 {
  657. margin: 0;
  658. }
  659. }
  660. }
  661. #second-column {
  662. max-width: 100%;
  663. height: 100%;
  664. overflow-y: auto;
  665. flex-grow: 1;
  666. }
  667. }
  668. </style>