Queue.vue 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. <template>
  2. <div id="queue">
  3. <draggable
  4. :class="{
  5. 'actionable-button-hidden': !actionableButtonVisible,
  6. 'scrollable-list': true
  7. }"
  8. v-if="queue.length > 0"
  9. v-model="queue"
  10. v-bind="dragOptions"
  11. @start="drag = true"
  12. @end="drag = false"
  13. @change="repositionSongInQueue"
  14. >
  15. <transition-group
  16. type="transition"
  17. :name="!drag ? 'draggable-list-transition' : null"
  18. >
  19. <song-item
  20. v-for="(song, index) in queue"
  21. :key="index + song.songId"
  22. :song="song"
  23. :requested-by="
  24. station.type === 'community' &&
  25. station.partyMode === true
  26. "
  27. :class="{
  28. 'item-draggable': isAdminOnly() || isOwnerOnly()
  29. }"
  30. >
  31. <div
  32. v-if="isAdminOnly() || isOwnerOnly()"
  33. class="song-actions"
  34. slot="actions"
  35. >
  36. <confirm
  37. v-if="isOwnerOnly() || isAdminOnly()"
  38. @confirm="removeFromQueue(song.songId)"
  39. >
  40. <i
  41. class="material-icons delete-icon"
  42. content="Remove Song from Queue"
  43. v-tippy
  44. >delete_forever</i
  45. >
  46. </confirm>
  47. <i
  48. class="material-icons"
  49. v-if="index > 0"
  50. @click="moveSongToTop(song, index)"
  51. content="Move to top of Queue"
  52. v-tippy
  53. >vertical_align_top</i
  54. >
  55. <i
  56. v-if="queue.length - 1 !== index"
  57. @click="moveSongToBottom(song, index)"
  58. class="material-icons"
  59. content="Move to bottom of Queue"
  60. v-tippy
  61. >vertical_align_bottom</i
  62. >
  63. </div>
  64. </song-item>
  65. </transition-group>
  66. </draggable>
  67. <p class="nothing-here-text" v-else>
  68. There are no songs currently queued
  69. </p>
  70. <button
  71. class="button is-primary tab-actionable-button"
  72. v-if="
  73. loggedIn &&
  74. station.type === 'community' &&
  75. station.partyMode &&
  76. ((station.locked && isOwnerOnly()) ||
  77. !station.locked ||
  78. (station.locked && isAdminOnly() && dismissedWarning))
  79. "
  80. @click="
  81. openModal({
  82. sector: 'station',
  83. modal: 'addSongToQueue'
  84. })
  85. "
  86. >
  87. <i class="material-icons icon-with-button">queue</i>
  88. <span class="optional-desktop-only-text"> Add Song To Queue </span>
  89. </button>
  90. <button
  91. class="button is-primary tab-actionable-button"
  92. v-if="loggedIn && station.type === 'official'"
  93. @click="
  94. openModal({
  95. sector: 'station',
  96. modal: 'requestSong'
  97. })
  98. "
  99. >
  100. <i class="material-icons icon-with-button">queue</i>
  101. <span class="optional-desktop-only-text"> Request Song </span>
  102. </button>
  103. <button
  104. class="button is-primary tab-actionable-button disabled"
  105. v-if="
  106. !loggedIn &&
  107. ((station.type === 'community' &&
  108. station.partyMode &&
  109. !station.locked) ||
  110. station.type === 'official')
  111. "
  112. content="Login to add songs to queue"
  113. v-tippy
  114. >
  115. <i class="material-icons icon-with-button">queue</i>
  116. <span class="optional-desktop-only-text"> Add Song To Queue </span>
  117. </button>
  118. <div
  119. id="queue-locked"
  120. v-if="station.type === 'community' && station.locked"
  121. >
  122. <button
  123. v-if="isAdminOnly() && !isOwnerOnly() && !dismissedWarning"
  124. class="button tab-actionable-button"
  125. @click="dismissedWarning = true"
  126. >
  127. THIS STATION'S QUEUE IS LOCKED.
  128. </button>
  129. <button
  130. v-if="!isAdminOnly() && !isOwnerOnly()"
  131. class="button tab-actionable-button"
  132. >
  133. THIS STATION'S QUEUE IS LOCKED.
  134. </button>
  135. </div>
  136. </div>
  137. </template>
  138. <script>
  139. import { mapActions, mapState, mapGetters } from "vuex";
  140. import draggable from "vuedraggable";
  141. import Toast from "toasters";
  142. import SongItem from "@/components/SongItem.vue";
  143. import Confirm from "@/components/Confirm.vue";
  144. export default {
  145. components: { draggable, SongItem, Confirm },
  146. data() {
  147. return {
  148. dismissedWarning: false,
  149. actionableButtonVisible: false,
  150. drag: false
  151. };
  152. },
  153. computed: {
  154. queue: {
  155. get() {
  156. return this.$store.state.station.songsList;
  157. },
  158. set(queue) {
  159. this.$store.commit("station/updateSongsList", queue);
  160. }
  161. },
  162. dragOptions() {
  163. return {
  164. animation: 200,
  165. group: "queue",
  166. disabled: !(this.isAdminOnly() || this.isOwnerOnly()),
  167. ghostClass: "draggable-list-ghost"
  168. };
  169. },
  170. ...mapState({
  171. loggedIn: state => state.user.auth.loggedIn,
  172. userId: state => state.user.auth.userId,
  173. userRole: state => state.user.auth.role,
  174. station: state => state.station.station,
  175. songsList: state => state.station.songsList,
  176. noSong: state => state.station.noSong
  177. }),
  178. ...mapGetters({
  179. socket: "websockets/getSocket"
  180. })
  181. },
  182. updated() {
  183. // check if actionable button is visible, if not: set max-height of queue items to 100%
  184. if (
  185. document
  186. .getElementById("queue")
  187. .querySelectorAll(".tab-actionable-button").length > 0
  188. )
  189. this.actionableButtonVisible = true;
  190. else this.actionableButtonVisible = false;
  191. },
  192. methods: {
  193. isOwnerOnly() {
  194. return this.loggedIn && this.userId === this.station.owner;
  195. },
  196. isAdminOnly() {
  197. return this.loggedIn && this.userRole === "admin";
  198. },
  199. removeFromQueue(songId) {
  200. this.socket.dispatch(
  201. "stations.removeFromQueue",
  202. this.station._id,
  203. songId,
  204. res => {
  205. if (res.status === "success") {
  206. new Toast("Successfully removed song from the queue.");
  207. } else new Toast(res.message);
  208. }
  209. );
  210. },
  211. repositionSongInQueue({ moved }) {
  212. if (!moved) return; // we only need to update when song is moved
  213. this.socket.dispatch(
  214. "stations.repositionSongInQueue",
  215. {
  216. ...moved.element,
  217. oldIndex: moved.oldIndex,
  218. newIndex: moved.newIndex
  219. },
  220. this.station._id,
  221. res => {
  222. new Toast({ content: res.message, timeout: 4000 });
  223. }
  224. );
  225. },
  226. moveSongToTop(song, index) {
  227. this.repositionSongInQueue({
  228. moved: {
  229. element: song,
  230. oldIndex: index,
  231. newIndex: 0
  232. }
  233. });
  234. },
  235. moveSongToBottom(song, index) {
  236. this.repositionSongInQueue({
  237. moved: {
  238. element: song,
  239. oldIndex: index,
  240. newIndex: this.songsList.length
  241. }
  242. });
  243. },
  244. ...mapActions("modalVisibility", ["openModal"])
  245. }
  246. };
  247. </script>
  248. <style lang="scss" scoped>
  249. .night-mode {
  250. #queue {
  251. background-color: var(--dark-grey-3) !important;
  252. border: 0 !important;
  253. }
  254. }
  255. #queue {
  256. background-color: var(--white);
  257. border-radius: 0 0 5px 5px;
  258. .actionable-button-hidden {
  259. max-height: 100%;
  260. }
  261. .song-item:not(:last-of-type) {
  262. margin-bottom: 10px;
  263. }
  264. #queue-locked {
  265. display: flex;
  266. justify-content: center;
  267. }
  268. button.disabled {
  269. filter: grayscale(0.4);
  270. }
  271. }
  272. </style>