Stations.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483
  1. <template>
  2. <div>
  3. <page-metadata title="Admin | Stations" />
  4. <div class="admin-tab">
  5. <div class="button-row">
  6. <button
  7. class="button is-primary"
  8. @click="
  9. openModal({
  10. modal: 'createStation',
  11. data: { official: true }
  12. })
  13. "
  14. >
  15. Create Station
  16. </button>
  17. <run-job-dropdown :jobs="jobs" />
  18. </div>
  19. <advanced-table
  20. :column-default="columnDefault"
  21. :columns="columns"
  22. :filters="filters"
  23. data-action="stations.getData"
  24. name="admin-stations"
  25. :events="events"
  26. >
  27. <template #column-options="slotProps">
  28. <div class="row-options">
  29. <button
  30. class="button is-primary icon-with-button material-icons"
  31. @click="
  32. openModal({
  33. modal: 'manageStation',
  34. data: {
  35. stationId: slotProps.item._id,
  36. sector: 'admin'
  37. }
  38. })
  39. "
  40. :disabled="slotProps.item.removed"
  41. content="Manage Station"
  42. v-tippy
  43. >
  44. settings
  45. </button>
  46. <quick-confirm
  47. @confirm="remove(slotProps.item._id)"
  48. :disabled="slotProps.item.removed"
  49. >
  50. <button
  51. class="button is-danger icon-with-button material-icons"
  52. content="Remove Station"
  53. v-tippy
  54. >
  55. delete_forever
  56. </button>
  57. </quick-confirm>
  58. <router-link
  59. :to="{ path: `/${slotProps.item.name}` }"
  60. target="_blank"
  61. class="button is-primary icon-with-button material-icons"
  62. :disabled="slotProps.item.removed"
  63. content="View Station"
  64. v-tippy
  65. >
  66. radio
  67. </router-link>
  68. </div>
  69. </template>
  70. <template #column-_id="slotProps">
  71. <span :title="slotProps.item._id">{{
  72. slotProps.item._id
  73. }}</span>
  74. </template>
  75. <template #column-name="slotProps">
  76. <span :title="slotProps.item.name">{{
  77. slotProps.item.name
  78. }}</span>
  79. </template>
  80. <template #column-displayName="slotProps">
  81. <span :title="slotProps.item.displayName">{{
  82. slotProps.item.displayName
  83. }}</span>
  84. </template>
  85. <template #column-type="slotProps">
  86. <span :title="slotProps.item.type">{{
  87. slotProps.item.type
  88. }}</span>
  89. </template>
  90. <template #column-description="slotProps">
  91. <span :title="slotProps.item.description">{{
  92. slotProps.item.description
  93. }}</span>
  94. </template>
  95. <template #column-privacy="slotProps">
  96. <span :title="slotProps.item.privacy">{{
  97. slotProps.item.privacy
  98. }}</span>
  99. </template>
  100. <template #column-owner="slotProps">
  101. <span v-if="slotProps.item.type === 'official'"
  102. >Musare</span
  103. >
  104. <user-id-to-username
  105. v-else
  106. :user-id="slotProps.item.owner"
  107. :link="true"
  108. />
  109. </template>
  110. <template #column-theme="slotProps">
  111. <span :title="slotProps.item.theme">{{
  112. slotProps.item.theme
  113. }}</span>
  114. </template>
  115. <template #column-requestsEnabled="slotProps">
  116. <span :title="slotProps.item.requests.enabled">{{
  117. slotProps.item.requests.enabled
  118. }}</span>
  119. </template>
  120. <template #column-requestsAccess="slotProps">
  121. <span :title="slotProps.item.requests.access">{{
  122. slotProps.item.requests.access
  123. }}</span>
  124. </template>
  125. <template #column-requestsLimit="slotProps">
  126. <span :title="slotProps.item.requests.limit">{{
  127. slotProps.item.requests.limit
  128. }}</span>
  129. </template>
  130. <template #column-autofillEnabled="slotProps">
  131. <span :title="slotProps.item.autofill.enabled">{{
  132. slotProps.item.autofill.enabled
  133. }}</span>
  134. </template>
  135. <template #column-autofillLimit="slotProps">
  136. <span :title="slotProps.item.autofill.limit">{{
  137. slotProps.item.autofill.limit
  138. }}</span>
  139. </template>
  140. <template #column-autofillMode="slotProps">
  141. <span :title="slotProps.item.autofill.mode">{{
  142. slotProps.item.autofill.mode
  143. }}</span>
  144. </template>
  145. </advanced-table>
  146. </div>
  147. <create-playlist v-if="modals.createPlaylist" />
  148. <edit-song v-if="modals.editSong" song-type="songs" sector="admin" />
  149. <report v-if="modals.report" />
  150. </div>
  151. </template>
  152. <script>
  153. import { mapState, mapActions, mapGetters } from "vuex";
  154. import { defineAsyncComponent } from "vue";
  155. import Toast from "toasters";
  156. import AdvancedTable from "@/components/AdvancedTable.vue";
  157. import RunJobDropdown from "@/components/RunJobDropdown.vue";
  158. export default {
  159. components: {
  160. CreatePlaylist: defineAsyncComponent(() =>
  161. import("@/components/modals/CreatePlaylist.vue")
  162. ),
  163. Report: defineAsyncComponent(() =>
  164. import("@/components/modals/Report.vue")
  165. ),
  166. EditSong: defineAsyncComponent(() =>
  167. import("@/components/modals/EditSong")
  168. ),
  169. AdvancedTable,
  170. RunJobDropdown
  171. },
  172. data() {
  173. return {
  174. editingStationId: "",
  175. columnDefault: {
  176. sortable: true,
  177. hidable: true,
  178. defaultVisibility: "shown",
  179. draggable: true,
  180. resizable: true,
  181. minWidth: 150,
  182. maxWidth: 600
  183. },
  184. columns: [
  185. {
  186. name: "options",
  187. displayName: "Options",
  188. properties: ["_id", "name"],
  189. sortable: false,
  190. hidable: false,
  191. resizable: false,
  192. minWidth: 129,
  193. defaultWidth: 129
  194. },
  195. {
  196. name: "_id",
  197. displayName: "Station ID",
  198. properties: ["_id"],
  199. sortProperty: "_id",
  200. minWidth: 230,
  201. defaultWidth: 230
  202. },
  203. {
  204. name: "name",
  205. displayName: "Name",
  206. properties: ["name"],
  207. sortProperty: "name"
  208. },
  209. {
  210. name: "displayName",
  211. displayName: "Display Name",
  212. properties: ["displayName"],
  213. sortProperty: "displayName"
  214. },
  215. {
  216. name: "description",
  217. displayName: "Description",
  218. properties: ["description"],
  219. sortProperty: "description",
  220. defaultVisibility: "hidden"
  221. },
  222. {
  223. name: "type",
  224. displayName: "Type",
  225. properties: ["type"],
  226. sortProperty: "type"
  227. },
  228. {
  229. name: "privacy",
  230. displayName: "Privacy",
  231. properties: ["privacy"],
  232. sortProperty: "privacy"
  233. },
  234. {
  235. name: "owner",
  236. displayName: "Owner",
  237. properties: ["owner", "type"],
  238. sortProperty: "owner",
  239. defaultWidth: 150
  240. },
  241. {
  242. name: "theme",
  243. displayName: "Theme",
  244. properties: ["theme"],
  245. sortProperty: "theme",
  246. defaultVisibility: "hidden"
  247. },
  248. {
  249. name: "requestsEnabled",
  250. displayName: "Requests Enabled",
  251. properties: ["requests.enabled"],
  252. sortProperty: "requests.enabled",
  253. minWidth: 180,
  254. defaultWidth: 180,
  255. defaultVisibility: "hidden"
  256. },
  257. {
  258. name: "requestsAccess",
  259. displayName: "Requests Access",
  260. properties: ["requests.access"],
  261. sortProperty: "requests.access",
  262. minWidth: 180,
  263. defaultWidth: 180,
  264. defaultVisibility: "hidden"
  265. },
  266. {
  267. name: "requestsLimit",
  268. displayName: "Requests Limit",
  269. properties: ["requests.limit"],
  270. sortProperty: "requests.limit",
  271. minWidth: 180,
  272. defaultWidth: 180,
  273. defaultVisibility: "hidden"
  274. },
  275. {
  276. name: "autofillEnabled",
  277. displayName: "Autofill Enabled",
  278. properties: ["autofill.enabled"],
  279. sortProperty: "autofill.enabled",
  280. minWidth: 180,
  281. defaultWidth: 180,
  282. defaultVisibility: "hidden"
  283. },
  284. {
  285. name: "autofillLimit",
  286. displayName: "Autofill Limit",
  287. properties: ["autofill.limit"],
  288. sortProperty: "autofill.limit",
  289. minWidth: 180,
  290. defaultWidth: 180,
  291. defaultVisibility: "hidden"
  292. },
  293. {
  294. name: "autofillMode",
  295. displayName: "Autofill Mode",
  296. properties: ["autofill.mode"],
  297. sortProperty: "autofill.mode",
  298. minWidth: 180,
  299. defaultWidth: 180,
  300. defaultVisibility: "hidden"
  301. }
  302. ],
  303. filters: [
  304. {
  305. name: "_id",
  306. displayName: "Station ID",
  307. property: "_id",
  308. filterTypes: ["exact"],
  309. defaultFilterType: "exact"
  310. },
  311. {
  312. name: "name",
  313. displayName: "Name",
  314. property: "name",
  315. filterTypes: ["contains", "exact", "regex"],
  316. defaultFilterType: "contains"
  317. },
  318. {
  319. name: "displayName",
  320. displayName: "Display Name",
  321. property: "displayName",
  322. filterTypes: ["contains", "exact", "regex"],
  323. defaultFilterType: "contains"
  324. },
  325. {
  326. name: "description",
  327. displayName: "Description",
  328. property: "description",
  329. filterTypes: ["contains", "exact", "regex"],
  330. defaultFilterType: "contains"
  331. },
  332. {
  333. name: "type",
  334. displayName: "Type",
  335. property: "type",
  336. filterTypes: ["exact"],
  337. defaultFilterType: "exact",
  338. dropdown: [
  339. ["official", "Official"],
  340. ["community", "Community"]
  341. ]
  342. },
  343. {
  344. name: "privacy",
  345. displayName: "Privacy",
  346. property: "privacy",
  347. filterTypes: ["exact"],
  348. defaultFilterType: "exact",
  349. dropdown: [
  350. ["public", "Public"],
  351. ["unlisted", "Unlisted"],
  352. ["private", "Private"]
  353. ]
  354. },
  355. {
  356. name: "owner",
  357. displayName: "Owner",
  358. property: "owner",
  359. filterTypes: ["contains", "exact", "regex"],
  360. defaultFilterType: "contains"
  361. },
  362. {
  363. name: "theme",
  364. displayName: "Theme",
  365. property: "theme",
  366. filterTypes: ["exact"],
  367. defaultFilterType: "exact",
  368. dropdown: [
  369. ["blue", "Blue"],
  370. ["purple", "Purple"],
  371. ["teal", "Teal"],
  372. ["orange", "Orange"],
  373. ["red", "Red"]
  374. ]
  375. },
  376. {
  377. name: "requestsEnabled",
  378. displayName: "Requests Enabled",
  379. property: "requests.enabled",
  380. filterTypes: ["boolean"],
  381. defaultFilterType: "boolean"
  382. },
  383. {
  384. name: "requestsAccess",
  385. displayName: "Requests Access",
  386. property: "requests.access",
  387. filterTypes: ["exact"],
  388. defaultFilterType: "exact",
  389. dropdown: [
  390. ["owner", "Owner"],
  391. ["user", "User"]
  392. ]
  393. },
  394. {
  395. name: "requestsLimit",
  396. displayName: "Requests Limit",
  397. property: "requests.limit",
  398. filterTypes: [
  399. "numberLesserEqual",
  400. "numberLesser",
  401. "numberGreater",
  402. "numberGreaterEqual",
  403. "numberEquals"
  404. ],
  405. defaultFilterType: "numberLesser"
  406. },
  407. {
  408. name: "autofillEnabled",
  409. displayName: "Autofill Enabled",
  410. property: "autofill.enabled",
  411. filterTypes: ["boolean"],
  412. defaultFilterType: "boolean"
  413. },
  414. {
  415. name: "autofillLimit",
  416. displayName: "Autofill Limit",
  417. property: "autofill.limit",
  418. filterTypes: [
  419. "numberLesserEqual",
  420. "numberLesser",
  421. "numberGreater",
  422. "numberGreaterEqual",
  423. "numberEquals"
  424. ],
  425. defaultFilterType: "numberLesser"
  426. },
  427. {
  428. name: "autofillMode",
  429. displayName: "Autofill Mode",
  430. property: "autofill.mode",
  431. filterTypes: ["exact"],
  432. defaultFilterType: "exact",
  433. dropdown: [
  434. ["random", "Random"],
  435. ["sequential", "Sequential"]
  436. ]
  437. }
  438. ],
  439. events: {
  440. adminRoom: "stations",
  441. updated: {
  442. event: "station.updated",
  443. id: "station._id",
  444. item: "station"
  445. },
  446. removed: {
  447. event: "admin.station.deleted",
  448. id: "stationId"
  449. }
  450. },
  451. jobs: [
  452. {
  453. name: "Clear every station queue",
  454. socket: "stations.clearEveryStationQueue"
  455. }
  456. ]
  457. };
  458. },
  459. computed: {
  460. ...mapState("modalVisibility", {
  461. modals: state => state.modals
  462. }),
  463. ...mapGetters({
  464. socket: "websockets/getSocket"
  465. })
  466. },
  467. methods: {
  468. remove(stationId) {
  469. this.socket.dispatch(
  470. "stations.remove",
  471. stationId,
  472. res => new Toast(res.message)
  473. );
  474. },
  475. ...mapActions("modalVisibility", ["openModal"])
  476. }
  477. };
  478. </script>