Search.vue 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. <template>
  2. <div class="search">
  3. <div class="musare-search">
  4. <label class="label"> Search for a song on Musare </label>
  5. <div class="control is-grouped input-with-button">
  6. <p class="control is-expanded">
  7. <input
  8. class="input"
  9. type="text"
  10. placeholder="Enter your song query here..."
  11. v-model="musareSearch.query"
  12. @keyup.enter="searchForMusareSongs(1)"
  13. />
  14. </p>
  15. <p class="control">
  16. <a class="button is-info" @click="searchForMusareSongs(1)"
  17. ><i class="material-icons icon-with-button">search</i
  18. >Search</a
  19. >
  20. </p>
  21. </div>
  22. <div v-if="musareSearch.results.length > 0">
  23. <song-item
  24. v-for="(song, index) in musareSearch.results"
  25. :key="index + song._id"
  26. :song="song"
  27. >
  28. <div class="song-actions" slot="actions">
  29. <i
  30. class="material-icons add-to-queue-icon"
  31. v-if="station.partyMode && !station.locked"
  32. @click="addSongToQueue(song.youtubeId)"
  33. content="Add Song to Queue"
  34. v-tippy
  35. >queue</i
  36. >
  37. </div>
  38. </song-item>
  39. </div>
  40. </div>
  41. <div class="youtube-search">
  42. <label class="label"> Search for a song on YouTube </label>
  43. <div class="control is-grouped input-with-button">
  44. <p class="control is-expanded">
  45. <input
  46. class="input"
  47. type="text"
  48. placeholder="Enter your YouTube query here..."
  49. v-model="search.songs.query"
  50. autofocus
  51. @keyup.enter="searchForSongs()"
  52. />
  53. </p>
  54. <p class="control">
  55. <a
  56. class="button is-info"
  57. @click.prevent="searchForSongs()"
  58. href="#"
  59. ><i class="material-icons icon-with-button">search</i
  60. >Search</a
  61. >
  62. </p>
  63. </div>
  64. <div v-if="search.songs.results.length > 0" id="song-query-results">
  65. <search-query-item
  66. v-for="(result, index) in search.songs.results"
  67. :key="index"
  68. :result="result"
  69. >
  70. <div slot="actions">
  71. <transition name="search-query-actions" mode="out-in">
  72. <a
  73. class="button is-success"
  74. v-if="result.isAddedToQueue"
  75. href="#"
  76. key="added-to-queue"
  77. >
  78. <i class="material-icons icon-with-button"
  79. >done</i
  80. >
  81. Added to queue
  82. </a>
  83. <a
  84. class="button is-dark"
  85. v-else
  86. @click.prevent="
  87. addSongToQueue(result.id, index)
  88. "
  89. href="#"
  90. key="add-to-queue"
  91. >
  92. <i class="material-icons icon-with-button"
  93. >add</i
  94. >
  95. Add to queue
  96. </a>
  97. </transition>
  98. </div>
  99. </search-query-item>
  100. <a
  101. class="button is-default load-more-button"
  102. @click.prevent="loadMoreSongs()"
  103. href="#"
  104. >
  105. Load more...
  106. </a>
  107. </div>
  108. </div>
  109. </div>
  110. </template>
  111. <script>
  112. import { mapState, mapGetters } from "vuex";
  113. import Toast from "toasters";
  114. import SearchYoutube from "@/mixins/SearchYoutube.vue";
  115. import SongItem from "@/components/SongItem.vue";
  116. import SearchQueryItem from "../../../SearchQueryItem.vue";
  117. export default {
  118. components: {
  119. SongItem,
  120. SearchQueryItem
  121. },
  122. mixins: [SearchYoutube],
  123. data() {
  124. return {
  125. musareSearch: {
  126. query: "",
  127. results: []
  128. }
  129. };
  130. },
  131. computed: {
  132. ...mapState("modals/manageStation", {
  133. station: state => state.station,
  134. originalStation: state => state.originalStation
  135. }),
  136. ...mapGetters({
  137. socket: "websockets/getSocket"
  138. })
  139. },
  140. methods: {
  141. addSongToQueue(youtubeId, index) {
  142. if (this.station.type === "community") {
  143. this.socket.dispatch(
  144. "stations.addToQueue",
  145. this.station._id,
  146. youtubeId,
  147. res => {
  148. if (res.status !== "success")
  149. new Toast(`Error: ${res.message}`);
  150. else {
  151. if (index)
  152. this.search.songs.results[
  153. index
  154. ].isAddedToQueue = true;
  155. new Toast(res.message);
  156. }
  157. }
  158. );
  159. } else {
  160. this.socket.dispatch("songs.request", youtubeId, res => {
  161. if (res.status !== "success")
  162. new Toast(`Error: ${res.message}`);
  163. else {
  164. this.search.songs.results[index].isAddedToQueue = true;
  165. new Toast(res.message);
  166. }
  167. });
  168. }
  169. },
  170. searchForMusareSongs(page) {
  171. const { query } = this.musareSearch;
  172. this.socket.dispatch("songs.searchOfficial", query, page, res => {
  173. if (res.status === "success") {
  174. this.musareSearch.results = res.data.songs;
  175. } else if (res.status === "error") {
  176. this.musareSearch.results = [];
  177. new Toast(res.message);
  178. }
  179. });
  180. }
  181. }
  182. };
  183. </script>
  184. <style lang="scss">
  185. .search {
  186. .musare-search,
  187. .universal-item:not(:last-of-type) {
  188. margin-bottom: 10px;
  189. }
  190. .load-more-button {
  191. width: 100%;
  192. margin-top: 10px;
  193. }
  194. }
  195. </style>