Songs.vue 20 KB

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