EditSong.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763
  1. <template>
  2. <div>
  3. <modal title="Edit Song">
  4. <div slot="body">
  5. <h5 class="has-text-centered">Video Preview</h5>
  6. <div class="video-container">
  7. <div id="player"></div>
  8. <div class="controls">
  9. <form action="#">
  10. <p style="margin-top: 0; position: relative;">
  11. <input
  12. type="range"
  13. id="volumeSlider"
  14. min="0"
  15. max="100"
  16. class="active"
  17. v-on:change="changeVolume()"
  18. v-on:input="changeVolume()"
  19. />
  20. </p>
  21. </form>
  22. <p class="control has-addons">
  23. <button
  24. class="button"
  25. v-on:click="settings('pause')"
  26. v-if="!video.paused"
  27. >
  28. <i class="material-icons">pause</i>
  29. </button>
  30. <button
  31. class="button"
  32. v-on:click="settings('play')"
  33. v-if="video.paused"
  34. >
  35. <i class="material-icons">play_arrow</i>
  36. </button>
  37. <button
  38. class="button"
  39. v-on:click="settings('stop')"
  40. >
  41. <i class="material-icons">stop</i>
  42. </button>
  43. <button
  44. class="button"
  45. v-on:click="settings('skipToLast10Secs')"
  46. >
  47. <i class="material-icons">fast_forward</i>
  48. </button>
  49. </p>
  50. <p>
  51. YouTube:
  52. <span>{{ youtubeVideoCurrentTime }}</span> /
  53. <span>{{ youtubeVideoDuration }}</span>
  54. {{ youtubeVideoNote }}
  55. </p>
  56. </div>
  57. </div>
  58. <h5 class="has-text-centered">Thumbnail Preview</h5>
  59. <img
  60. class="thumbnail-preview"
  61. :src="editing.song.thumbnail"
  62. onerror="this.src='/assets/notes-transparent.png'"
  63. />
  64. <div class="control is-horizontal">
  65. <div class="control-label">
  66. <label class="label">Thumbnail URL</label>
  67. </div>
  68. <div class="control">
  69. <input
  70. class="input"
  71. type="text"
  72. v-model="editing.song.thumbnail"
  73. />
  74. </div>
  75. </div>
  76. <h5 class="has-text-centered">Edit Information</h5>
  77. <p class="control">
  78. <label class="checkbox">
  79. <input
  80. type="checkbox"
  81. v-model="editing.song.explicit"
  82. />
  83. Explicit
  84. </label>
  85. </p>
  86. <label class="label">Song ID & Title</label>
  87. <div class="control is-horizontal">
  88. <div class="control is-grouped">
  89. <p class="control is-expanded">
  90. <input
  91. class="input"
  92. type="text"
  93. v-model="editing.song.songId"
  94. />
  95. </p>
  96. <p class="control is-expanded">
  97. <input
  98. class="input"
  99. type="text"
  100. v-model="editing.song.title"
  101. autofocus
  102. />
  103. </p>
  104. </div>
  105. </div>
  106. <label class="label">Artists & Genres</label>
  107. <div class="control is-horizontal">
  108. <div class="control is-grouped artist-genres">
  109. <div>
  110. <p class="control has-addons">
  111. <input
  112. class="input"
  113. id="new-artist"
  114. type="text"
  115. placeholder="Artist"
  116. />
  117. <button
  118. class="button is-info"
  119. v-on:click="addTag('artists')"
  120. >
  121. Add Artist
  122. </button>
  123. </p>
  124. <span
  125. class="tag is-info"
  126. v-for="(artist, index) in editing.song.artists"
  127. :key="index"
  128. >
  129. {{ artist }}
  130. <button
  131. class="delete is-info"
  132. v-on:click="removeTag('artists', index)"
  133. ></button>
  134. </span>
  135. </div>
  136. <div>
  137. <p class="control has-addons">
  138. <input
  139. class="input"
  140. id="new-genre"
  141. type="text"
  142. placeholder="Genre"
  143. />
  144. <button
  145. class="button is-info"
  146. v-on:click="addTag('genres')"
  147. >
  148. Add Genre
  149. </button>
  150. </p>
  151. <span
  152. class="tag is-info"
  153. v-for="(genre, index) in editing.song.genres"
  154. :key="index"
  155. >
  156. {{ genre }}
  157. <button
  158. class="delete is-info"
  159. v-on:click="removeTag('genres', index)"
  160. ></button>
  161. </span>
  162. </div>
  163. </div>
  164. </div>
  165. <label class="label">Song Duration</label>
  166. <p class="control">
  167. <input
  168. class="input"
  169. type="text"
  170. v-model="editing.song.duration"
  171. />
  172. </p>
  173. <label class="label">Skip Duration</label>
  174. <p class="control">
  175. <input
  176. class="input"
  177. type="text"
  178. v-model="editing.song.skipDuration"
  179. />
  180. </p>
  181. <article class="message" v-if="editing.type === 'songs'">
  182. <div class="message-body">
  183. <span class="reports-length">
  184. {{ reports.length }}
  185. <span
  186. v-if="reports.length > 1 || reports.length <= 0"
  187. >&nbsp;Reports</span
  188. >
  189. <span v-else>&nbsp;Report</span>
  190. </span>
  191. <div v-for="(report, index) in reports" :key="index">
  192. <a
  193. :href="`/admin/reports?id=${report}`"
  194. class="report-link"
  195. >Report - {{ report }}</a
  196. >
  197. </div>
  198. </div>
  199. </article>
  200. <hr />
  201. <h5 class="has-text-centered">Spotify Information</h5>
  202. <label class="label">Song title</label>
  203. <p class="control">
  204. <input class="input" type="text" v-model="spotify.title" />
  205. </p>
  206. <label class="label">Song artist (1 artist full name)</label>
  207. <p class="control">
  208. <input class="input" type="text" v-model="spotify.artist" />
  209. </p>
  210. <button
  211. class="button is-success"
  212. v-on:click="getSpotifySongs()"
  213. >
  214. Get Spotify songs
  215. </button>
  216. <hr />
  217. <article
  218. class="media"
  219. v-for="(song, index) in spotify.songs"
  220. :key="index"
  221. >
  222. <figure class="media-left">
  223. <p class="image is-64x64">
  224. <img
  225. :src="song.thumbnail"
  226. onerror="this.src='/assets/notes-transparent.png'"
  227. />
  228. </p>
  229. </figure>
  230. <div class="media-content">
  231. <div class="content">
  232. <p>
  233. <strong>{{ song.title }}</strong>
  234. <br />
  235. <small>Artists: {{ song.artists }}</small
  236. >, <small>Duration: {{ song.duration }}</small
  237. >,
  238. <small>Explicit: {{ song.explicit }}</small>
  239. <br />
  240. <small>Thumbnail: {{ song.thumbnail }}</small>
  241. </p>
  242. </div>
  243. </div>
  244. </article>
  245. </div>
  246. <div slot="footer">
  247. <button
  248. class="button is-success"
  249. v-on:click="save(editing.song, false)"
  250. >
  251. <i class="material-icons save-changes">done</i>
  252. <span>&nbsp;Save</span>
  253. </button>
  254. <button
  255. class="button is-success"
  256. v-on:click="save(editing.song, true)"
  257. >
  258. <i class="material-icons save-changes">done</i>
  259. <span>&nbsp;Save and close</span>
  260. </button>
  261. <button
  262. class="button is-danger"
  263. v-on:click="
  264. toggleModal({ sector: 'admin', modal: 'editSong' })
  265. "
  266. >
  267. <span>&nbsp;Close</span>
  268. </button>
  269. </div>
  270. </modal>
  271. </div>
  272. </template>
  273. <script>
  274. import { mapState, mapActions } from "vuex";
  275. import io from "../../io";
  276. import validation from "../../validation";
  277. import { Toast } from "vue-roaster";
  278. import Modal from "./Modal.vue";
  279. export default {
  280. components: { Modal },
  281. data() {
  282. return {
  283. reports: 0,
  284. spotify: {
  285. title: "",
  286. artist: "",
  287. songs: []
  288. },
  289. youtubeVideoDuration: 0.0,
  290. youtubeVideoCurrentTime: 0.0,
  291. youtubeVideoNote: ""
  292. };
  293. },
  294. computed: {
  295. ...mapState("admin/songs", {
  296. video: state => state.video,
  297. editing: state => state.editing
  298. }),
  299. ...mapState("modals", {
  300. modals: state => state.modals.admin
  301. })
  302. },
  303. methods: {
  304. save: function(song, close) {
  305. let _this = this;
  306. if (!song.title)
  307. return Toast.methods.addToast(
  308. "Please fill in all fields",
  309. 8000
  310. );
  311. if (!song.thumbnail)
  312. return Toast.methods.addToast(
  313. "Please fill in all fields",
  314. 8000
  315. );
  316. // Duration
  317. if (
  318. Number(song.skipDuration) + Number(song.duration) >
  319. this.youtubeVideoDuration
  320. ) {
  321. return Toast.methods.addToast(
  322. "Duration can't be higher than the length of the video",
  323. 8000
  324. );
  325. }
  326. // Title
  327. if (!validation.isLength(song.title, 1, 64))
  328. return Toast.methods.addToast(
  329. "Title must have between 1 and 64 characters.",
  330. 8000
  331. );
  332. if (!validation.regex.ascii.test(song.title))
  333. return Toast.methods.addToast(
  334. "Invalid title format. Only ascii characters are allowed.",
  335. 8000
  336. );
  337. // Artists
  338. if (song.artists.length < 1 || song.artists.length > 10)
  339. return Toast.methods.addToast(
  340. "Invalid artists. You must have at least 1 artist and a maximum of 10 artists.",
  341. 8000
  342. );
  343. let error;
  344. song.artists.forEach(artist => {
  345. if (!validation.isLength(artist, 1, 32))
  346. return (error =
  347. "Artist must have between 1 and 32 characters.");
  348. if (!validation.regex.ascii.test(artist))
  349. return (error =
  350. "Invalid artist format. Only ascii characters are allowed.");
  351. if (artist === "NONE")
  352. return (error =
  353. 'Invalid artist format. Artists are not allowed to be named "NONE".');
  354. });
  355. if (error) return Toast.methods.addToast(error, 8000);
  356. // Genres
  357. error = undefined;
  358. song.genres.forEach(genre => {
  359. if (!validation.isLength(genre, 1, 16))
  360. return (error =
  361. "Genre must have between 1 and 16 characters.");
  362. if (!validation.regex.az09_.test(genre))
  363. return (error =
  364. "Invalid genre format. Only ascii characters are allowed.");
  365. });
  366. if (error) return Toast.methods.addToast(error, 8000);
  367. // Thumbnail
  368. if (!validation.isLength(song.thumbnail, 8, 256))
  369. return Toast.methods.addToast(
  370. "Thumbnail must have between 8 and 256 characters.",
  371. 8000
  372. );
  373. if (song.thumbnail.indexOf("https://") !== 0)
  374. return Toast.methods.addToast(
  375. 'Thumbnail must start with "https://".',
  376. 8000
  377. );
  378. this.socket.emit(
  379. `${_this.editing.type}.update`,
  380. song._id,
  381. song,
  382. res => {
  383. Toast.methods.addToast(res.message, 4000);
  384. if (res.status === "success") {
  385. _this.$parent.songs.forEach(lSong => {
  386. if (song._id === lSong._id) {
  387. for (let n in song) {
  388. lSong[n] = song[n];
  389. }
  390. }
  391. });
  392. }
  393. if (close) _this.closeCurrentModal();
  394. }
  395. );
  396. },
  397. settings: function(type) {
  398. let _this = this;
  399. switch (type) {
  400. case "stop":
  401. _this.stopVideo();
  402. _this.pauseVideo(true);
  403. break;
  404. case "pause":
  405. _this.pauseVideo(true);
  406. break;
  407. case "play":
  408. _this.pauseVideo(false);
  409. break;
  410. case "skipToLast10Secs":
  411. _this.video.player.seekTo(
  412. _this.editing.song.duration -
  413. 10 +
  414. _this.editing.song.skipDuration
  415. );
  416. break;
  417. }
  418. },
  419. changeVolume: function() {
  420. let local = this;
  421. let volume = document.getElementById("volumeSlider").value;
  422. localStorage.setItem("volume", volume);
  423. local.video.player.setVolume(volume);
  424. if (volume > 0) local.video.player.unMute();
  425. },
  426. addTag: function(type) {
  427. if (type == "genres") {
  428. let genre = document
  429. .getElementById("new-genre")
  430. .value.toLowerCase()
  431. .trim();
  432. if (this.editing.song.genres.indexOf(genre) !== -1)
  433. return Toast.methods.addToast("Genre already exists", 3000);
  434. if (genre) {
  435. this.editing.song.genres.push(genre);
  436. document.getElementById("new-genre").value = "";
  437. } else Toast.methods.addToast("Genre cannot be empty", 3000);
  438. } else if (type == "artists") {
  439. let artist = document.getElementById("new-artist").value;
  440. if (this.editing.song.artists.indexOf(artist) !== -1)
  441. return Toast.methods.addToast(
  442. "Artist already exists",
  443. 3000
  444. );
  445. if (document.getElementById("new-artist").value !== "") {
  446. this.editing.song.artists.push(artist);
  447. document.getElementById("new-artist").value = "";
  448. } else Toast.methods.addToast("Artist cannot be empty", 3000);
  449. }
  450. },
  451. removeTag: function(type, index) {
  452. if (type == "genres") this.editing.song.genres.splice(index, 1);
  453. else if (type == "artists")
  454. this.editing.song.artists.splice(index, 1);
  455. },
  456. getSpotifySongs: function() {
  457. this.socket.emit(
  458. "apis.getSpotifySongs",
  459. this.spotify.title,
  460. this.spotify.artist,
  461. res => {
  462. if (res.status === "success") {
  463. Toast.methods.addToast(
  464. `Succesfully got ${res.songs.length} song${
  465. res.songs.length !== 1 ? "s" : ""
  466. }.`,
  467. 3000
  468. );
  469. this.spotify.songs = res.songs;
  470. } else
  471. Toast.methods.addToast(
  472. `Failed to get songs. ${res.message}`,
  473. 3000
  474. );
  475. }
  476. );
  477. },
  478. ...mapActions("admin/songs", [
  479. "stopVideo",
  480. "loadVideoById",
  481. "pauseVideo",
  482. "getCurrentTime",
  483. "editSong"
  484. ]),
  485. ...mapActions("modals", ["toggleModal", "closeCurrentModal"])
  486. },
  487. mounted: function() {
  488. let _this = this;
  489. // if (this.modals.editSong = false) this.video.player.stopVideo();
  490. // this.loadVideoById(
  491. // this.editing.song.songId,
  492. // this.editing.song.skipDuration
  493. // );
  494. io.getSocket(socket => (_this.socket = socket));
  495. setInterval(() => {
  496. if (
  497. _this.video.paused === false &&
  498. _this.playerReady &&
  499. _this.getCurrentTime().then(time => {
  500. return time;
  501. }) -
  502. _this.editing.song.skipDuration >
  503. _this.editing.song.duration
  504. ) {
  505. _this.video.paused = false;
  506. _this.video.player.stopVideo();
  507. }
  508. if (this.playerReady) {
  509. _this
  510. .getCurrentTime(3)
  511. .then(time => (this.youtubeVideoCurrentTime = time));
  512. }
  513. }, 200);
  514. this.video.player = new window.YT.Player("player", {
  515. height: 315,
  516. width: 560,
  517. videoId: this.editing.song.songId,
  518. playerVars: {
  519. controls: 0,
  520. iv_load_policy: 3,
  521. rel: 0,
  522. showinfo: 0,
  523. autoplay: 1
  524. },
  525. startSeconds: _this.editing.song.skipDuration,
  526. events: {
  527. onReady: () => {
  528. let volume = parseInt(localStorage.getItem("volume"));
  529. volume = typeof volume === "number" ? volume : 20;
  530. console.log("Seekto: " + _this.editing.song.skipDuration);
  531. _this.video.player.seekTo(_this.editing.song.skipDuration);
  532. _this.video.player.setVolume(volume);
  533. if (volume > 0) _this.video.player.unMute();
  534. this.youtubeVideoDuration = _this.video.player.getDuration();
  535. this.youtubeVideoNote = "(~)";
  536. _this.playerReady = true;
  537. },
  538. onStateChange: event => {
  539. if (event.data === 1) {
  540. if (!_this.video.autoPlayed) {
  541. _this.video.autoPlayed = true;
  542. return _this.video.player.stopVideo();
  543. }
  544. _this.video.paused = false;
  545. let youtubeDuration = _this.video.player.getDuration();
  546. this.youtubeVideoDuration = youtubeDuration;
  547. this.youtubeVideoNote = "";
  548. youtubeDuration -= _this.editing.song.skipDuration;
  549. if (_this.editing.song.duration > youtubeDuration) {
  550. this.video.player.stopVideo();
  551. _this.video.paused = true;
  552. Toast.methods.addToast(
  553. "Video can't play. Specified duration is bigger than the YouTube song duration.",
  554. 4000
  555. );
  556. } else if (_this.editing.song.duration <= 0) {
  557. this.video.player.stopVideo();
  558. _this.video.paused = true;
  559. Toast.methods.addToast(
  560. "Video can't play. Specified duration has to be more than 0 seconds.",
  561. 4000
  562. );
  563. }
  564. if (
  565. _this.getCurrentTime(time => {
  566. return time;
  567. }) < _this.editing.song.skipDuration
  568. ) {
  569. _this.video.player.seekTo(
  570. _this.editing.song.skipDuration
  571. );
  572. }
  573. } else if (event.data === 2) {
  574. this.video.paused = true;
  575. }
  576. }
  577. }
  578. });
  579. let volume = parseInt(localStorage.getItem("volume"));
  580. document.getElementById("volumeSlider").value = volume =
  581. typeof volume === "number" ? volume : 20;
  582. }
  583. };
  584. </script>
  585. <style lang="scss" scoped>
  586. input[type="range"] {
  587. -webkit-appearance: none;
  588. width: 100%;
  589. margin: 7.3px 0;
  590. }
  591. input[type="range"]:focus {
  592. outline: none;
  593. }
  594. input[type="range"]::-webkit-slider-runnable-track {
  595. width: 100%;
  596. height: 5.2px;
  597. cursor: pointer;
  598. box-shadow: 0;
  599. background: #c2c0c2;
  600. border-radius: 0;
  601. border: 0;
  602. }
  603. input[type="range"]::-webkit-slider-thumb {
  604. box-shadow: 0;
  605. border: 0;
  606. height: 19px;
  607. width: 19px;
  608. border-radius: 15px;
  609. background: #03a9f4;
  610. cursor: pointer;
  611. -webkit-appearance: none;
  612. margin-top: -6.5px;
  613. }
  614. input[type="range"]::-moz-range-track {
  615. width: 100%;
  616. height: 5.2px;
  617. cursor: pointer;
  618. box-shadow: 0;
  619. background: #c2c0c2;
  620. border-radius: 0;
  621. border: 0;
  622. }
  623. input[type="range"]::-moz-range-thumb {
  624. box-shadow: 0;
  625. border: 0;
  626. height: 19px;
  627. width: 19px;
  628. border-radius: 15px;
  629. background: #03a9f4;
  630. cursor: pointer;
  631. -webkit-appearance: none;
  632. margin-top: -6.5px;
  633. }
  634. input[type="range"]::-ms-track {
  635. width: 100%;
  636. height: 5.2px;
  637. cursor: pointer;
  638. box-shadow: 0;
  639. background: #c2c0c2;
  640. border-radius: 1.3px;
  641. }
  642. input[type="range"]::-ms-fill-lower {
  643. background: #c2c0c2;
  644. border: 0;
  645. border-radius: 0;
  646. box-shadow: 0;
  647. }
  648. input[type="range"]::-ms-fill-upper {
  649. background: #c2c0c2;
  650. border: 0;
  651. border-radius: 0;
  652. box-shadow: 0;
  653. }
  654. input[type="range"]::-ms-thumb {
  655. box-shadow: 0;
  656. border: 0;
  657. height: 15px;
  658. width: 15px;
  659. border-radius: 15px;
  660. background: #03a9f4;
  661. cursor: pointer;
  662. -webkit-appearance: none;
  663. margin-top: 1.5px;
  664. }
  665. .controls {
  666. display: flex;
  667. flex-direction: column;
  668. align-items: center;
  669. }
  670. .artist-genres {
  671. display: flex;
  672. justify-content: space-between;
  673. }
  674. #volumeSlider {
  675. margin-bottom: 15px;
  676. }
  677. .has-text-centered {
  678. padding: 10px;
  679. }
  680. .thumbnail-preview {
  681. display: flex;
  682. margin: 0 auto 25px auto;
  683. max-width: 200px;
  684. width: 100%;
  685. }
  686. .modal-card-body,
  687. .modal-card-foot {
  688. border-top: 0;
  689. }
  690. .label,
  691. .checkbox,
  692. h5 {
  693. font-weight: normal;
  694. }
  695. .video-container {
  696. display: flex;
  697. flex-direction: column;
  698. align-items: center;
  699. padding: 10px;
  700. iframe {
  701. pointer-events: none;
  702. }
  703. }
  704. .save-changes {
  705. color: #fff;
  706. }
  707. .tag:not(:last-child) {
  708. margin-right: 5px;
  709. }
  710. .reports-length {
  711. color: #ff4545;
  712. font-weight: bold;
  713. display: flex;
  714. justify-content: center;
  715. }
  716. .report-link {
  717. color: #000;
  718. }
  719. </style>