Stations.vue 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. <template>
  2. <div>
  3. <page-metadata title="Admin | Stations" />
  4. <div class="container">
  5. <div class="button-row">
  6. <button
  7. class="button is-primary"
  8. @click="openModal('createStation')"
  9. >
  10. Create Station
  11. </button>
  12. <run-job-dropdown :jobs="jobs" />
  13. </div>
  14. <table class="table">
  15. <thead>
  16. <tr>
  17. <td>ID</td>
  18. <td>Name</td>
  19. <td>Type</td>
  20. <td>Display Name</td>
  21. <td>Description</td>
  22. <td>Owner</td>
  23. <td>Options</td>
  24. </tr>
  25. </thead>
  26. <tbody>
  27. <tr v-for="(station, index) in stations" :key="station._id">
  28. <td>
  29. <span>{{ station._id }}</span>
  30. </td>
  31. <td>
  32. <span>
  33. <router-link
  34. :to="{
  35. name: 'station',
  36. params: { id: station.name }
  37. }"
  38. >
  39. {{ station.name }}
  40. </router-link>
  41. </span>
  42. </td>
  43. <td>
  44. <span>{{ station.type }}</span>
  45. </td>
  46. <td>
  47. <span>{{ station.displayName }}</span>
  48. </td>
  49. <td>
  50. <span>{{ station.description }}</span>
  51. </td>
  52. <td>
  53. <span
  54. v-if="station.type === 'official'"
  55. title="Musare"
  56. >Musare</span
  57. >
  58. <user-id-to-username
  59. v-else
  60. :user-id="station.owner"
  61. :link="true"
  62. />
  63. </td>
  64. <td>
  65. <a class="button is-info" @click="manage(station)"
  66. >Manage</a
  67. >
  68. <confirm @confirm="removeStation(index)">
  69. <a class="button is-danger">Remove</a>
  70. </confirm>
  71. </td>
  72. </tr>
  73. </tbody>
  74. </table>
  75. </div>
  76. <request-song v-if="modals.requestSong" />
  77. <create-playlist v-if="modals.createPlaylist" />
  78. <manage-station
  79. v-if="modals.manageStation"
  80. :station-id="editingStationId"
  81. sector="admin"
  82. />
  83. <edit-playlist v-if="modals.editPlaylist" />
  84. <edit-song v-if="modals.editSong" song-type="songs" sector="admin" />
  85. <report v-if="modals.report" />
  86. <create-station v-if="modals.createStation" :official="true" />
  87. </div>
  88. </template>
  89. <script>
  90. import { mapState, mapActions, mapGetters } from "vuex";
  91. import { defineAsyncComponent } from "vue";
  92. import Toast from "toasters";
  93. import UserIdToUsername from "@/components/UserIdToUsername.vue";
  94. import Confirm from "@/components/Confirm.vue";
  95. import RunJobDropdown from "@/components/RunJobDropdown.vue";
  96. import ws from "@/ws";
  97. export default {
  98. components: {
  99. RequestSong: defineAsyncComponent(() =>
  100. import("@/components/modals/RequestSong.vue")
  101. ),
  102. EditPlaylist: defineAsyncComponent(() =>
  103. import("@/components/modals/EditPlaylist")
  104. ),
  105. CreatePlaylist: defineAsyncComponent(() =>
  106. import("@/components/modals/CreatePlaylist.vue")
  107. ),
  108. ManageStation: defineAsyncComponent(() =>
  109. import("@/components/modals/ManageStation/index.vue")
  110. ),
  111. Report: defineAsyncComponent(() =>
  112. import("@/components/modals/Report.vue")
  113. ),
  114. EditSong: defineAsyncComponent(() =>
  115. import("@/components/modals/EditSong")
  116. ),
  117. CreateStation: defineAsyncComponent(() =>
  118. import("@/components/modals/CreateStation.vue")
  119. ),
  120. UserIdToUsername,
  121. Confirm,
  122. RunJobDropdown
  123. },
  124. data() {
  125. return {
  126. editingStationId: "",
  127. jobs: [
  128. {
  129. name: "Clear every station queue",
  130. socket: "stations.clearEveryStationQueue"
  131. }
  132. ]
  133. };
  134. },
  135. computed: {
  136. ...mapState("admin/stations", {
  137. stations: state => state.stations
  138. }),
  139. ...mapState("modalVisibility", {
  140. modals: state => state.modals
  141. }),
  142. ...mapGetters({
  143. socket: "websockets/getSocket"
  144. })
  145. },
  146. mounted() {
  147. ws.onConnect(this.init);
  148. this.socket.on("event:admin.station.created", res =>
  149. this.stationAdded(res.data.station)
  150. );
  151. this.socket.on("event:admin.station.deleted", res =>
  152. this.stationRemoved(res.data.stationId)
  153. );
  154. },
  155. methods: {
  156. removeStation(index) {
  157. this.socket.dispatch(
  158. "stations.remove",
  159. this.stations[index]._id,
  160. res => new Toast(res.message)
  161. );
  162. },
  163. manage(station) {
  164. this.editingStationId = station._id;
  165. this.openModal("manageStation");
  166. },
  167. init() {
  168. this.socket.dispatch("stations.index", res => {
  169. if (res.status === "success")
  170. this.loadStations(res.data.stations);
  171. });
  172. this.socket.dispatch("apis.joinAdminRoom", "stations", () => {});
  173. },
  174. ...mapActions("modalVisibility", ["openModal"]),
  175. ...mapActions("admin/stations", [
  176. "manageStation",
  177. "loadStations",
  178. "stationRemoved",
  179. "stationAdded"
  180. ])
  181. }
  182. };
  183. </script>
  184. <style lang="scss" scoped>
  185. .night-mode {
  186. .table {
  187. color: var(--light-grey-2);
  188. background-color: var(--dark-grey-3);
  189. thead tr {
  190. background: var(--dark-grey-3);
  191. td {
  192. color: var(--white);
  193. }
  194. }
  195. tbody tr:hover {
  196. background-color: var(--dark-grey-4) !important;
  197. }
  198. tbody tr:nth-child(even) {
  199. background-color: var(--dark-grey-2);
  200. }
  201. strong {
  202. color: var(--light-grey-2);
  203. }
  204. }
  205. }
  206. td {
  207. word-wrap: break-word;
  208. max-width: 10vw;
  209. vertical-align: middle;
  210. & > div {
  211. display: inline-flex;
  212. }
  213. }
  214. .is-info:focus {
  215. background-color: var(--primary-color);
  216. }
  217. </style>