Songs.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800
  1. <template>
  2. <div>
  3. <page-metadata title="Admin | Songs" />
  4. <div class="admin-tab">
  5. <div class="button-row">
  6. <button
  7. class="button is-primary"
  8. @click="toggleKeyboardShortcutsHelper"
  9. @dblclick="resetKeyboardShortcutsHelper"
  10. >
  11. Keyboard shortcuts helper
  12. </button>
  13. <button
  14. class="button is-primary"
  15. @click="openModal('requestSong')"
  16. >
  17. Request song
  18. </button>
  19. <button
  20. class="button is-primary"
  21. @click="openModal('importAlbum')"
  22. >
  23. Import album
  24. </button>
  25. <run-job-dropdown :jobs="jobs" />
  26. </div>
  27. <advanced-table
  28. :column-default="columnDefault"
  29. :columns="columns"
  30. :filters="filters"
  31. data-action="songs.getData"
  32. name="admin-songs"
  33. :events="events"
  34. >
  35. <template #column-options="slotProps">
  36. <div class="row-options">
  37. <button
  38. class="
  39. button
  40. is-primary
  41. icon-with-button
  42. material-icons
  43. "
  44. @click="editOne(slotProps.item)"
  45. :disabled="slotProps.item.removed"
  46. content="Edit Song"
  47. v-tippy
  48. >
  49. edit
  50. </button>
  51. </div>
  52. </template>
  53. <template #column-thumbnailImage="slotProps">
  54. <img
  55. class="song-thumbnail"
  56. :src="slotProps.item.thumbnail"
  57. onerror="this.src='/assets/notes-transparent.png'"
  58. loading="lazy"
  59. />
  60. </template>
  61. <template #column-thumbnailUrl="slotProps">
  62. <a :href="slotProps.item.thumbnail" target="_blank">
  63. {{ slotProps.item.thumbnail }}
  64. </a>
  65. </template>
  66. <template #column-title="slotProps">
  67. <span :title="slotProps.item.title">{{
  68. slotProps.item.title
  69. }}</span>
  70. </template>
  71. <template #column-artists="slotProps">
  72. <span :title="slotProps.item.artists.join(', ')">{{
  73. slotProps.item.artists.join(", ")
  74. }}</span>
  75. </template>
  76. <template #column-genres="slotProps">
  77. <span :title="slotProps.item.genres.join(', ')">{{
  78. slotProps.item.genres.join(", ")
  79. }}</span>
  80. </template>
  81. <template #column-likes="slotProps">
  82. <span :title="slotProps.item.likes">{{
  83. slotProps.item.likes
  84. }}</span>
  85. </template>
  86. <template #column-dislikes="slotProps">
  87. <span :title="slotProps.item.dislikes">{{
  88. slotProps.item.dislikes
  89. }}</span>
  90. </template>
  91. <template #column-_id="slotProps">
  92. <span :title="slotProps.item._id">{{
  93. slotProps.item._id
  94. }}</span>
  95. </template>
  96. <template #column-youtubeId="slotProps">
  97. <a
  98. :href="
  99. 'https://www.youtube.com/watch?v=' +
  100. `${slotProps.item.youtubeId}`
  101. "
  102. target="_blank"
  103. >
  104. {{ slotProps.item.youtubeId }}
  105. </a>
  106. </template>
  107. <template #column-status="slotProps">
  108. <span :title="slotProps.item.status">{{
  109. slotProps.item.status
  110. }}</span>
  111. </template>
  112. <template #column-duration="slotProps">
  113. <span :title="slotProps.item.duration">{{
  114. slotProps.item.duration
  115. }}</span>
  116. </template>
  117. <template #column-skipDuration="slotProps">
  118. <span :title="slotProps.item.skipDuration">{{
  119. slotProps.item.skipDuration
  120. }}</span>
  121. </template>
  122. <template #column-requestedBy="slotProps">
  123. <user-id-to-username
  124. :user-id="slotProps.item.requestedBy"
  125. :link="true"
  126. />
  127. </template>
  128. <template #column-requestedAt="slotProps">
  129. <span :title="new Date(slotProps.item.requestedAt)">{{
  130. getDateFormatted(slotProps.item.requestedAt)
  131. }}</span>
  132. </template>
  133. <template #column-verifiedBy="slotProps">
  134. <user-id-to-username
  135. :user-id="slotProps.item.verifiedBy"
  136. :link="true"
  137. />
  138. </template>
  139. <template #column-verifiedAt="slotProps">
  140. <span :title="new Date(slotProps.item.verifiedAt)">{{
  141. getDateFormatted(slotProps.item.verifiedAt)
  142. }}</span>
  143. </template>
  144. <template #bulk-actions="slotProps">
  145. <div class="bulk-actions">
  146. <i
  147. class="material-icons edit-songs-icon"
  148. @click.prevent="editMany(slotProps.item)"
  149. content="Edit Songs"
  150. v-tippy
  151. >
  152. edit
  153. </i>
  154. <i
  155. class="material-icons verify-songs-icon"
  156. @click.prevent="verifyMany(slotProps.item)"
  157. content="Verify Songs"
  158. v-tippy
  159. >
  160. check_circle
  161. </i>
  162. <i
  163. class="material-icons unverify-songs-icon"
  164. @click.prevent="unverifyMany(slotProps.item)"
  165. content="Unverify Songs"
  166. v-tippy
  167. >
  168. cancel
  169. </i>
  170. <i
  171. class="material-icons tag-songs-icon"
  172. @click.prevent="tagMany(slotProps.item)"
  173. content="Tag Songs"
  174. v-tippy
  175. >
  176. local_offer
  177. </i>
  178. <i
  179. class="material-icons artists-songs-icon"
  180. @click.prevent="setArtists(slotProps.item)"
  181. content="Set Artists"
  182. v-tippy
  183. >
  184. group
  185. </i>
  186. <i
  187. class="material-icons genres-songs-icon"
  188. @click.prevent="setGenres(slotProps.item)"
  189. content="Set Genres"
  190. v-tippy
  191. >
  192. theater_comedy
  193. </i>
  194. <quick-confirm
  195. placement="left"
  196. @confirm="deleteMany(slotProps.item)"
  197. >
  198. <i
  199. class="material-icons delete-icon"
  200. content="Delete Songs"
  201. v-tippy
  202. >
  203. delete_forever
  204. </i>
  205. </quick-confirm>
  206. </div>
  207. </template>
  208. </advanced-table>
  209. </div>
  210. <import-album v-if="modals.importAlbum" />
  211. <edit-song v-if="modals.editSong" song-type="songs" :key="song._id" />
  212. <report v-if="modals.report" />
  213. <request-song v-if="modals.requestSong" />
  214. <floating-box
  215. id="keyboardShortcutsHelper"
  216. ref="keyboardShortcutsHelper"
  217. >
  218. <template #body>
  219. <div>
  220. <div>
  221. <span class="biggest"
  222. ><b>Keyboard shortcuts helper</b></span
  223. >
  224. <span
  225. ><b>Ctrl + /</b> - Toggles this keyboard shortcuts
  226. helper</span
  227. >
  228. <span
  229. ><b>Ctrl + Shift + /</b> - Resets the position of
  230. this keyboard shortcuts helper</span
  231. >
  232. <hr />
  233. </div>
  234. <div>
  235. <span class="biggest"><b>Edit song modal</b></span>
  236. <span class="bigger"><b>Navigation</b></span>
  237. <span><b>Home</b> - Edit</span>
  238. <span><b>End</b> - Edit</span>
  239. <hr />
  240. </div>
  241. <div>
  242. <span class="bigger"><b>Player controls</b></span>
  243. <span class="bigger"
  244. ><i>Don't forget to turn off numlock!</i></span
  245. >
  246. <span><b>Numpad up/down</b> - Volume up/down 10%</span>
  247. <span
  248. ><b>Ctrl + Numpad up/down</b> - Volume up/down
  249. 1%</span
  250. >
  251. <span><b>Numpad center</b> - Pause/resume</span>
  252. <span><b>Ctrl + Numpad center</b> - Stop</span>
  253. <span
  254. ><b>Numpad Right</b> - Skip to last 10 seconds</span
  255. >
  256. <hr />
  257. </div>
  258. <div>
  259. <span class="bigger"><b>Form control</b></span>
  260. <span
  261. ><b>Enter</b> - Executes blue button in that
  262. input</span
  263. >
  264. <span
  265. ><b>Shift + Enter</b> - Executes purple/red button
  266. in that input</span
  267. >
  268. <span
  269. ><b>Ctrl + Alt + D</b> - Fill in all Discogs
  270. fields</span
  271. >
  272. <hr />
  273. </div>
  274. <div>
  275. <span class="bigger"><b>Modal control</b></span>
  276. <span><b>Ctrl + S</b> - Save</span>
  277. <span><b>Ctrl + Alt + S</b> - Save and close</span>
  278. <span
  279. ><b>Ctrl + Alt + V</b> - Save, verify and
  280. close</span
  281. >
  282. <span><b>F4</b> - Close without saving</span>
  283. <hr />
  284. </div>
  285. </div>
  286. </template>
  287. </floating-box>
  288. </div>
  289. </template>
  290. <script>
  291. import { mapState, mapActions, mapGetters } from "vuex";
  292. import { defineAsyncComponent } from "vue";
  293. import Toast from "toasters";
  294. import keyboardShortcuts from "@/keyboardShortcuts";
  295. import AdvancedTable from "@/components/AdvancedTable.vue";
  296. import UserIdToUsername from "@/components/UserIdToUsername.vue";
  297. import FloatingBox from "@/components/FloatingBox.vue";
  298. import QuickConfirm from "@/components/QuickConfirm.vue";
  299. import RunJobDropdown from "@/components/RunJobDropdown.vue";
  300. export default {
  301. components: {
  302. EditSong: defineAsyncComponent(() =>
  303. import("@/components/modals/EditSong")
  304. ),
  305. Report: defineAsyncComponent(() =>
  306. import("@/components/modals/Report.vue")
  307. ),
  308. ImportAlbum: defineAsyncComponent(() =>
  309. import("@/components/modals/ImportAlbum.vue")
  310. ),
  311. RequestSong: defineAsyncComponent(() =>
  312. import("@/components/modals/RequestSong.vue")
  313. ),
  314. AdvancedTable,
  315. UserIdToUsername,
  316. FloatingBox,
  317. QuickConfirm,
  318. RunJobDropdown
  319. },
  320. data() {
  321. return {
  322. columnDefault: {
  323. sortable: true,
  324. hidable: true,
  325. defaultVisibility: "shown",
  326. draggable: true,
  327. resizable: true,
  328. minWidth: 200,
  329. maxWidth: 600
  330. },
  331. columns: [
  332. {
  333. name: "options",
  334. displayName: "Edit",
  335. properties: [],
  336. sortable: false,
  337. hidable: false,
  338. resizable: false,
  339. minWidth: 51,
  340. defaultWidth: 51
  341. },
  342. {
  343. name: "thumbnailImage",
  344. displayName: "Thumb",
  345. properties: ["thumbnail"],
  346. sortable: false,
  347. minWidth: 75,
  348. defaultWidth: 75,
  349. maxWidth: 75,
  350. resizable: false
  351. },
  352. {
  353. name: "title",
  354. displayName: "Title",
  355. properties: ["title"],
  356. sortProperty: "title"
  357. },
  358. {
  359. name: "artists",
  360. displayName: "Artists",
  361. properties: ["artists"],
  362. sortable: false
  363. },
  364. {
  365. name: "genres",
  366. displayName: "Genres",
  367. properties: ["genres"],
  368. sortable: false
  369. },
  370. {
  371. name: "likes",
  372. displayName: "Likes",
  373. properties: ["likes"],
  374. sortProperty: "likes",
  375. minWidth: 100,
  376. defaultWidth: 100,
  377. defaultVisibility: "hidden"
  378. },
  379. {
  380. name: "dislikes",
  381. displayName: "Dislikes",
  382. properties: ["dislikes"],
  383. sortProperty: "dislikes",
  384. minWidth: 100,
  385. defaultWidth: 100,
  386. defaultVisibility: "hidden"
  387. },
  388. {
  389. name: "_id",
  390. displayName: "Song ID",
  391. properties: ["_id"],
  392. sortProperty: "_id",
  393. minWidth: 215,
  394. defaultWidth: 215
  395. },
  396. {
  397. name: "youtubeId",
  398. displayName: "YouTube ID",
  399. properties: ["youtubeId"],
  400. sortProperty: "youtubeId",
  401. minWidth: 120,
  402. defaultWidth: 120
  403. },
  404. {
  405. name: "status",
  406. displayName: "Status",
  407. properties: ["status"],
  408. sortProperty: "status",
  409. defaultVisibility: "hidden",
  410. minWidth: 120,
  411. defaultWidth: 120
  412. },
  413. {
  414. name: "thumbnailUrl",
  415. displayName: "Thumbnail (URL)",
  416. properties: ["thumbnail"],
  417. sortProperty: "thumbnail",
  418. defaultVisibility: "hidden"
  419. },
  420. {
  421. name: "duration",
  422. displayName: "Duration",
  423. properties: ["duration"],
  424. sortProperty: "duration",
  425. defaultWidth: 200,
  426. defaultVisibility: "hidden"
  427. },
  428. {
  429. name: "skipDuration",
  430. displayName: "Skip Duration",
  431. properties: ["skipDuration"],
  432. sortProperty: "skipDuration",
  433. defaultWidth: 200,
  434. defaultVisibility: "hidden"
  435. },
  436. {
  437. name: "requestedBy",
  438. displayName: "Requested By",
  439. properties: ["requestedBy"],
  440. sortProperty: "requestedBy",
  441. defaultWidth: 200,
  442. defaultVisibility: "hidden"
  443. },
  444. {
  445. name: "requestedAt",
  446. displayName: "Requested At",
  447. properties: ["requestedAt"],
  448. sortProperty: "requestedAt",
  449. defaultWidth: 200,
  450. defaultVisibility: "hidden"
  451. },
  452. {
  453. name: "verifiedBy",
  454. displayName: "Verified By",
  455. properties: ["verifiedBy"],
  456. sortProperty: "verifiedBy",
  457. defaultWidth: 200,
  458. defaultVisibility: "hidden"
  459. },
  460. {
  461. name: "verifiedAt",
  462. displayName: "Verified At",
  463. properties: ["verifiedAt"],
  464. sortProperty: "verifiedAt",
  465. defaultWidth: 200,
  466. defaultVisibility: "hidden"
  467. }
  468. ],
  469. filters: [
  470. {
  471. name: "_id",
  472. displayName: "Song ID",
  473. property: "_id",
  474. filterTypes: ["exact"],
  475. defaultFilterType: "exact"
  476. },
  477. {
  478. name: "youtubeId",
  479. displayName: "YouTube ID",
  480. property: "youtubeId",
  481. filterTypes: ["contains", "exact", "regex"],
  482. defaultFilterType: "contains"
  483. },
  484. {
  485. name: "title",
  486. displayName: "Title",
  487. property: "title",
  488. filterTypes: ["contains", "exact", "regex"],
  489. defaultFilterType: "contains"
  490. },
  491. {
  492. name: "artists",
  493. displayName: "Artists",
  494. property: "artists",
  495. filterTypes: ["contains", "exact", "regex"],
  496. defaultFilterType: "contains"
  497. },
  498. {
  499. name: "genres",
  500. displayName: "Genres",
  501. property: "genres",
  502. filterTypes: ["contains", "exact", "regex"],
  503. defaultFilterType: "contains",
  504. autocomplete: true,
  505. autocompleteDataAction: "songs.getGenres"
  506. },
  507. {
  508. name: "thumbnail",
  509. displayName: "Thumbnail",
  510. property: "thumbnail",
  511. filterTypes: ["contains", "exact", "regex"],
  512. defaultFilterType: "contains"
  513. },
  514. {
  515. name: "requestedBy",
  516. displayName: "Requested By",
  517. property: "requestedBy",
  518. filterTypes: ["contains", "exact", "regex"],
  519. defaultFilterType: "contains"
  520. },
  521. {
  522. name: "requestedAt",
  523. displayName: "Requested At",
  524. property: "requestedAt",
  525. filterTypes: ["datetimeBefore", "datetimeAfter"],
  526. defaultFilterType: "datetimeBefore"
  527. },
  528. {
  529. name: "verifiedBy",
  530. displayName: "Verified By",
  531. property: "verifiedBy",
  532. filterTypes: ["contains", "exact", "regex"],
  533. defaultFilterType: "contains"
  534. },
  535. {
  536. name: "verifiedAt",
  537. displayName: "Verified At",
  538. property: "verifiedAt",
  539. filterTypes: ["datetimeBefore", "datetimeAfter"],
  540. defaultFilterType: "datetimeBefore"
  541. },
  542. {
  543. name: "status",
  544. displayName: "Status",
  545. property: "status",
  546. filterTypes: ["exact", "regex"],
  547. defaultFilterType: "exact",
  548. filterValues: ["verified", "unverified"]
  549. },
  550. {
  551. name: "likes",
  552. displayName: "Likes",
  553. property: "likes",
  554. filterTypes: [
  555. "numberLesserEqual",
  556. "numberLesser",
  557. "numberGreater",
  558. "numberGreaterEqual",
  559. "numberEquals",
  560. "exact",
  561. "regex"
  562. ],
  563. defaultFilterType: "numberLesser"
  564. },
  565. {
  566. name: "dislikes",
  567. displayName: "Dislikes",
  568. property: "dislikes",
  569. filterTypes: [
  570. "numberLesserEqual",
  571. "numberLesser",
  572. "numberGreater",
  573. "numberGreaterEqual",
  574. "numberEquals",
  575. "exact",
  576. "regex"
  577. ],
  578. defaultFilterType: "numberLesser"
  579. },
  580. {
  581. name: "duration",
  582. displayName: "Duration",
  583. property: "duration",
  584. filterTypes: [
  585. "numberLesserEqual",
  586. "numberLesser",
  587. "numberGreater",
  588. "numberGreaterEqual",
  589. "numberEquals",
  590. "exact",
  591. "regex"
  592. ],
  593. defaultFilterType: "numberLesser"
  594. },
  595. {
  596. name: "skipDuration",
  597. displayName: "Skip Duration",
  598. property: "skipDuration",
  599. filterTypes: [
  600. "numberLesserEqual",
  601. "numberLesser",
  602. "numberGreater",
  603. "numberGreaterEqual",
  604. "numberEquals",
  605. "exact",
  606. "regex"
  607. ],
  608. defaultFilterType: "numberLesser"
  609. }
  610. ],
  611. events: {
  612. adminRoom: "songs",
  613. updated: {
  614. event: "admin.song.updated",
  615. id: "song._id",
  616. item: "song"
  617. },
  618. removed: {
  619. event: "admin.song.removed",
  620. id: "songId"
  621. }
  622. },
  623. jobs: [
  624. {
  625. name: "Update all songs",
  626. socket: "songs.updateAll"
  627. },
  628. {
  629. name: "Recalculate all song ratings",
  630. socket: "songs.recalculateAllRatings"
  631. }
  632. ]
  633. };
  634. },
  635. computed: {
  636. ...mapState("modalVisibility", {
  637. modals: state => state.modals
  638. }),
  639. ...mapState("modals/editSong", {
  640. song: state => state.song
  641. }),
  642. ...mapGetters({
  643. socket: "websockets/getSocket"
  644. })
  645. },
  646. mounted() {
  647. if (this.$route.query.songId) {
  648. this.socket.dispatch(
  649. "songs.getSongFromSongId",
  650. this.$route.query.songId,
  651. res => {
  652. if (res.status === "success")
  653. this.editMany([res.data.song]);
  654. else new Toast("Song with that ID not found");
  655. }
  656. );
  657. }
  658. keyboardShortcuts.registerShortcut(
  659. "songs.toggleKeyboardShortcutsHelper",
  660. {
  661. keyCode: 191, // '/' key
  662. ctrl: true,
  663. preventDefault: true,
  664. handler: () => {
  665. this.toggleKeyboardShortcutsHelper();
  666. }
  667. }
  668. );
  669. keyboardShortcuts.registerShortcut(
  670. "songs.resetKeyboardShortcutsHelper",
  671. {
  672. keyCode: 191, // '/' key
  673. ctrl: true,
  674. shift: true,
  675. preventDefault: true,
  676. handler: () => {
  677. this.resetKeyboardShortcutsHelper();
  678. }
  679. }
  680. );
  681. },
  682. beforeUnmount() {
  683. const shortcutNames = [
  684. "songs.toggleKeyboardShortcutsHelper",
  685. "songs.resetKeyboardShortcutsHelper"
  686. ];
  687. shortcutNames.forEach(shortcutName => {
  688. keyboardShortcuts.unregisterShortcut(shortcutName);
  689. });
  690. },
  691. methods: {
  692. editOne(song) {
  693. this.editSong(song);
  694. this.openModal("editSong");
  695. },
  696. editMany(selectedRows) {
  697. if (selectedRows.length === 1) {
  698. this.editSong(selectedRows[0]);
  699. this.openModal("editSong");
  700. } else {
  701. new Toast("Bulk editing not yet implemented.");
  702. }
  703. },
  704. verifyMany(selectedRows) {
  705. this.socket.dispatch(
  706. "songs.verifyMany",
  707. selectedRows.map(row => row._id),
  708. res => {
  709. new Toast(res.message);
  710. }
  711. );
  712. },
  713. unverifyMany(selectedRows) {
  714. this.socket.dispatch(
  715. "songs.unverifyMany",
  716. selectedRows.map(row => row._id),
  717. res => {
  718. new Toast(res.message);
  719. }
  720. );
  721. },
  722. tagMany() {
  723. new Toast("Bulk tagging not yet implemented.");
  724. },
  725. setArtists() {
  726. new Toast("Bulk setting artists not yet implemented.");
  727. },
  728. setGenres() {
  729. new Toast("Bulk setting genres not yet implemented.");
  730. },
  731. deleteMany(selectedRows) {
  732. this.socket.dispatch(
  733. "songs.removeMany",
  734. selectedRows.map(row => row._id),
  735. res => {
  736. new Toast(res.message);
  737. }
  738. );
  739. },
  740. toggleKeyboardShortcutsHelper() {
  741. this.$refs.keyboardShortcutsHelper.toggleBox();
  742. },
  743. resetKeyboardShortcutsHelper() {
  744. this.$refs.keyboardShortcutsHelper.resetBox();
  745. },
  746. getDateFormatted(createdAt) {
  747. const date = new Date(createdAt);
  748. const year = date.getFullYear();
  749. const month = `${date.getMonth() + 1}`.padStart(2, 0);
  750. const day = `${date.getDate()}`.padStart(2, 0);
  751. const hour = `${date.getHours()}`.padStart(2, 0);
  752. const minute = `${date.getMinutes()}`.padStart(2, 0);
  753. return `${year}-${month}-${day} ${hour}:${minute}`;
  754. },
  755. ...mapActions("modals/editSong", ["editSong"]),
  756. ...mapActions("modalVisibility", ["openModal"])
  757. }
  758. };
  759. </script>
  760. <style lang="scss" scoped>
  761. #keyboardShortcutsHelper {
  762. .box-body {
  763. .biggest {
  764. font-size: 18px;
  765. }
  766. .bigger {
  767. font-size: 16px;
  768. }
  769. span {
  770. display: block;
  771. }
  772. }
  773. }
  774. .song-thumbnail {
  775. display: block;
  776. max-width: 50px;
  777. margin: 0 auto;
  778. }
  779. .bulk-popup .bulk-actions {
  780. .verify-songs-icon {
  781. color: var(--green);
  782. }
  783. .unverify-songs-icon {
  784. color: var(--dark-red);
  785. }
  786. }
  787. </style>