Playlists.vue 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. <template>
  2. <div>
  3. <metadata title="Admin | Playlists" />
  4. <div class="container">
  5. <button
  6. class="button is-primary"
  7. @click="deleteOrphanedStationPlaylists()"
  8. >
  9. Delete orphaned station playlists
  10. </button>
  11. <button
  12. class="button is-primary"
  13. @click="deleteOrphanedGenrePlaylists()"
  14. >
  15. Delete orphaned genre playlists
  16. </button>
  17. <button
  18. class="button is-primary"
  19. @click="requestOrphanedPlaylistSongs()"
  20. >
  21. Request orphaned playlist songs
  22. </button>
  23. <br />
  24. <br />
  25. <table class="table is-striped">
  26. <thead>
  27. <tr>
  28. <td>Display name</td>
  29. <td>Type</td>
  30. <td>Is user modifiable</td>
  31. <td>Songs #</td>
  32. <td>Playlist length</td>
  33. <td>Created by</td>
  34. <td>Created at</td>
  35. <td>Created for</td>
  36. <td>Playlist id</td>
  37. <td>Options</td>
  38. </tr>
  39. </thead>
  40. <tbody>
  41. <tr v-for="playlist in playlists" :key="playlist._id">
  42. <td>{{ playlist.displayName }}</td>
  43. <td>{{ playlist.type }}</td>
  44. <td>{{ playlist.isUserModifiable }}</td>
  45. <td>{{ playlist.songs.length }}</td>
  46. <td>{{ totalLengthForPlaylist(playlist.songs) }}</td>
  47. <td v-if="playlist.createdBy === 'Musare'">Musare</td>
  48. <td v-else>
  49. <user-id-to-username
  50. :user-id="playlist.createdBy"
  51. :link="true"
  52. />
  53. </td>
  54. <td :title="new Date(playlist.createdAt)">
  55. {{ getDateFormatted(playlist.createdAt) }}
  56. </td>
  57. <td>{{ playlist.createdFor }}</td>
  58. <td>{{ playlist._id }}</td>
  59. <td>
  60. <button
  61. class="button is-primary"
  62. @click="edit(playlist._id)"
  63. >
  64. View
  65. </button>
  66. </td>
  67. </tr>
  68. </tbody>
  69. </table>
  70. </div>
  71. <edit-playlist v-if="modals.admin.editPlaylist" sector="admin" />
  72. <report v-if="modals.station.report" />
  73. <edit-song v-if="modals.admin.editSong" song-type="songs" />
  74. </div>
  75. </template>
  76. <script>
  77. import { mapState, mapActions, mapGetters } from "vuex";
  78. import Toast from "toasters";
  79. import UserIdToUsername from "../../../components/common/UserIdToUsername.vue";
  80. import ws from "../../../ws";
  81. import utils from "../../../../js/utils";
  82. export default {
  83. components: {
  84. EditPlaylist: () =>
  85. import("../../../components/modals/EditPlaylist/index.vue"),
  86. UserIdToUsername,
  87. Report: () => import("../../../components/modals/Report.vue"),
  88. EditSong: () => import("../../../components/modals/EditSong.vue")
  89. },
  90. data() {
  91. return {
  92. utils,
  93. playlists: []
  94. };
  95. },
  96. computed: {
  97. ...mapState("modalVisibility", {
  98. modals: state => state.modals
  99. }),
  100. ...mapGetters({
  101. socket: "websockets/getSocket"
  102. })
  103. },
  104. mounted() {
  105. if (this.socket.readyState === 1) this.init();
  106. ws.onConnect(() => this.init());
  107. },
  108. methods: {
  109. edit(playlistId) {
  110. this.editPlaylist(playlistId);
  111. this.openModal({ sector: "admin", modal: "editPlaylist" });
  112. },
  113. init() {
  114. this.socket.dispatch("playlists.index", res => {
  115. if (res.status === "success") {
  116. this.playlists = res.data;
  117. // if (this.$route.query.userId) {
  118. // const user = this.users.find(
  119. // user => user._id === this.$route.query.userId
  120. // );
  121. // if (user) this.edit(user);
  122. // }
  123. }
  124. });
  125. this.socket.dispatch("apis.joinAdminRoom", "playlists", () => {});
  126. },
  127. getDateFormatted(createdAt) {
  128. const date = new Date(createdAt);
  129. const year = date.getFullYear();
  130. const month = `${date.getMonth() + 1}`.padStart(2, 0);
  131. const day = `${date.getDate()}`.padStart(2, 0);
  132. const hour = `${date.getHours()}`.padStart(2, 0);
  133. const minute = `${date.getMinutes()}`.padStart(2, 0);
  134. return `${year}-${month}-${day} ${hour}:${minute}`;
  135. },
  136. totalLengthForPlaylist(songs) {
  137. let length = 0;
  138. songs.forEach(song => {
  139. length += song.duration;
  140. });
  141. return this.utils.formatTimeLong(length);
  142. },
  143. deleteOrphanedStationPlaylists() {
  144. this.socket.dispatch(
  145. "playlists.deleteOrphanedStationPlaylists",
  146. res => {
  147. if (res.status === "success") {
  148. new Toast({
  149. content: `${res.message}`,
  150. timeout: 4000
  151. });
  152. } else {
  153. new Toast({
  154. content: `Error: ${res.message}`,
  155. timeout: 8000
  156. });
  157. }
  158. }
  159. );
  160. },
  161. deleteOrphanedGenrePlaylists() {
  162. this.socket.dispatch(
  163. "playlists.deleteOrphanedGenrePlaylists",
  164. res => {
  165. if (res.status === "success") {
  166. new Toast({
  167. content: `${res.message}`,
  168. timeout: 4000
  169. });
  170. } else {
  171. new Toast({
  172. content: `Error: ${res.message}`,
  173. timeout: 8000
  174. });
  175. }
  176. }
  177. );
  178. },
  179. requestOrphanedPlaylistSongs() {
  180. this.socket.dispatch(
  181. "playlists.requestOrphanedPlaylistSongs",
  182. res => {
  183. if (res.status === "success") {
  184. new Toast({
  185. content: `${res.message}`,
  186. timeout: 4000
  187. });
  188. } else {
  189. new Toast({
  190. content: `Error: ${res.message}`,
  191. timeout: 8000
  192. });
  193. }
  194. }
  195. );
  196. },
  197. ...mapActions("modalVisibility", ["openModal"]),
  198. ...mapActions("user/playlists", ["editPlaylist"])
  199. }
  200. };
  201. </script>
  202. <style lang="scss" scoped>
  203. .night-mode {
  204. .table {
  205. color: var(--light-grey-2);
  206. background-color: var(--dark-grey-3);
  207. thead tr {
  208. background: var(--dark-grey-3);
  209. td {
  210. color: var(--white);
  211. }
  212. }
  213. tbody tr:hover {
  214. background-color: var(--dark-grey-4) !important;
  215. }
  216. tbody tr:nth-child(even) {
  217. background-color: var(--dark-grey-2);
  218. }
  219. strong {
  220. color: var(--light-grey-2);
  221. }
  222. }
  223. }
  224. body {
  225. font-family: "Hind", sans-serif;
  226. }
  227. td {
  228. vertical-align: middle;
  229. }
  230. .is-primary:focus {
  231. background-color: var(--primary-color) !important;
  232. }
  233. </style>