AddToPlaylistDropdown.vue 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. <template>
  2. <div id="nav-dropdown">
  3. <div class="nav-dropdown-items" v-if="playlists.length > 0">
  4. <!-- <a class="nav-item" id="nightmode-toggle">
  5. <span>Nightmode</span>
  6. <label class="switch">
  7. <input type="checkbox" checked />
  8. <span class="slider round"></span>
  9. </label>
  10. </a> -->
  11. <button
  12. class="nav-item"
  13. href="#"
  14. v-for="(playlist, index) in playlists"
  15. :key="index"
  16. @click.prevent="toggleSongInPlaylist(index, playlist._id)"
  17. :title="playlist.displayName"
  18. >
  19. <p class="control is-expanded checkbox-control">
  20. <input
  21. type="checkbox"
  22. :id="index"
  23. v-model="playlist.hasSong"
  24. />
  25. <label :for="index">
  26. <span></span>
  27. <p>{{ playlist.displayName }}</p>
  28. </label>
  29. </p>
  30. </button>
  31. </div>
  32. <p class="nav-dropdown-items" id="no-playlists" v-else>
  33. You haven't created any playlists.
  34. </p>
  35. </div>
  36. </template>
  37. <script>
  38. import { mapState } from "vuex";
  39. import Toast from "toasters";
  40. import io from "../../../io";
  41. export default {
  42. data() {
  43. return {
  44. playlists: []
  45. };
  46. },
  47. computed: {
  48. ...mapState("station", {
  49. currentSong: state => state.currentSong
  50. })
  51. },
  52. mounted() {
  53. io.getSocket(socket => {
  54. this.socket = socket;
  55. this.socket.emit("playlists.indexMyPlaylists", false, res => {
  56. if (res.status === "success") {
  57. this.playlists = res.data;
  58. this.checkIfPlaylistsHaveSong();
  59. }
  60. });
  61. this.socket.on("event:songs.next", () => {
  62. this.checkIfPlaylistsHaveSong();
  63. });
  64. this.socket.on("event:playlist.create", playlist => {
  65. this.playlists.push(playlist);
  66. });
  67. this.socket.on("event:playlist.delete", playlistId => {
  68. this.playlists.forEach((playlist, index) => {
  69. if (playlist._id === playlistId) {
  70. this.playlists.splice(index, 1);
  71. }
  72. });
  73. });
  74. this.socket.on("event:playlist.updateDisplayName", data => {
  75. this.playlists.forEach((playlist, index) => {
  76. if (playlist._id === data.playlistId) {
  77. this.playlists[index].displayName = data.displayName;
  78. }
  79. });
  80. });
  81. });
  82. },
  83. methods: {
  84. toggleSongInPlaylist(index, playlistId) {
  85. if (!this.playlists[index].hasSong) {
  86. this.socket.emit(
  87. "playlists.addSongToPlaylist",
  88. false,
  89. this.currentSong.songId,
  90. playlistId,
  91. res => {
  92. new Toast({ content: res.message, timeout: 4000 });
  93. if (res.status === "success") {
  94. this.playlists[index].songs.push(this.currentSong);
  95. this.playlists[index].hasSong = true;
  96. }
  97. }
  98. );
  99. } else {
  100. this.socket.emit(
  101. "playlists.removeSongFromPlaylist",
  102. this.currentSong.songId,
  103. playlistId,
  104. res => {
  105. new Toast({ content: res.message, timeout: 4000 });
  106. if (res.status === "success") {
  107. this.playlists[index].songs.forEach(
  108. (song, index) => {
  109. if (song.songId === this.currentSong.songId)
  110. this.playlists[index].songs.splice(
  111. index,
  112. 1
  113. );
  114. }
  115. );
  116. this.playlists[index].hasSong = false;
  117. }
  118. }
  119. );
  120. }
  121. },
  122. checkIfPlaylistsHaveSong() {
  123. this.playlists.forEach((playlist, index) => {
  124. let hasSong = false;
  125. for (let song = 0; song < playlist.songs.length; song += 1) {
  126. if (playlist.songs[song].songId === this.currentSong.songId)
  127. hasSong = true;
  128. }
  129. this.$set(this.playlists[index], "hasSong", hasSong);
  130. });
  131. }
  132. }
  133. };
  134. </script>
  135. <style lang="scss" scoped>
  136. @import "../../../styles/global.scss";
  137. .night-mode {
  138. .nav-dropdown-items {
  139. background-color: var(--dark-grey-2);
  140. border: 0 !important;
  141. .nav-item {
  142. background-color: var(--dark-grey);
  143. &:focus {
  144. outline-color: var(--dark-grey);
  145. }
  146. p {
  147. color: var(--white);
  148. }
  149. .checkbox-control label span {
  150. background-color: var(--dark-grey-2);
  151. }
  152. }
  153. }
  154. }
  155. #nav-dropdown {
  156. position: absolute;
  157. margin-left: 4px;
  158. z-index: 1;
  159. margin-bottom: 36px;
  160. }
  161. #nav-dropdown-triangle {
  162. border-style: solid;
  163. border-width: 15px 15px 0 15px;
  164. border-color: var(--dark-grey-2) transparent transparent transparent;
  165. }
  166. .nav-dropdown-items {
  167. border: 1px solid var(--light-grey-3);
  168. box-shadow: 0 14px 28px rgba(0, 0, 0, 0.25), 0 10px 10px rgba(0, 0, 0, 0.22);
  169. background-color: var(--white);
  170. padding: 5px;
  171. border-radius: 5px;
  172. position: relative;
  173. right: calc(100% - 110px);
  174. .nav-item {
  175. width: 100%;
  176. justify-content: flex-start;
  177. border: 0;
  178. padding: 10px;
  179. font-size: 15.5px;
  180. height: 36px;
  181. background: var(--light-grey);
  182. border-radius: 5px;
  183. cursor: pointer;
  184. .checkbox-control {
  185. display: flex;
  186. align-items: center;
  187. margin-bottom: 0 !important;
  188. input {
  189. margin-right: 5px;
  190. }
  191. input[type="checkbox"] {
  192. opacity: 0;
  193. position: absolute;
  194. }
  195. label {
  196. display: flex;
  197. flex-direction: row;
  198. align-items: center;
  199. span {
  200. cursor: pointer;
  201. width: 24px;
  202. height: 24px;
  203. background-color: var(--white);
  204. display: inline-block;
  205. border: 1px solid var(--dark-grey-2);
  206. position: relative;
  207. border-radius: 3px;
  208. }
  209. p {
  210. margin-left: 10px;
  211. cursor: pointer;
  212. color: var(--black);
  213. }
  214. }
  215. input[type="checkbox"]:checked + label span::after {
  216. content: "";
  217. width: 18px;
  218. height: 18px;
  219. left: 2px;
  220. top: 2px;
  221. border-radius: 3px;
  222. background-color: var(--station-theme);
  223. position: absolute;
  224. }
  225. }
  226. &:focus {
  227. outline-color: var(--light-grey-3);
  228. }
  229. &:not(:last-of-type) {
  230. margin-bottom: 5px;
  231. }
  232. }
  233. }
  234. #nightmode-toggle {
  235. display: flex;
  236. justify-content: space-evenly;
  237. }
  238. /* CSS - Toggle Switch */
  239. .switch {
  240. position: relative;
  241. display: inline-block;
  242. width: 50px;
  243. height: 24px;
  244. input {
  245. opacity: 0;
  246. width: 0;
  247. height: 0;
  248. }
  249. }
  250. .slider {
  251. position: absolute;
  252. cursor: pointer;
  253. top: 0;
  254. left: 0;
  255. right: 0;
  256. bottom: 0;
  257. background-color: var(--light-grey-3);
  258. transition: 0.4s;
  259. border-radius: 34px;
  260. &:before {
  261. position: absolute;
  262. content: "";
  263. height: 16px;
  264. width: 16px;
  265. left: 4px;
  266. bottom: 4px;
  267. background-color: var(--white);
  268. transition: 0.4s;
  269. border-radius: 50%;
  270. }
  271. }
  272. input:checked + .slider {
  273. background-color: var(--primary-color);
  274. }
  275. input:focus + .slider {
  276. box-shadow: 0 0 1px var(--primary-color);
  277. }
  278. input:checked + .slider:before {
  279. transform: translateX(26px);
  280. }
  281. #no-playlists {
  282. padding: 10px;
  283. }
  284. </style>