Playlists.vue 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. <template>
  2. <div>
  3. <page-metadata title="Admin | Playlists" />
  4. <div class="admin-tab">
  5. <div class="button-row">
  6. <run-job-dropdown :jobs="jobs" />
  7. </div>
  8. <advanced-table
  9. :column-default="columnDefault"
  10. :columns="columns"
  11. :filters="filters"
  12. data-action="playlists.getData"
  13. name="admin-playlists"
  14. :events="events"
  15. >
  16. <template #column-options="slotProps">
  17. <div class="row-options">
  18. <button
  19. class="
  20. button
  21. is-primary
  22. icon-with-button
  23. material-icons
  24. "
  25. @click="edit(slotProps.item._id)"
  26. :disabled="slotProps.item.removed"
  27. content="Edit Playlist"
  28. v-tippy
  29. >
  30. edit
  31. </button>
  32. </div>
  33. </template>
  34. <template #column-displayName="slotProps">
  35. <span :title="slotProps.item.displayName">{{
  36. slotProps.item.displayName
  37. }}</span>
  38. </template>
  39. <template #column-type="slotProps">
  40. <span :title="slotProps.item.type">{{
  41. slotProps.item.type
  42. }}</span>
  43. </template>
  44. <template #column-privacy="slotProps">
  45. <span :title="slotProps.item.privacy">{{
  46. slotProps.item.privacy
  47. }}</span>
  48. </template>
  49. <template #column-songsCount="slotProps">
  50. <span :title="slotProps.item.songs.length">{{
  51. slotProps.item.songs.length
  52. }}</span>
  53. </template>
  54. <template #column-totalLength="slotProps">
  55. <span
  56. :title="totalLengthForPlaylist(slotProps.item.songs)"
  57. >{{
  58. totalLengthForPlaylist(slotProps.item.songs)
  59. }}</span
  60. >
  61. </template>
  62. <template #column-createdBy="slotProps">
  63. <span v-if="slotProps.item.createdBy === 'Musare'"
  64. >Musare</span
  65. >
  66. <user-id-to-username
  67. v-else
  68. :user-id="slotProps.item.createdBy"
  69. :link="true"
  70. />
  71. </template>
  72. <template #column-createdAt="slotProps">
  73. <span :title="new Date(slotProps.item.createdAt)">{{
  74. getDateFormatted(slotProps.item.createdAt)
  75. }}</span>
  76. </template>
  77. <template #column-createdFor="slotProps">
  78. <span :title="slotProps.item.createdFor">{{
  79. slotProps.item.createdFor
  80. }}</span>
  81. </template>
  82. <template #column-_id="slotProps">
  83. <span :title="slotProps.item._id">{{
  84. slotProps.item._id
  85. }}</span>
  86. </template>
  87. </advanced-table>
  88. </div>
  89. <edit-playlist v-if="modals.editPlaylist" sector="admin" />
  90. <edit-song v-if="modals.editSong" song-type="songs" />
  91. <report v-if="modals.report" />
  92. </div>
  93. </template>
  94. <script>
  95. import { mapState, mapActions, mapGetters } from "vuex";
  96. import { defineAsyncComponent } from "vue";
  97. import AdvancedTable from "@/components/AdvancedTable.vue";
  98. import RunJobDropdown from "@/components/RunJobDropdown.vue";
  99. import UserIdToUsername from "@/components/UserIdToUsername.vue";
  100. import utils from "../../../../js/utils";
  101. export default {
  102. components: {
  103. EditPlaylist: defineAsyncComponent(() =>
  104. import("@/components/modals/EditPlaylist")
  105. ),
  106. Report: defineAsyncComponent(() =>
  107. import("@/components/modals/Report.vue")
  108. ),
  109. EditSong: defineAsyncComponent(() =>
  110. import("@/components/modals/EditSong")
  111. ),
  112. AdvancedTable,
  113. RunJobDropdown,
  114. UserIdToUsername
  115. },
  116. data() {
  117. return {
  118. utils,
  119. columnDefault: {
  120. sortable: true,
  121. hidable: true,
  122. defaultVisibility: "shown",
  123. draggable: true,
  124. resizable: true,
  125. minWidth: 150,
  126. maxWidth: 600
  127. },
  128. columns: [
  129. {
  130. name: "options",
  131. displayName: "Edit",
  132. properties: ["_id"],
  133. sortable: false,
  134. hidable: false,
  135. resizable: false,
  136. minWidth: 51,
  137. defaultWidth: 51
  138. },
  139. {
  140. name: "displayName",
  141. displayName: "Display Name",
  142. properties: ["displayName"],
  143. sortProperty: "displayName"
  144. },
  145. {
  146. name: "type",
  147. displayName: "Type",
  148. properties: ["type"],
  149. sortProperty: "type"
  150. },
  151. {
  152. name: "privacy",
  153. displayName: "Privacy",
  154. properties: ["privacy"],
  155. sortProperty: "privacy"
  156. },
  157. {
  158. name: "songsCount",
  159. displayName: "Songs #",
  160. properties: ["songs"],
  161. sortable: false,
  162. minWidth: 80,
  163. defaultWidth: 80
  164. },
  165. {
  166. name: "totalLength",
  167. displayName: "Total Length",
  168. properties: ["songs"],
  169. sortable: false,
  170. minWidth: 250,
  171. defaultWidth: 250
  172. },
  173. {
  174. name: "createdBy",
  175. displayName: "Created By",
  176. properties: ["createdBy"],
  177. sortProperty: "createdBy",
  178. defaultWidth: 150
  179. },
  180. {
  181. name: "createdAt",
  182. displayName: "Created At",
  183. properties: ["createdAt"],
  184. sortProperty: "createdAt",
  185. defaultWidth: 150
  186. },
  187. {
  188. name: "createdFor",
  189. displayName: "Created For",
  190. properties: ["createdFor"],
  191. sortProperty: "createdFor",
  192. minWidth: 230,
  193. defaultWidth: 230
  194. },
  195. {
  196. name: "_id",
  197. displayName: "Playlist ID",
  198. properties: ["_id"],
  199. sortProperty: "_id",
  200. minWidth: 230,
  201. defaultWidth: 230
  202. }
  203. ],
  204. filters: [
  205. {
  206. name: "_id",
  207. displayName: "Playlist ID",
  208. property: "_id",
  209. filterTypes: ["exact"],
  210. defaultFilterType: "exact"
  211. },
  212. {
  213. name: "displayName",
  214. displayName: "Display Name",
  215. property: "displayName",
  216. filterTypes: ["contains", "exact", "regex"],
  217. defaultFilterType: "contains"
  218. },
  219. {
  220. name: "type",
  221. displayName: "Type",
  222. property: "type",
  223. filterTypes: ["contains", "exact", "regex"],
  224. defaultFilterType: "contains"
  225. },
  226. {
  227. name: "privacy",
  228. displayName: "Privacy",
  229. property: "privacy",
  230. filterTypes: ["array", "regex"],
  231. defaultFilterType: "array",
  232. filterValues: ["public", "private"]
  233. },
  234. {
  235. name: "createdBy",
  236. displayName: "Created By",
  237. property: "createdBy",
  238. filterTypes: ["contains", "exact", "regex"],
  239. defaultFilterType: "contains"
  240. },
  241. {
  242. name: "createdAt",
  243. displayName: "Created At",
  244. property: "createdAt",
  245. filterTypes: ["datetimeBefore", "datetimeAfter"],
  246. defaultFilterType: "datetimeBefore"
  247. },
  248. {
  249. name: "createdFor",
  250. displayName: "Created For",
  251. property: "createdFor",
  252. filterTypes: ["contains", "exact", "regex"],
  253. defaultFilterType: "contains"
  254. }
  255. ],
  256. events: {
  257. adminRoom: "playlists",
  258. removed: {
  259. event: "admin.playlist.deleted",
  260. id: "playlistId"
  261. }
  262. },
  263. jobs: [
  264. {
  265. name: "Delete orphaned station playlists",
  266. socket: "playlists.deleteOrphanedStationPlaylists"
  267. },
  268. {
  269. name: "Delete orphaned genre playlists",
  270. socket: "playlists.deleteOrphanedGenrePlaylists"
  271. },
  272. {
  273. name: "Request orphaned playlist songs",
  274. socket: "playlists.requestOrphanedPlaylistSongs"
  275. },
  276. {
  277. name: "Clear and refill all station playlists",
  278. socket: "playlists.clearAndRefillAllStationPlaylists"
  279. },
  280. {
  281. name: "Clear and refill all genre playlists",
  282. socket: "playlists.clearAndRefillAllGenrePlaylists"
  283. },
  284. {
  285. name: "Create missing genre playlists",
  286. socket: "playlists.createMissingGenrePlaylists"
  287. }
  288. ]
  289. };
  290. },
  291. computed: {
  292. ...mapState("modalVisibility", {
  293. modals: state => state.modals
  294. }),
  295. ...mapGetters({
  296. socket: "websockets/getSocket"
  297. })
  298. },
  299. mounted() {
  300. // TODO
  301. // this.socket.on("event:admin.playlist.song.added", res =>
  302. // this.addPlaylistSong({
  303. // playlistId: res.data.playlistId,
  304. // song: res.data.song
  305. // })
  306. // );
  307. // this.socket.on("event:admin.playlist.song.removed", res =>
  308. // this.removePlaylistSong({
  309. // playlistId: res.data.playlistId,
  310. // youtubeId: res.data.youtubeId
  311. // })
  312. // );
  313. // this.socket.on("event:admin.playlist.displayName.updated", res =>
  314. // this.updatePlaylistDisplayName({
  315. // playlistId: res.data.playlistId,
  316. // displayName: res.data.displayName
  317. // })
  318. // );
  319. // this.socket.on("event:admin.playlist.privacy.updated", res =>
  320. // this.updatePlaylistPrivacy({
  321. // playlistId: res.data.playlistId,
  322. // privacy: res.data.privacy
  323. // })
  324. // );
  325. },
  326. methods: {
  327. edit(playlistId) {
  328. this.editPlaylist(playlistId);
  329. this.openModal("editPlaylist");
  330. },
  331. getDateFormatted(createdAt) {
  332. const date = new Date(createdAt);
  333. const year = date.getFullYear();
  334. const month = `${date.getMonth() + 1}`.padStart(2, 0);
  335. const day = `${date.getDate()}`.padStart(2, 0);
  336. const hour = `${date.getHours()}`.padStart(2, 0);
  337. const minute = `${date.getMinutes()}`.padStart(2, 0);
  338. return `${year}-${month}-${day} ${hour}:${minute}`;
  339. },
  340. totalLengthForPlaylist(songs) {
  341. let length = 0;
  342. songs.forEach(song => {
  343. length += song.duration;
  344. });
  345. return this.utils.formatTimeLong(length);
  346. },
  347. ...mapActions("modalVisibility", ["openModal"]),
  348. ...mapActions("user/playlists", ["editPlaylist"])
  349. }
  350. };
  351. </script>