AddSongToQueue.vue 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477
  1. <template>
  2. <modal title="Add Song To Queue">
  3. <div slot="body">
  4. <div class="vertical-padding">
  5. <!-- Choosing a song from youtube -->
  6. <h4 class="section-title">Choose a song</h4>
  7. <p class="section-description">
  8. Choose a song by searching or using a link from YouTube.
  9. </p>
  10. <br />
  11. <div class="control is-grouped input-with-button">
  12. <p class="control is-expanded">
  13. <input
  14. class="input"
  15. type="text"
  16. placeholder="Enter your YouTube query here..."
  17. v-model="querySearch"
  18. autofocus
  19. @keyup.enter="submitQuery()"
  20. />
  21. </p>
  22. <p class="control">
  23. <a
  24. class="button is-info"
  25. @click="submitQuery()"
  26. href="#"
  27. ><i class="material-icons icon-with-button"
  28. >search</i
  29. >Search</a
  30. >
  31. </p>
  32. </div>
  33. <!-- Choosing a song from youtube - query results -->
  34. <table
  35. class="table"
  36. style="margin-top: 20px;"
  37. v-if="queryResults.length > 0"
  38. >
  39. <tbody>
  40. <tr
  41. v-for="(result, index) in queryResults"
  42. :key="index"
  43. >
  44. <td class="song-thumbnail">
  45. <div
  46. :style="
  47. `background-image: url('${result.thumbnail}'`
  48. "
  49. ></div>
  50. </td>
  51. <td><strong v-html="result.title"></strong></td>
  52. <td class="song-actions">
  53. <transition
  54. name="song-actions-transition"
  55. mode="out-in"
  56. >
  57. <a
  58. class="button is-success"
  59. v-if="result.isAddedToQueue"
  60. href="#"
  61. key="added-to-queue"
  62. >
  63. <i
  64. class="material-icons icon-with-button"
  65. >done</i
  66. >
  67. Added to queue
  68. </a>
  69. <a
  70. class="button is-dark"
  71. v-else
  72. @click="
  73. addSongToQueue(result.id, index)
  74. "
  75. href="#"
  76. key="add-to-queue"
  77. >
  78. <i
  79. class="material-icons icon-with-button"
  80. >add</i
  81. >
  82. Add to queue
  83. </a>
  84. </transition>
  85. </td>
  86. </tr>
  87. </tbody>
  88. </table>
  89. <!-- Import a playlist from youtube -->
  90. <div v-if="station.type === 'official'">
  91. <hr style="margin: 30px 0;" />
  92. <h4 class="section-title">
  93. Import a playlist
  94. </h4>
  95. <p class="section-description">
  96. Import a playlist by using a link from YouTube.
  97. </p>
  98. <br />
  99. <div class="control is-grouped input-with-button">
  100. <p class="control is-expanded">
  101. <input
  102. class="input"
  103. type="text"
  104. placeholder="YouTube Playlist URL"
  105. v-model="importQuery"
  106. @keyup.enter="importPlaylist()"
  107. />
  108. </p>
  109. <p class="control has-addons">
  110. <span class="select" id="playlist-import-type">
  111. <select
  112. v-model="isImportingOnlyMusicOfPlaylist"
  113. >
  114. <option :value="false">Import all</option>
  115. <option :value="true"
  116. >Import only music</option
  117. >
  118. </select>
  119. </span>
  120. <a
  121. class="button is-info"
  122. @click="importPlaylist()"
  123. href="#"
  124. ><i class="material-icons icon-with-button"
  125. >publish</i
  126. >Import</a
  127. >
  128. </p>
  129. </div>
  130. </div>
  131. <!-- Choose a playlist from your account -->
  132. <div
  133. v-if="
  134. loggedIn &&
  135. station.type === 'community' &&
  136. playlists.length > 0
  137. "
  138. >
  139. <hr style="margin: 30px 0;" />
  140. <aside id="playlist-to-queue-selection">
  141. <h4 class="section-title">Choose a playlist</h4>
  142. <p class="section-description">
  143. Choose one of your playlists to add to the queue.
  144. </p>
  145. <br />
  146. <div id="playlists">
  147. <div
  148. class="playlist"
  149. v-for="(playlist, index) in playlists"
  150. :key="index"
  151. >
  152. <playlist-item :playlist="playlist">
  153. <div slot="actions">
  154. <a
  155. class="button is-danger"
  156. href="#"
  157. @click="
  158. togglePlaylistSelection(
  159. playlist._id
  160. )
  161. "
  162. v-if="
  163. isPlaylistSelected(playlist._id)
  164. "
  165. >
  166. <i
  167. class="material-icons icon-with-button"
  168. >stop</i
  169. >
  170. Stop playing
  171. </a>
  172. <a
  173. class="button is-success"
  174. @click="
  175. togglePlaylistSelection(
  176. playlist._id
  177. )
  178. "
  179. href="#"
  180. v-else
  181. ><i
  182. class="material-icons icon-with-button"
  183. >play_arrow</i
  184. >Play in queue
  185. </a>
  186. </div>
  187. </playlist-item>
  188. </div>
  189. </div>
  190. </aside>
  191. </div>
  192. </div>
  193. </div>
  194. </modal>
  195. </template>
  196. <script>
  197. import { mapState, mapActions } from "vuex";
  198. import Toast from "toasters";
  199. import PlaylistItem from "../../components/ui/PlaylistItem.vue";
  200. import Modal from "../../components/Modal.vue";
  201. import io from "../../io";
  202. export default {
  203. components: { Modal, PlaylistItem },
  204. data() {
  205. return {
  206. querySearch: "",
  207. queryResults: [],
  208. playlists: [],
  209. importQuery: "",
  210. isImportingOnlyMusicOfPlaylist: false
  211. };
  212. },
  213. computed: mapState({
  214. loggedIn: state => state.user.auth.loggedIn,
  215. station: state => state.station.station,
  216. privatePlaylistQueueSelected: state =>
  217. state.station.privatePlaylistQueueSelected
  218. }),
  219. mounted() {
  220. io.getSocket(socket => {
  221. this.socket = socket;
  222. this.socket.emit("playlists.indexForUser", true, res => {
  223. if (res.status === "success") this.playlists = res.data;
  224. });
  225. });
  226. },
  227. methods: {
  228. isPlaylistSelected(playlistId) {
  229. return this.privatePlaylistQueueSelected === playlistId;
  230. },
  231. togglePlaylistSelection(playlistId) {
  232. if (this.station.type === "community") {
  233. if (this.isPlaylistSelected(playlistId)) {
  234. this.updatePrivatePlaylistQueueSelected(null);
  235. } else {
  236. this.updatePrivatePlaylistQueueSelected(playlistId);
  237. this.$parent.addFirstPrivatePlaylistSongToQueue();
  238. console.log(this.isPlaylistSelected(playlistId));
  239. }
  240. }
  241. },
  242. addSongToQueue(songId, index) {
  243. if (this.station.type === "community") {
  244. this.socket.emit(
  245. "stations.addToQueue",
  246. this.station._id,
  247. songId,
  248. data => {
  249. if (data.status !== "success")
  250. new Toast({
  251. content: `Error: ${data.message}`,
  252. timeout: 8000
  253. });
  254. else {
  255. this.queryResults[index].isAddedToQueue = true;
  256. new Toast({
  257. content: `${data.message}`,
  258. timeout: 4000
  259. });
  260. }
  261. }
  262. );
  263. } else {
  264. this.socket.emit("queueSongs.add", songId, data => {
  265. if (data.status !== "success")
  266. new Toast({
  267. content: `Error: ${data.message}`,
  268. timeout: 8000
  269. });
  270. else {
  271. this.queryResults[index].isAddedToQueue = true;
  272. new Toast({
  273. content: `${data.message}`,
  274. timeout: 4000
  275. });
  276. }
  277. });
  278. }
  279. },
  280. importPlaylist() {
  281. let isImportingPlaylist = true;
  282. // import query is blank
  283. if (!this.importQuery)
  284. return new Toast({
  285. content: "Please enter a YouTube playlist URL.",
  286. timeout: 4000
  287. });
  288. // don't give starting import message instantly in case of instant error
  289. setTimeout(() => {
  290. if (isImportingPlaylist) {
  291. new Toast({
  292. content:
  293. "Starting to import your playlist. This can take some time to do.",
  294. timeout: 4000
  295. });
  296. }
  297. }, 750);
  298. return this.socket.emit(
  299. "queueSongs.addSetToQueue",
  300. this.importQuery,
  301. this.isImportingOnlyMusicOfPlaylist,
  302. res => {
  303. isImportingPlaylist = false;
  304. return new Toast({ content: res.message, timeout: 20000 });
  305. }
  306. );
  307. },
  308. submitQuery() {
  309. let query = this.querySearch;
  310. if (!this.querySearch)
  311. return new Toast({
  312. content: "Please input a search query or a YouTube link",
  313. timeout: 4000
  314. });
  315. if (query.indexOf("&index=") !== -1) {
  316. query = query.split("&index=");
  317. query.pop();
  318. query = query.join("");
  319. }
  320. if (query.indexOf("&list=") !== -1) {
  321. query = query.split("&list=");
  322. query.pop();
  323. query = query.join("");
  324. }
  325. return this.socket.emit("apis.searchYoutube", query, res => {
  326. if (res.status === "failure")
  327. return new Toast({
  328. content: "Error searching on YouTube",
  329. timeout: 4000
  330. });
  331. const { data } = res;
  332. this.queryResults = [];
  333. console.log(res.data);
  334. for (let i = 0; i < data.items.length; i += 1) {
  335. this.queryResults.push({
  336. id: data.items[i].id.videoId,
  337. url: `https://www.youtube.com/watch?v=${this.id}`,
  338. title: data.items[i].snippet.title,
  339. thumbnail: data.items[i].snippet.thumbnails.default.url,
  340. isAddedToQueue: false
  341. });
  342. }
  343. return this.queryResults;
  344. });
  345. },
  346. ...mapActions("station", ["updatePrivatePlaylistQueueSelected"]),
  347. ...mapActions("user/playlists", ["editPlaylist"])
  348. }
  349. };
  350. </script>
  351. <style lang="scss" scoped>
  352. @import "../../styles/global.scss";
  353. .night-mode {
  354. tr {
  355. background-color: $night-mode-bg-secondary;
  356. }
  357. }
  358. tr td {
  359. vertical-align: middle;
  360. }
  361. .song-actions {
  362. .button {
  363. height: 36px;
  364. width: 140px;
  365. }
  366. }
  367. .song-thumbnail div {
  368. width: 96px;
  369. height: 54px;
  370. background-position: center;
  371. background-repeat: no-repeat;
  372. }
  373. .table {
  374. margin-bottom: 0;
  375. }
  376. .night-mode {
  377. div {
  378. color: #4d4d4d;
  379. }
  380. }
  381. #playlist-to-queue-selection {
  382. margin-top: 0;
  383. #playlists {
  384. font-size: 18px;
  385. .radio {
  386. display: flex;
  387. flex-direction: row;
  388. align-items: center;
  389. input {
  390. transform: scale(1.25);
  391. }
  392. }
  393. }
  394. }
  395. #playlist-import-type {
  396. &:hover {
  397. z-index: initial;
  398. }
  399. select {
  400. border-radius: 0;
  401. }
  402. }
  403. .vertical-padding {
  404. padding: 20px;
  405. }
  406. #playlists {
  407. .playlist {
  408. .button {
  409. width: 146px;
  410. }
  411. }
  412. }
  413. .song-actions-transition-enter-active {
  414. transition: all 0.2s ease;
  415. }
  416. .song-actions-transition-leave-active {
  417. transition: all 0.2s cubic-bezier(1, 0.5, 0.8, 1);
  418. }
  419. .song-actions-transition-enter {
  420. transform: translateX(-20px);
  421. opacity: 0;
  422. }
  423. .song-actions-transition-leave-to {
  424. transform: translateX(20px);
  425. opacity: 0;
  426. }
  427. </style>