EditPlaylist.vue 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924
  1. <template>
  2. <modal
  3. :title="
  4. userId === playlist.createdBy ? 'Edit Playlist' : 'View Playlist'
  5. "
  6. class="edit-playlist-modal"
  7. >
  8. <div
  9. slot="body"
  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
  22. id="playlist-settings-section"
  23. v-if="userId === playlist.createdBy || isEditable()"
  24. class="section"
  25. >
  26. <div v-if="isEditable()">
  27. <h4 class="section-title">Edit Details</h4>
  28. <p class="section-description">
  29. Change the display name and privacy of the playlist.
  30. </p>
  31. <hr class="section-horizontal-rule" />
  32. <label class="label"> Change display name </label>
  33. <div class="control is-grouped input-with-button">
  34. <p class="control is-expanded">
  35. <input
  36. v-model="playlist.displayName"
  37. class="input"
  38. type="text"
  39. placeholder="Playlist Display Name"
  40. @keyup.enter="renamePlaylist()"
  41. />
  42. </p>
  43. <p class="control">
  44. <a
  45. class="button is-info"
  46. @click.prevent="renamePlaylist()"
  47. href="#"
  48. >Rename</a
  49. >
  50. </p>
  51. </div>
  52. </div>
  53. <div>
  54. <label class="label"> Change privacy </label>
  55. <div class="control is-grouped input-with-button">
  56. <div class="control is-expanded select">
  57. <select v-model="playlist.privacy">
  58. <option value="private">Private</option>
  59. <option value="public">Public</option>
  60. </select>
  61. </div>
  62. <p class="control">
  63. <a
  64. class="button is-info"
  65. @click.prevent="updatePrivacy()"
  66. href="#"
  67. >Update Privacy</a
  68. >
  69. </p>
  70. </div>
  71. </div>
  72. </div>
  73. <div
  74. id="import-from-youtube-section"
  75. class="section"
  76. v-if="isEditable()"
  77. >
  78. <h4 class="section-title">Import from YouTube</h4>
  79. <p class="section-description">
  80. Import a playlist or song by searching or using a link
  81. from YouTube.
  82. </p>
  83. <hr class="section-horizontal-rule" />
  84. <label class="label">
  85. Search for a playlist from YouTube
  86. </label>
  87. <div class="control is-grouped input-with-button">
  88. <p class="control is-expanded">
  89. <input
  90. class="input"
  91. type="text"
  92. placeholder="Enter YouTube Playlist URL here..."
  93. v-model="search.playlist.query"
  94. @keyup.enter="importPlaylist()"
  95. />
  96. </p>
  97. <p class="control has-addons">
  98. <span class="select" id="playlist-import-type">
  99. <select
  100. v-model="
  101. search.playlist.isImportingOnlyMusic
  102. "
  103. >
  104. <option :value="false">Import all</option>
  105. <option :value="true">
  106. Import only music
  107. </option>
  108. </select>
  109. </span>
  110. <a
  111. class="button is-info"
  112. @click.prevent="importPlaylist()"
  113. href="#"
  114. ><i class="material-icons icon-with-button"
  115. >publish</i
  116. >Import</a
  117. >
  118. </p>
  119. </div>
  120. <label class="label">
  121. Search for a song from YouTube
  122. </label>
  123. <div class="control is-grouped input-with-button">
  124. <p class="control is-expanded">
  125. <input
  126. class="input"
  127. type="text"
  128. placeholder="Enter your YouTube query here..."
  129. v-model="search.songs.query"
  130. autofocus
  131. @keyup.enter="searchForSongs()"
  132. />
  133. </p>
  134. <p class="control">
  135. <a
  136. class="button is-info"
  137. @click.prevent="searchForSongs()"
  138. href="#"
  139. ><i class="material-icons icon-with-button"
  140. >search</i
  141. >Search</a
  142. >
  143. </p>
  144. </div>
  145. <div
  146. v-if="search.songs.results.length > 0"
  147. id="song-query-results"
  148. >
  149. <search-query-item
  150. v-for="(result, index) in search.songs.results"
  151. :key="index"
  152. :result="result"
  153. >
  154. <div slot="actions">
  155. <transition
  156. name="search-query-actions"
  157. mode="out-in"
  158. >
  159. <a
  160. class="button is-success"
  161. v-if="result.isAddedToQueue"
  162. href="#"
  163. key="added-to-playlist"
  164. >
  165. <i
  166. class="material-icons icon-with-button"
  167. >done</i
  168. >
  169. Added to playlist
  170. </a>
  171. <a
  172. class="button is-dark"
  173. v-else
  174. @click.prevent="
  175. addSongToPlaylist(result.id, index)
  176. "
  177. href="#"
  178. key="add-to-playlist"
  179. >
  180. <i
  181. class="material-icons icon-with-button"
  182. >add</i
  183. >
  184. Add to playlist
  185. </a>
  186. </transition>
  187. </div>
  188. </search-query-item>
  189. <a
  190. class="button is-default load-more-button"
  191. @click.prevent="loadMoreSongs()"
  192. href="#"
  193. >
  194. Load more...
  195. </a>
  196. </div>
  197. </div>
  198. </div>
  199. <div id="second-column">
  200. <div id="rearrange-songs-section" class="section">
  201. <div v-if="isEditable()">
  202. <h4 class="section-title">Rearrange Songs</h4>
  203. <p class="section-description">
  204. Drag and drop songs to change their order
  205. </p>
  206. <hr class="section-horizontal-rule" />
  207. </div>
  208. <aside class="menu">
  209. <draggable
  210. class="menu-list scrollable-list"
  211. tag="ul"
  212. v-if="playlist.songs.length > 0"
  213. v-model="playlist.songs"
  214. v-bind="dragOptions"
  215. @start="drag = true"
  216. @end="drag = false"
  217. @change="updateSongPositioning"
  218. >
  219. <transition-group
  220. type="transition"
  221. :name="
  222. !drag ? 'draggable-list-transition' : null
  223. "
  224. >
  225. <li
  226. v-for="(song, index) in playlist.songs"
  227. :key="'key-' + song._id"
  228. >
  229. <song-item
  230. :song="song"
  231. :class="{
  232. 'item-draggable': isEditable()
  233. }"
  234. >
  235. <div
  236. class="song-actions"
  237. slot="actions"
  238. >
  239. <i
  240. class="material-icons add-to-queue-icon"
  241. v-if="
  242. station.partyMode &&
  243. !station.locked
  244. "
  245. @click="
  246. addSongToQueue(
  247. song.youtubeId
  248. )
  249. "
  250. content="Add Song to Queue"
  251. v-tippy
  252. >queue</i
  253. >
  254. <confirm
  255. v-if="
  256. userId ===
  257. playlist.createdBy ||
  258. isEditable()
  259. "
  260. placement="left"
  261. @confirm="
  262. removeSongFromPlaylist(
  263. song.youtubeId
  264. )
  265. "
  266. >
  267. <i
  268. class="material-icons delete-icon"
  269. content="Remove Song from Playlist"
  270. v-tippy
  271. >delete_forever</i
  272. >
  273. </confirm>
  274. <i
  275. class="material-icons"
  276. v-if="isEditable() && index > 0"
  277. @click="moveSongToTop(index)"
  278. content="Move to top of Playlist"
  279. v-tippy
  280. >vertical_align_top</i
  281. >
  282. <i
  283. v-if="
  284. isEditable() &&
  285. playlist.songs.length -
  286. 1 !==
  287. index
  288. "
  289. @click="moveSongToBottom(index)"
  290. class="material-icons"
  291. content="Move to bottom of Playlist"
  292. v-tippy
  293. >vertical_align_bottom</i
  294. >
  295. </div>
  296. </song-item>
  297. </li>
  298. </transition-group>
  299. </draggable>
  300. <p v-else class="nothing-here-text">
  301. This playlist doesn't have any songs.
  302. </p>
  303. </aside>
  304. </div>
  305. </div>
  306. <!--
  307. <button
  308. class="button is-info"
  309. @click="shuffle()"
  310. v-if="playlist.isUserModifiable"
  311. >
  312. Shuffle
  313. </button>
  314. <h5>Edit playlist details:</h5>
  315. -->
  316. </div>
  317. <div slot="footer">
  318. <a
  319. class="button is-default"
  320. v-if="
  321. this.userId === this.playlist.createdBy ||
  322. isEditable() ||
  323. this.playlist.privacy === 'public'
  324. "
  325. @click="downloadPlaylist()"
  326. href="#"
  327. >
  328. Download Playlist
  329. </a>
  330. <div class="right">
  331. <confirm
  332. v-if="playlist.type === 'station'"
  333. @confirm="clearAndRefillStationPlaylist()"
  334. >
  335. <a class="button is-danger">
  336. Clear and refill station playlist
  337. </a>
  338. </confirm>
  339. <confirm
  340. v-if="playlist.type === 'genre'"
  341. @confirm="clearAndRefillGenrePlaylist()"
  342. >
  343. <a class="button is-danger">
  344. Clear and refill genre playlist
  345. </a>
  346. </confirm>
  347. <confirm v-if="isEditable()" @confirm="removePlaylist()">
  348. <a class="button is-danger"> Remove Playlist </a>
  349. </confirm>
  350. </div>
  351. </div>
  352. </modal>
  353. </template>
  354. <script>
  355. import { mapState, mapGetters, mapActions } from "vuex";
  356. import draggable from "vuedraggable";
  357. import Toast from "toasters";
  358. import SearchYoutube from "@/mixins/SearchYoutube.vue";
  359. import validation from "@/validation";
  360. import Confirm from "@/components/Confirm.vue";
  361. import Modal from "../Modal.vue";
  362. import SearchQueryItem from "../SearchQueryItem.vue";
  363. import SongItem from "../SongItem.vue";
  364. import utils from "../../../js/utils";
  365. export default {
  366. components: { Modal, draggable, Confirm, SearchQueryItem, SongItem },
  367. mixins: [SearchYoutube],
  368. data() {
  369. return {
  370. utils,
  371. drag: false,
  372. apiDomain: "",
  373. playlist: { songs: [] }
  374. };
  375. },
  376. computed: {
  377. ...mapState("station", {
  378. station: state => state.station
  379. }),
  380. ...mapState("user/playlists", {
  381. editing: state => state.editing
  382. }),
  383. ...mapState({
  384. userId: state => state.user.auth.userId,
  385. userRole: state => state.user.auth.role
  386. }),
  387. dragOptions() {
  388. return {
  389. animation: 200,
  390. group: "songs",
  391. disabled: !this.isEditable(),
  392. ghostClass: "draggable-list-ghost"
  393. };
  394. },
  395. ...mapGetters({
  396. socket: "websockets/getSocket"
  397. })
  398. },
  399. watch: {
  400. "search.songs.results": function checkIfSongInPlaylist(songs) {
  401. songs.forEach((searchItem, index) =>
  402. this.playlist.songs.find(song => {
  403. if (song.youtubeId === searchItem.id)
  404. this.search.songs.results[index].isAddedToQueue = true;
  405. return song.youtubeId === searchItem.id;
  406. })
  407. );
  408. }
  409. },
  410. mounted() {
  411. this.socket.dispatch("playlists.getPlaylist", this.editing, res => {
  412. if (res.status === "success") {
  413. this.playlist = res.data.playlist;
  414. this.playlist.songs.sort((a, b) => a.position - b.position);
  415. this.playlist.oldId = res.data.playlist._id;
  416. } else new Toast(res.message);
  417. });
  418. this.socket.on("event:playlist.addSong", res => {
  419. if (this.playlist._id === res.data.playlistId)
  420. this.playlist.songs.push(res.data.song);
  421. });
  422. this.socket.on("event:playlist.removeSong", res => {
  423. if (this.playlist._id === res.data.playlistId) {
  424. // remove song from array of playlists
  425. this.playlist.songs.forEach((song, index) => {
  426. if (song.youtubeId === res.data.youtubeId)
  427. this.playlist.songs.splice(index, 1);
  428. });
  429. // if this song is in search results, mark it available to add to the playlist again
  430. this.search.songs.results.forEach((searchItem, index) => {
  431. if (res.data.youtubeId === searchItem.id) {
  432. this.search.songs.results[index].isAddedToQueue = false;
  433. }
  434. });
  435. }
  436. });
  437. this.socket.on("event:playlist.updateDisplayName", res => {
  438. if (this.playlist._id === res.data.playlistId)
  439. this.playlist.displayName = res.data.displayName;
  440. });
  441. this.socket.on("event:playlist.repositionSongs", res => {
  442. if (this.playlist._id === res.data.playlistId) {
  443. // for each song that has a new position
  444. res.data.songsBeingChanged.forEach(changedSong => {
  445. this.playlist.songs.forEach((song, index) => {
  446. // find song locally
  447. if (song.youtubeId === changedSong.youtubeId) {
  448. // change song position attribute
  449. this.playlist.songs[index].position =
  450. changedSong.position;
  451. // reposition in array if needed
  452. if (index !== changedSong.position - 1)
  453. this.playlist.songs.splice(
  454. changedSong.position - 1,
  455. 0,
  456. this.playlist.songs.splice(index, 1)[0]
  457. );
  458. }
  459. });
  460. });
  461. }
  462. });
  463. },
  464. methods: {
  465. importPlaylist() {
  466. let isImportingPlaylist = true;
  467. // import query is blank
  468. if (!this.search.playlist.query)
  469. return new Toast("Please enter a YouTube playlist URL.");
  470. const regex = new RegExp(`[\\?&]list=([^&#]*)`);
  471. const splitQuery = regex.exec(this.search.playlist.query);
  472. if (!splitQuery) {
  473. return new Toast({
  474. content: "Please enter a valid YouTube playlist URL.",
  475. timeout: 4000
  476. });
  477. }
  478. // don't give starting import message instantly in case of instant error
  479. setTimeout(() => {
  480. if (isImportingPlaylist) {
  481. new Toast(
  482. "Starting to import your playlist. This can take some time to do."
  483. );
  484. }
  485. }, 750);
  486. return this.socket.dispatch(
  487. "playlists.addSetToPlaylist",
  488. this.search.playlist.query,
  489. this.playlist._id,
  490. this.search.playlist.isImportingOnlyMusic,
  491. res => {
  492. new Toast({ content: res.message, timeout: 20000 });
  493. if (res.status === "success") {
  494. isImportingPlaylist = false;
  495. if (this.search.playlist.isImportingOnlyMusic) {
  496. new Toast({
  497. content: `${res.data.stats.songsInPlaylistTotal} of the ${res.data.stats.videosInPlaylistTotal} videos in the playlist were songs.`,
  498. timeout: 20000
  499. });
  500. }
  501. }
  502. }
  503. );
  504. },
  505. isEditable() {
  506. return (
  507. this.playlist.isUserModifiable &&
  508. (this.userId === this.playlist.createdBy ||
  509. this.userRole === "admin")
  510. );
  511. },
  512. updateSongPositioning({ moved }) {
  513. if (!moved) return; // we only need to update when song is moved
  514. const songsBeingChanged = [];
  515. this.playlist.songs.forEach((song, index) => {
  516. if (song.position !== index + 1)
  517. songsBeingChanged.push({
  518. youtubeId: song.youtubeId,
  519. position: index + 1
  520. });
  521. });
  522. this.socket.dispatch(
  523. "playlists.repositionSongs",
  524. this.playlist._id,
  525. songsBeingChanged,
  526. res => {
  527. new Toast(res.message);
  528. }
  529. );
  530. },
  531. totalLength() {
  532. let length = 0;
  533. this.playlist.songs.forEach(song => {
  534. length += song.duration;
  535. });
  536. return this.utils.formatTimeLong(length);
  537. },
  538. shuffle() {
  539. this.socket.dispatch(
  540. "playlists.shuffle",
  541. this.playlist._id,
  542. res => {
  543. new Toast(res.message);
  544. if (res.status === "success") {
  545. this.playlist.songs = res.data.playlist.songs.sort(
  546. (a, b) => a.position - b.position
  547. );
  548. }
  549. }
  550. );
  551. },
  552. addSongToPlaylist(id, index) {
  553. this.socket.dispatch(
  554. "playlists.addSongToPlaylist",
  555. false,
  556. id,
  557. this.playlist._id,
  558. res => {
  559. new Toast(res.message);
  560. if (res.status === "success")
  561. this.search.songs.results[index].isAddedToQueue = true;
  562. }
  563. );
  564. },
  565. removeSongFromPlaylist(id) {
  566. if (this.playlist.displayName === "Liked Songs") {
  567. this.socket.dispatch("songs.unlike", id, res => {
  568. new Toast(res.message);
  569. });
  570. }
  571. if (this.playlist.displayName === "Disliked Songs") {
  572. this.socket.dispatch("songs.undislike", id, res => {
  573. new Toast(res.message);
  574. });
  575. } else {
  576. this.socket.dispatch(
  577. "playlists.removeSongFromPlaylist",
  578. id,
  579. this.playlist._id,
  580. res => {
  581. new Toast(res.message);
  582. }
  583. );
  584. }
  585. },
  586. renamePlaylist() {
  587. const { displayName } = this.playlist;
  588. if (!validation.isLength(displayName, 2, 32))
  589. return new Toast(
  590. "Display name must have between 2 and 32 characters."
  591. );
  592. if (!validation.regex.ascii.test(displayName))
  593. return new Toast(
  594. "Invalid display name format. Only ASCII characters are allowed."
  595. );
  596. return this.socket.dispatch(
  597. "playlists.updateDisplayName",
  598. this.playlist._id,
  599. this.playlist.displayName,
  600. res => {
  601. new Toast(res.message);
  602. }
  603. );
  604. },
  605. removePlaylist() {
  606. this.socket.dispatch("playlists.remove", this.playlist._id, res => {
  607. new Toast(res.message);
  608. if (res.status === "success") this.closeModal("editPlaylist");
  609. });
  610. },
  611. async downloadPlaylist() {
  612. if (this.apiDomain === "")
  613. this.apiDomain = await lofig.get("apiDomain");
  614. fetch(
  615. `${this.apiDomain}/export/privatePlaylist/${this.playlist._id}`,
  616. { credentials: "include" }
  617. )
  618. .then(res => res.blob())
  619. .then(blob => {
  620. const url = window.URL.createObjectURL(blob);
  621. const a = document.createElement("a");
  622. a.style.display = "none";
  623. a.href = url;
  624. a.download = `musare-privateplaylist-${
  625. this.playlist._id
  626. }-${new Date().toISOString()}.json`;
  627. document.body.appendChild(a);
  628. a.click();
  629. window.URL.revokeObjectURL(url);
  630. new Toast("Successfully downloaded playlist.");
  631. })
  632. .catch(
  633. () => new Toast("Failed to export and download playlist.")
  634. );
  635. },
  636. moveSongToTop(index) {
  637. this.playlist.songs.splice(
  638. 0,
  639. 0,
  640. this.playlist.songs.splice(index, 1)[0]
  641. );
  642. this.updateSongPositioning({ moved: {} });
  643. },
  644. moveSongToBottom(index) {
  645. this.playlist.songs.splice(
  646. this.playlist.songs.length,
  647. 0,
  648. this.playlist.songs.splice(index, 1)[0]
  649. );
  650. this.updateSongPositioning({ moved: {} });
  651. },
  652. updatePrivacy() {
  653. const { privacy } = this.playlist;
  654. if (privacy === "public" || privacy === "private") {
  655. this.socket.dispatch(
  656. "playlists.updatePrivacy",
  657. this.playlist._id,
  658. privacy,
  659. res => {
  660. new Toast(res.message);
  661. }
  662. );
  663. }
  664. },
  665. addSongToQueue(youtubeId) {
  666. this.socket.dispatch(
  667. "stations.addToQueue",
  668. this.station._id,
  669. youtubeId,
  670. data => {
  671. if (data.status !== "success")
  672. new Toast({
  673. content: `Error: ${data.message}`,
  674. timeout: 8000
  675. });
  676. else new Toast({ content: data.message, timeout: 4000 });
  677. }
  678. );
  679. },
  680. clearAndRefillStationPlaylist() {
  681. this.socket.dispatch(
  682. "playlists.clearAndRefillStationPlaylist",
  683. this.playlist._id,
  684. data => {
  685. console.log(data.message);
  686. if (data.status !== "success")
  687. new Toast({
  688. content: `Error: ${data.message}`,
  689. timeout: 8000
  690. });
  691. else new Toast({ content: data.message, timeout: 4000 });
  692. }
  693. );
  694. },
  695. clearAndRefillGenrePlaylist() {
  696. this.socket.dispatch(
  697. "playlists.clearAndRefillGenrePlaylist",
  698. this.playlist._id,
  699. data => {
  700. if (data.status !== "success")
  701. new Toast({
  702. content: `Error: ${data.message}`,
  703. timeout: 8000
  704. });
  705. else new Toast({ content: data.message, timeout: 4000 });
  706. }
  707. );
  708. },
  709. ...mapActions("modalVisibility", ["openModal", "closeModal"])
  710. }
  711. };
  712. </script>
  713. <style lang="scss">
  714. .edit-playlist-modal {
  715. .modal-card {
  716. width: 1300px;
  717. .modal-card-body {
  718. padding: 16px;
  719. }
  720. }
  721. }
  722. </style>
  723. <style lang="scss" scoped>
  724. .night-mode {
  725. .section {
  726. background-color: var(--dark-grey-3) !important;
  727. }
  728. .label,
  729. p,
  730. strong {
  731. color: var(--light-grey-2);
  732. }
  733. }
  734. .menu-list li {
  735. display: flex;
  736. justify-content: space-between;
  737. &:not(:last-of-type) {
  738. margin-bottom: 10px;
  739. }
  740. a {
  741. display: flex;
  742. }
  743. }
  744. .controls {
  745. display: flex;
  746. a {
  747. display: flex;
  748. align-items: center;
  749. }
  750. }
  751. @media screen and (max-width: 1300px) {
  752. .edit-playlist-modal .edit-playlist-modal-inner-container {
  753. height: auto !important;
  754. #import-from-youtube-section #song-query-results,
  755. .section {
  756. max-width: 100% !important;
  757. }
  758. }
  759. }
  760. .edit-playlist-modal {
  761. .edit-playlist-modal-inner-container {
  762. display: flex;
  763. flex-wrap: wrap;
  764. height: 100%;
  765. &.view-only {
  766. height: auto !important;
  767. #first-column {
  768. flex-basis: 100%;
  769. }
  770. .section {
  771. max-width: 100% !important;
  772. }
  773. }
  774. }
  775. .nothing-here-text {
  776. display: flex;
  777. align-items: center;
  778. justify-content: center;
  779. }
  780. .section {
  781. padding: 15px !important;
  782. margin: 0 10px;
  783. max-width: 600px;
  784. display: flex;
  785. flex-direction: column;
  786. flex-grow: 1;
  787. }
  788. .label {
  789. font-size: 1rem;
  790. font-weight: normal;
  791. }
  792. .input-with-button .button {
  793. width: 150px;
  794. }
  795. #first-column {
  796. max-width: 100%;
  797. height: 100%;
  798. overflow-y: auto;
  799. flex-grow: 1;
  800. .section {
  801. width: auto;
  802. }
  803. #playlist-info-section {
  804. border: 1px solid var(--light-grey-3);
  805. border-radius: 3px;
  806. padding: 15px !important;
  807. h3 {
  808. font-weight: 600;
  809. font-size: 30px;
  810. }
  811. h5 {
  812. font-size: 18px;
  813. }
  814. h3,
  815. h5 {
  816. margin: 0;
  817. }
  818. }
  819. #import-from-youtube-section {
  820. #playlist-import-type select {
  821. border-radius: 0;
  822. }
  823. #song-query-results {
  824. padding: 10px;
  825. margin-top: 10px;
  826. border: 1px solid var(--light-grey-3);
  827. border-radius: 3px;
  828. max-width: 565px;
  829. .search-query-item:not(:last-of-type) {
  830. margin-bottom: 10px;
  831. }
  832. }
  833. .load-more-button {
  834. width: 100%;
  835. margin-top: 10px;
  836. }
  837. }
  838. }
  839. #second-column {
  840. max-width: 100%;
  841. height: 100%;
  842. overflow-y: auto;
  843. flex-grow: 1;
  844. }
  845. }
  846. </style>