EditStation.vue 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272
  1. <template>
  2. <modal title="Edit Station" class="edit-station-modal">
  3. <template #body>
  4. <div class="custom-modal-body">
  5. <!-- Station Preferences -->
  6. <div class="section left-section">
  7. <div class="col col-2">
  8. <div>
  9. <label class="label">Name</label>
  10. <p class="control">
  11. <input
  12. class="input"
  13. type="text"
  14. v-model="editing.name"
  15. />
  16. </p>
  17. </div>
  18. <div>
  19. <label class="label">Display name</label>
  20. <p class="control">
  21. <input
  22. class="input"
  23. type="text"
  24. v-model="editing.displayName"
  25. />
  26. </p>
  27. </div>
  28. </div>
  29. <div class="col col-1">
  30. <div>
  31. <label class="label">Description</label>
  32. <p class="control">
  33. <input
  34. class="input"
  35. type="text"
  36. v-model="editing.description"
  37. />
  38. </p>
  39. </div>
  40. </div>
  41. <div
  42. class="col col-2"
  43. v-if="editing.type === 'official' && editing.genres"
  44. >
  45. <div>
  46. <label class="label">Genre(s)</label>
  47. <p class="control has-addons">
  48. <input
  49. class="input"
  50. type="text"
  51. id="new-genre"
  52. v-model="genreInputValue"
  53. @blur="blurGenreInput()"
  54. @focus="focusGenreInput()"
  55. @keydown="keydownGenreInput()"
  56. @keyup.enter="addTag('genres')"
  57. />
  58. <button
  59. class="button is-info add-button blue"
  60. @click="addTag('genres')"
  61. >
  62. <i class="material-icons">add</i>
  63. </button>
  64. </p>
  65. <div
  66. class="autosuggest-container"
  67. v-if="
  68. (genreInputFocussed ||
  69. genreAutosuggestContainerFocussed) &&
  70. genreAutosuggestItems.length > 0
  71. "
  72. @mouseover="focusGenreContainer()"
  73. @mouseleave="blurGenreContainer()"
  74. >
  75. <span
  76. class="autosuggest-item"
  77. tabindex="0"
  78. @click="selectGenreAutosuggest(item)"
  79. v-for="(item,
  80. index) in genreAutosuggestItems"
  81. :key="index"
  82. >{{ item }}</span
  83. >
  84. </div>
  85. <div class="list-container">
  86. <div
  87. class="list-item"
  88. v-for="(genre, index) in editing.genres"
  89. :key="index"
  90. >
  91. <div
  92. class="list-item-circle blue"
  93. @click="removeTag('genres', index)"
  94. >
  95. <i class="material-icons">close</i>
  96. </div>
  97. <p>{{ genre }}</p>
  98. </div>
  99. </div>
  100. </div>
  101. <div>
  102. <label class="label">Blacklist genre(s)</label>
  103. <p class="control has-addons">
  104. <input
  105. class="input"
  106. type="text"
  107. v-model="blacklistGenreInputValue"
  108. @blur="blurBlacklistGenreInput()"
  109. @focus="focusBlacklistGenreInput()"
  110. @keydown="keydownBlacklistGenreInput()"
  111. @keyup.enter="addTag('blacklist-genres')"
  112. />
  113. <button
  114. class="button is-info add-button red"
  115. @click="addTag('blacklist-genres')"
  116. >
  117. <i class="material-icons">add</i>
  118. </button>
  119. </p>
  120. <div
  121. class="autosuggest-container"
  122. v-if="
  123. (blacklistGenreInputFocussed ||
  124. blacklistGenreAutosuggestContainerFocussed) &&
  125. blacklistGenreAutosuggestItems.length >
  126. 0
  127. "
  128. @mouseover="focusBlacklistGenreContainer()"
  129. @mouseleave="blurBlacklistGenreContainer()"
  130. >
  131. <span
  132. class="autosuggest-item"
  133. tabindex="0"
  134. @click="
  135. selectBlacklistGenreAutosuggest(item)
  136. "
  137. v-for="(item,
  138. index) in blacklistGenreAutosuggestItems"
  139. :key="index"
  140. >{{ item }}</span
  141. >
  142. </div>
  143. <div class="list-container">
  144. <div
  145. class="list-item"
  146. v-for="(genre,
  147. index) in editing.blacklistedGenres"
  148. :key="index"
  149. >
  150. <div
  151. class="list-item-circle red"
  152. @click="
  153. removeTag('blacklist-genres', index)
  154. "
  155. >
  156. <i class="material-icons">close</i>
  157. </div>
  158. <p>{{ genre }}</p>
  159. </div>
  160. </div>
  161. </div>
  162. </div>
  163. <!-- Choose a playlist -->
  164. <div
  165. v-if="
  166. editing.type === 'community' &&
  167. !editing.partyMode &&
  168. playlists.length > 0
  169. "
  170. >
  171. <hr style="margin: 10px 0 20px 0" />
  172. <h4 class="section-title">Choose a playlist</h4>
  173. <p class="section-description">
  174. Choose one of your playlists to add to the queue.
  175. </p>
  176. <br />
  177. <div id="playlists">
  178. <playlist-item
  179. :playlist="playlist"
  180. v-for="(playlist, index) in playlists"
  181. :key="index"
  182. >
  183. <div slot="actions">
  184. <!-- <a
  185. class="button is-danger"
  186. href="#"
  187. @click="
  188. togglePlaylistSelection(
  189. playlist._id
  190. )
  191. "
  192. v-if="isPlaylistSelected(playlist._id)"
  193. >
  194. <i
  195. class="material-icons icon-with-button"
  196. >stop</i
  197. >
  198. Stop playing
  199. </a> -->
  200. <a
  201. class="button is-success"
  202. href="#"
  203. @click="selectPlaylist(playlist._id)"
  204. ><i
  205. class="material-icons icon-with-button"
  206. >play_arrow</i
  207. >Play in queue
  208. </a>
  209. </div>
  210. </playlist-item>
  211. </div>
  212. </div>
  213. </div>
  214. <!-- Buttons changing the privacy settings -->
  215. <div class="section right-section">
  216. <div>
  217. <label class="label">Privacy</label>
  218. <div
  219. @mouseenter="privacyDropdownActive = true"
  220. @mouseleave="privacyDropdownActive = false"
  221. class="button-wrapper"
  222. >
  223. <button
  224. :class="privacyButtons[editing.privacy].style"
  225. style="text-transform: capitalize"
  226. @click="updatePrivacyLocal(editing.privacy)"
  227. >
  228. <i class="material-icons">{{
  229. privacyButtons[editing.privacy].iconName
  230. }}</i>
  231. {{ editing.privacy }}
  232. </button>
  233. <transition name="slide-down">
  234. <button
  235. class="green"
  236. v-if="
  237. privacyDropdownActive &&
  238. editing.privacy !== 'public'
  239. "
  240. @click="updatePrivacyLocal('public')"
  241. >
  242. <i class="material-icons">{{
  243. privacyButtons["public"].iconName
  244. }}</i>
  245. Public
  246. </button>
  247. </transition>
  248. <transition name="slide-down">
  249. <button
  250. class="orange"
  251. v-if="
  252. privacyDropdownActive &&
  253. editing.privacy !== 'unlisted'
  254. "
  255. @click="updatePrivacyLocal('unlisted')"
  256. >
  257. <i class="material-icons">{{
  258. privacyButtons["unlisted"].iconName
  259. }}</i>
  260. Unlisted
  261. </button>
  262. </transition>
  263. <transition name="slide-down">
  264. <button
  265. class="red"
  266. v-if="
  267. privacyDropdownActive &&
  268. editing.privacy !== 'private'
  269. "
  270. @click="updatePrivacyLocal('private')"
  271. >
  272. <i class="material-icons">{{
  273. privacyButtons["private"].iconName
  274. }}</i>
  275. Private
  276. </button>
  277. </transition>
  278. </div>
  279. </div>
  280. <!-- Buttons changing the mode of the station -->
  281. <div v-if="editing.type === 'community'">
  282. <label class="label">Mode</label>
  283. <div
  284. @mouseenter="modeDropdownActive = true"
  285. @mouseleave="modeDropdownActive = false"
  286. class="button-wrapper"
  287. >
  288. <button
  289. :class="{
  290. blue: !editing.partyMode,
  291. yellow: editing.partyMode
  292. }"
  293. @click="
  294. editing.partyMode
  295. ? updatePartyModeLocal(true)
  296. : updatePartyModeLocal(false)
  297. "
  298. >
  299. <i class="material-icons">{{
  300. editing.partyMode
  301. ? "emoji_people"
  302. : "playlist_play"
  303. }}</i>
  304. {{ editing.partyMode ? "Party" : "Playlist" }}
  305. </button>
  306. <transition name="slide-down">
  307. <button
  308. class="blue"
  309. v-if="
  310. modeDropdownActive && editing.partyMode
  311. "
  312. @click="updatePartyModeLocal(false)"
  313. >
  314. <i class="material-icons">playlist_play</i>
  315. Playlist
  316. </button>
  317. </transition>
  318. <transition name="slide-down">
  319. <button
  320. class="yellow"
  321. v-if="
  322. modeDropdownActive && !editing.partyMode
  323. "
  324. @click="updatePartyModeLocal(true)"
  325. >
  326. <i class="material-icons">emoji_people</i>
  327. Party
  328. </button>
  329. </transition>
  330. </div>
  331. </div>
  332. <div
  333. v-if="
  334. editing.type === 'community' &&
  335. editing.partyMode === true
  336. "
  337. >
  338. <label class="label">Queue lock</label>
  339. <div
  340. @mouseenter="queueLockDropdownActive = true"
  341. @mouseleave="queueLockDropdownActive = false"
  342. class="button-wrapper"
  343. >
  344. <button
  345. :class="{
  346. green: editing.locked,
  347. red: !editing.locked
  348. }"
  349. @click="
  350. editing.locked
  351. ? updateQueueLockLocal(true)
  352. : updateQueueLockLocal(false)
  353. "
  354. >
  355. <i class="material-icons">{{
  356. editing.locked ? "lock" : "lock_open"
  357. }}</i>
  358. {{ editing.locked ? "Locked" : "Unlocked" }}
  359. </button>
  360. <transition name="slide-down">
  361. <button
  362. class="green"
  363. v-if="
  364. queueLockDropdownActive &&
  365. !editing.locked
  366. "
  367. @click="updateQueueLockLocal(true)"
  368. >
  369. <i class="material-icons">lock</i>
  370. Locked
  371. </button>
  372. </transition>
  373. <transition name="slide-down">
  374. <button
  375. class="red"
  376. v-if="
  377. queueLockDropdownActive &&
  378. editing.locked
  379. "
  380. @click="updateQueueLockLocal(false)"
  381. >
  382. <i class="material-icons">lock_open</i>
  383. Unlocked
  384. </button>
  385. </transition>
  386. </div>
  387. </div>
  388. </div>
  389. </div>
  390. </template>
  391. <template #footer>
  392. <button class="button is-success" @click="update()">
  393. Update Settings
  394. </button>
  395. <button
  396. v-if="station.type === 'community'"
  397. class="button is-danger"
  398. @click="deleteStation()"
  399. >
  400. Delete station
  401. </button>
  402. </template>
  403. </modal>
  404. </template>
  405. <script>
  406. import { mapState, mapActions } from "vuex";
  407. import Toast from "toasters";
  408. import PlaylistItem from "../ui/PlaylistItem.vue";
  409. import Modal from "../Modal.vue";
  410. import io from "../../io";
  411. import validation from "../../validation";
  412. export default {
  413. components: { Modal, PlaylistItem },
  414. props: { store: { type: String, default: "" } },
  415. data() {
  416. return {
  417. genreInputValue: "",
  418. genreInputFocussed: false,
  419. genreAutosuggestContainerFocussed: false,
  420. keydownGenreInputTimeout: 0,
  421. genreAutosuggestItems: [],
  422. blacklistGenreInputValue: "",
  423. blacklistGenreInputFocussed: false,
  424. blacklistGenreAutosuggestContainerFocussed: false,
  425. blacklistKeydownGenreInputTimeout: 0,
  426. blacklistGenreAutosuggestItems: [],
  427. privacyDropdownActive: false,
  428. modeDropdownActive: false,
  429. queueLockDropdownActive: false,
  430. genres: [
  431. "Blues",
  432. "Country",
  433. "Disco",
  434. "Funk",
  435. "Hip-Hop",
  436. "Jazz",
  437. "Metal",
  438. "Oldies",
  439. "Other",
  440. "Pop",
  441. "Rap",
  442. "Reggae",
  443. "Rock",
  444. "Techno",
  445. "Trance",
  446. "Classical",
  447. "Instrumental",
  448. "House",
  449. "Electronic",
  450. "Christian Rap",
  451. "Lo-Fi",
  452. "Musical",
  453. "Rock 'n' Roll",
  454. "Opera",
  455. "Drum & Bass",
  456. "Club-House",
  457. "Indie",
  458. "Heavy Metal",
  459. "Christian rock",
  460. "Dubstep"
  461. ],
  462. privacyButtons: {
  463. public: {
  464. style: "green",
  465. iconName: "public"
  466. },
  467. private: {
  468. style: "red",
  469. iconName: "lock"
  470. },
  471. unlisted: {
  472. style: "orange",
  473. iconName: "link"
  474. }
  475. },
  476. playlists: []
  477. };
  478. },
  479. computed: {
  480. ...mapState("admin/stations", {
  481. stations: state => state.stations
  482. }),
  483. ...mapState({
  484. editing(state) {
  485. return this.$props.store
  486. .split("/")
  487. .reduce((a, v) => a[v], state).editing;
  488. },
  489. station(state) {
  490. return this.$props.store
  491. .split("/")
  492. .reduce((a, v) => a[v], state).station;
  493. }
  494. })
  495. },
  496. mounted() {
  497. io.getSocket(socket => {
  498. this.socket = socket;
  499. this.socket.emit("playlists.indexForUser", res => {
  500. if (res.status === "success") this.playlists = res.data;
  501. });
  502. this.socket.on("event:playlist.create", playlist => {
  503. this.playlists.push(playlist);
  504. });
  505. this.socket.on("event:playlist.delete", playlistId => {
  506. this.playlists.forEach((playlist, index) => {
  507. if (playlist._id === playlistId) {
  508. this.playlists.splice(index, 1);
  509. }
  510. });
  511. });
  512. this.socket.on("event:playlist.addSong", data => {
  513. this.playlists.forEach((playlist, index) => {
  514. if (playlist._id === data.playlistId) {
  515. this.playlists[index].songs.push(data.song);
  516. }
  517. });
  518. });
  519. this.socket.on("event:playlist.removeSong", data => {
  520. this.playlists.forEach((playlist, index) => {
  521. if (playlist._id === data.playlistId) {
  522. this.playlists[index].songs.forEach((song, index2) => {
  523. if (song._id === data.songId) {
  524. this.playlists[index].songs.splice(index2, 1);
  525. }
  526. });
  527. }
  528. });
  529. });
  530. this.socket.on("event:playlist.updateDisplayName", data => {
  531. this.playlists.forEach((playlist, index) => {
  532. if (playlist._id === data.playlistId) {
  533. this.playlists[index].displayName = data.displayName;
  534. }
  535. });
  536. });
  537. return socket;
  538. });
  539. },
  540. methods: {
  541. isPlaylistSelected(id) {
  542. // TODO Also change this once it changes for a station
  543. if (this.station && this.station.privatePlaylist === id)
  544. return true;
  545. return false;
  546. },
  547. selectPlaylist(playlistId) {
  548. this.socket.emit(
  549. "stations.selectPrivatePlaylist",
  550. this.station._id,
  551. playlistId,
  552. res => {
  553. if (res.status === "failure")
  554. return new Toast({
  555. content: res.message,
  556. timeout: 8000
  557. });
  558. return new Toast({ content: res.message, timeout: 4000 });
  559. }
  560. );
  561. },
  562. update() {
  563. if (this.station.name !== this.editing.name) this.updateName();
  564. if (this.station.displayName !== this.editing.displayName)
  565. this.updateDisplayName();
  566. if (this.station.description !== this.editing.description)
  567. this.updateDescription();
  568. if (this.station.privacy !== this.editing.privacy)
  569. this.updatePrivacy();
  570. if (
  571. this.station.type === "community" &&
  572. this.station.partyMode !== this.editing.partyMode
  573. )
  574. this.updatePartyMode();
  575. if (
  576. this.station.type === "community" &&
  577. this.editing.partyMode &&
  578. this.station.locked !== this.editing.locked
  579. )
  580. this.updateQueueLock();
  581. if (this.$props.store !== "station") {
  582. if (
  583. this.station.genres.toString() !==
  584. this.editing.genres.toString()
  585. )
  586. this.updateGenres();
  587. if (
  588. this.station.blacklistedGenres.toString() !==
  589. this.editing.blacklistedGenres.toString()
  590. )
  591. this.updateBlacklistedGenres();
  592. }
  593. },
  594. updateName() {
  595. const { name } = this.editing;
  596. if (!validation.isLength(name, 2, 16))
  597. return new Toast({
  598. content: "Name must have between 2 and 16 characters.",
  599. timeout: 8000
  600. });
  601. if (!validation.regex.az09_.test(name))
  602. return new Toast({
  603. content:
  604. "Invalid name format. Allowed characters: a-z, 0-9 and _.",
  605. timeout: 8000
  606. });
  607. return this.socket.emit(
  608. "stations.updateName",
  609. this.editing._id,
  610. name,
  611. res => {
  612. if (res.status === "success") {
  613. if (this.station) this.station.name = name;
  614. else {
  615. this.stations.forEach((station, index) => {
  616. if (station._id === this.editing._id) {
  617. this.stations[index].name = name;
  618. return name;
  619. }
  620. return false;
  621. });
  622. }
  623. }
  624. new Toast({ content: res.message, timeout: 8000 });
  625. }
  626. );
  627. },
  628. updateDisplayName() {
  629. const { displayName } = this.editing;
  630. if (!validation.isLength(displayName, 2, 32))
  631. return new Toast({
  632. content:
  633. "Display name must have between 2 and 32 characters.",
  634. timeout: 8000
  635. });
  636. if (!validation.regex.ascii.test(displayName))
  637. return new Toast({
  638. content:
  639. "Invalid display name format. Only ASCII characters are allowed.",
  640. timeout: 8000
  641. });
  642. return this.socket.emit(
  643. "stations.updateDisplayName",
  644. this.editing._id,
  645. displayName,
  646. res => {
  647. if (res.status === "success") {
  648. if (this.station)
  649. this.station.displayName = displayName;
  650. else {
  651. this.stations.forEach((station, index) => {
  652. if (station._id === this.editing._id) {
  653. this.stations[
  654. index
  655. ].displayName = displayName;
  656. return displayName;
  657. }
  658. return false;
  659. });
  660. }
  661. }
  662. new Toast({ content: res.message, timeout: 8000 });
  663. }
  664. );
  665. },
  666. updateDescription() {
  667. const { description } = this.editing;
  668. if (!validation.isLength(description, 2, 200))
  669. return new Toast({
  670. content:
  671. "Description must have between 2 and 200 characters.",
  672. timeout: 8000
  673. });
  674. let characters = description.split("");
  675. characters = characters.filter(character => {
  676. return character.charCodeAt(0) === 21328;
  677. });
  678. if (characters.length !== 0)
  679. return new Toast({
  680. content: "Invalid description format.",
  681. timeout: 8000
  682. });
  683. return this.socket.emit(
  684. "stations.updateDescription",
  685. this.editing._id,
  686. description,
  687. res => {
  688. if (res.status === "success") {
  689. if (this.station)
  690. this.station.description = description;
  691. else {
  692. this.stations.forEach((station, index) => {
  693. if (station._id === this.editing._id) {
  694. this.stations[
  695. index
  696. ].description = description;
  697. return description;
  698. }
  699. return false;
  700. });
  701. }
  702. return new Toast({
  703. content: res.message,
  704. timeout: 4000
  705. });
  706. }
  707. return new Toast({ content: res.message, timeout: 8000 });
  708. }
  709. );
  710. },
  711. updatePrivacyLocal(privacy) {
  712. if (this.editing.privacy === privacy) return;
  713. this.editing.privacy = privacy;
  714. this.privacyDropdownActive = false;
  715. },
  716. updatePrivacy() {
  717. this.socket.emit(
  718. "stations.updatePrivacy",
  719. this.editing._id,
  720. this.editing.privacy,
  721. res => {
  722. if (res.status === "success") {
  723. if (this.station)
  724. this.station.privacy = this.editing.privacy;
  725. else {
  726. this.stations.forEach((station, index) => {
  727. if (station._id === this.editing._id) {
  728. this.stations[
  729. index
  730. ].privacy = this.editing.privacy;
  731. return this.editing.privacy;
  732. }
  733. return false;
  734. });
  735. }
  736. return new Toast({
  737. content: res.message,
  738. timeout: 4000
  739. });
  740. }
  741. return new Toast({ content: res.message, timeout: 8000 });
  742. }
  743. );
  744. },
  745. updateGenres() {
  746. this.socket.emit(
  747. "stations.updateGenres",
  748. this.editing._id,
  749. this.editing.genres,
  750. res => {
  751. if (res.status === "success") {
  752. const genres = JSON.parse(
  753. JSON.stringify(this.editing.genres)
  754. );
  755. if (this.station) this.station.genres = genres;
  756. this.stations.forEach((station, index) => {
  757. if (station._id === this.editing._id) {
  758. this.stations[index].genres = genres;
  759. return genres;
  760. }
  761. return false;
  762. });
  763. return new Toast({
  764. content: res.message,
  765. timeout: 4000
  766. });
  767. }
  768. return new Toast({ content: res.message, timeout: 8000 });
  769. }
  770. );
  771. },
  772. updateBlacklistedGenres() {
  773. this.socket.emit(
  774. "stations.updateBlacklistedGenres",
  775. this.editing._id,
  776. this.editing.blacklistedGenres,
  777. res => {
  778. if (res.status === "success") {
  779. const blacklistedGenres = JSON.parse(
  780. JSON.stringify(this.editing.blacklistedGenres)
  781. );
  782. if (this.station)
  783. this.station.blacklistedGenres = blacklistedGenres;
  784. this.stations.forEach((station, index) => {
  785. if (station._id === this.editing._id) {
  786. this.stations[
  787. index
  788. ].blacklistedGenres = blacklistedGenres;
  789. return blacklistedGenres;
  790. }
  791. return false;
  792. });
  793. return new Toast({
  794. content: res.message,
  795. timeout: 4000
  796. });
  797. }
  798. return new Toast({ content: res.message, timeout: 8000 });
  799. }
  800. );
  801. },
  802. updatePartyModeLocal(partyMode) {
  803. if (this.editing.partyMode === partyMode) return;
  804. this.editing.partyMode = partyMode;
  805. this.modeDropdownActive = false;
  806. },
  807. updatePartyMode() {
  808. this.socket.emit(
  809. "stations.updatePartyMode",
  810. this.editing._id,
  811. this.editing.partyMode,
  812. res => {
  813. if (res.status === "success") {
  814. if (this.station)
  815. this.station.partyMode = this.editing.partyMode;
  816. // if (this.station)
  817. // this.station.partyMode = this.editing.partyMode;
  818. // this.stations.forEach((station, index) => {
  819. // if (station._id === this.editing._id) {
  820. // this.stations[
  821. // index
  822. // ].partyMode = this.editing.partyMode;
  823. // return this.editing.partyMode;
  824. // }
  825. // return false;
  826. // });
  827. return new Toast({
  828. content: res.message,
  829. timeout: 4000
  830. });
  831. }
  832. return new Toast({ content: res.message, timeout: 8000 });
  833. }
  834. );
  835. },
  836. updateQueueLockLocal(locked) {
  837. if (this.editing.locked === locked) return;
  838. this.editing.locked = locked;
  839. this.queueLockDropdownActive = false;
  840. },
  841. updateQueueLock() {
  842. this.socket.emit("stations.toggleLock", this.editing._id, res => {
  843. console.log(res);
  844. if (res.status === "success") {
  845. if (this.station) this.station.locked = res.data;
  846. return new Toast({
  847. content: `Toggled queue lock succesfully to ${res.data}`,
  848. timeout: 4000
  849. });
  850. }
  851. return new Toast({
  852. content: "Failed to toggle queue lock.",
  853. timeout: 8000
  854. });
  855. });
  856. },
  857. deleteStation() {
  858. this.socket.emit("stations.remove", this.editing._id, res => {
  859. if (res.status === "success")
  860. this.closeModal({
  861. sector: "station",
  862. modal: "editStation"
  863. });
  864. return new Toast({ content: res.message, timeout: 8000 });
  865. });
  866. },
  867. blurGenreInput() {
  868. this.genreInputFocussed = false;
  869. },
  870. focusGenreInput() {
  871. this.genreInputFocussed = true;
  872. },
  873. keydownGenreInput() {
  874. clearTimeout(this.keydownGenreInputTimeout);
  875. this.keydownGenreInputTimeout = setTimeout(() => {
  876. if (this.genreInputValue.length > 1) {
  877. this.genreAutosuggestItems = this.genres.filter(genre => {
  878. return genre
  879. .toLowerCase()
  880. .startsWith(this.genreInputValue.toLowerCase());
  881. });
  882. } else this.genreAutosuggestItems = [];
  883. }, 1000);
  884. },
  885. focusGenreContainer() {
  886. this.genreAutosuggestContainerFocussed = true;
  887. },
  888. blurGenreContainer() {
  889. this.genreAutosuggestContainerFocussed = false;
  890. },
  891. selectGenreAutosuggest(value) {
  892. this.genreInputValue = value;
  893. },
  894. blurBlacklistGenreInput() {
  895. this.blacklistGenreInputFocussed = false;
  896. },
  897. focusBlacklistGenreInput() {
  898. this.blacklistGenreInputFocussed = true;
  899. },
  900. keydownBlacklistGenreInput() {
  901. clearTimeout(this.keydownBlacklistGenreInputTimeout);
  902. this.keydownBlacklistGenreInputTimeout = setTimeout(() => {
  903. if (this.blacklistGenreInputValue.length > 1) {
  904. this.blacklistGenreAutosuggestItems = this.genres.filter(
  905. genre => {
  906. return genre
  907. .toLowerCase()
  908. .startsWith(
  909. this.blacklistGenreInputValue.toLowerCase()
  910. );
  911. }
  912. );
  913. } else this.blacklistGenreAutosuggestItems = [];
  914. }, 1000);
  915. },
  916. focusBlacklistGenreContainer() {
  917. this.blacklistGenreAutosuggestContainerFocussed = true;
  918. },
  919. blurBlacklistGenreContainer() {
  920. this.blacklistGenreAutosuggestContainerFocussed = false;
  921. },
  922. selectBlacklistGenreAutosuggest(value) {
  923. this.blacklistGenreInputValue = value;
  924. },
  925. addTag(type) {
  926. if (type === "genres") {
  927. const genre = this.genreInputValue.toLowerCase().trim();
  928. if (this.editing.genres.indexOf(genre) !== -1)
  929. return new Toast({
  930. content: "Genre already exists",
  931. timeout: 3000
  932. });
  933. if (genre) {
  934. this.editing.genres.push(genre);
  935. this.genreInputValue = "";
  936. return false;
  937. }
  938. return new Toast({
  939. content: "Genre cannot be empty",
  940. timeout: 3000
  941. });
  942. }
  943. if (type === "blacklist-genres") {
  944. const genre = this.blacklistGenreInputValue
  945. .toLowerCase()
  946. .trim();
  947. if (this.editing.blacklistedGenres.indexOf(genre) !== -1)
  948. return new Toast({
  949. content: "Blacklist genre already exists",
  950. timeout: 3000
  951. });
  952. if (genre) {
  953. this.editing.blacklistedGenres.push(genre);
  954. this.blacklistGenreInputValue = "";
  955. return false;
  956. }
  957. return new Toast({
  958. content: "Blacklist genre cannot be empty",
  959. timeout: 3000
  960. });
  961. }
  962. return false;
  963. },
  964. removeTag(type, index) {
  965. if (type === "genres") this.editing.genres.splice(index, 1);
  966. else if (type === "blacklist-genres")
  967. this.editing.blacklistedGenres.splice(index, 1);
  968. },
  969. ...mapActions("modals", ["closeModal"])
  970. }
  971. };
  972. </script>
  973. <style lang="scss" scoped>
  974. @import "../../styles/global.scss";
  975. .night-mode {
  976. .modal-card,
  977. .modal-card-head,
  978. .modal-card-body,
  979. .modal-card-foot {
  980. background-color: $night-mode-bg-secondary;
  981. }
  982. .section {
  983. background-color: $night-mode-bg-secondary !important;
  984. border: 0 !important;
  985. }
  986. .label,
  987. p,
  988. strong {
  989. color: $night-mode-text;
  990. }
  991. }
  992. .modal-card-title {
  993. text-align: center;
  994. margin-left: 24px;
  995. }
  996. .custom-modal-body {
  997. padding: 16px;
  998. display: flex;
  999. }
  1000. .section {
  1001. border: 1px solid #a3e0ff;
  1002. background-color: #f4f4f4;
  1003. border-radius: 5px;
  1004. padding: 16px;
  1005. }
  1006. .left-section {
  1007. width: 595px;
  1008. display: grid;
  1009. gap: 16px;
  1010. grid-template-rows: min-content min-content auto;
  1011. .control {
  1012. input {
  1013. width: 100%;
  1014. height: 36px;
  1015. }
  1016. .add-button {
  1017. width: 32px;
  1018. &.blue {
  1019. background-color: $musare-blue !important;
  1020. }
  1021. &.red {
  1022. background-color: $red !important;
  1023. }
  1024. i {
  1025. font-size: 32px;
  1026. }
  1027. }
  1028. }
  1029. .col {
  1030. > div {
  1031. position: relative;
  1032. }
  1033. }
  1034. .list-item-circle {
  1035. width: 16px;
  1036. height: 16px;
  1037. border-radius: 8px;
  1038. cursor: pointer;
  1039. margin-right: 8px;
  1040. float: left;
  1041. -webkit-touch-callout: none;
  1042. -webkit-user-select: none;
  1043. -khtml-user-select: none;
  1044. -moz-user-select: none;
  1045. -ms-user-select: none;
  1046. user-select: none;
  1047. &.blue {
  1048. background-color: $musare-blue;
  1049. i {
  1050. color: $musare-blue;
  1051. }
  1052. }
  1053. &.red {
  1054. background-color: $red;
  1055. i {
  1056. color: $red;
  1057. }
  1058. }
  1059. i {
  1060. font-size: 14px;
  1061. margin-left: 1px;
  1062. }
  1063. }
  1064. .list-item-circle:hover,
  1065. .list-item-circle:focus {
  1066. i {
  1067. color: white;
  1068. }
  1069. }
  1070. .list-item > p {
  1071. line-height: 16px;
  1072. word-wrap: break-word;
  1073. width: calc(100% - 24px);
  1074. left: 24px;
  1075. float: left;
  1076. margin-bottom: 8px;
  1077. }
  1078. .list-item:last-child > p {
  1079. margin-bottom: 0;
  1080. }
  1081. .autosuggest-container {
  1082. position: absolute;
  1083. background: white;
  1084. width: calc(100% + 1px);
  1085. top: 57px;
  1086. z-index: 200;
  1087. overflow: auto;
  1088. max-height: 100%;
  1089. clear: both;
  1090. .autosuggest-item {
  1091. padding: 8px;
  1092. display: block;
  1093. border: 1px solid #dbdbdb;
  1094. margin-top: -1px;
  1095. line-height: 16px;
  1096. cursor: pointer;
  1097. -webkit-user-select: none;
  1098. -ms-user-select: none;
  1099. -moz-user-select: none;
  1100. user-select: none;
  1101. }
  1102. .autosuggest-item:hover,
  1103. .autosuggest-item:focus {
  1104. background-color: #eee;
  1105. }
  1106. .autosuggest-item:first-child {
  1107. border-top: none;
  1108. }
  1109. .autosuggest-item:last-child {
  1110. border-radius: 0 0 3px 3px;
  1111. }
  1112. }
  1113. }
  1114. .right-section {
  1115. width: 157px;
  1116. min-height: 375px;
  1117. margin-left: 16px;
  1118. display: grid;
  1119. gap: 16px;
  1120. grid-template-rows: min-content min-content min-content;
  1121. .button-wrapper {
  1122. display: flex;
  1123. flex-direction: column;
  1124. }
  1125. button {
  1126. width: 100%;
  1127. height: 36px;
  1128. border: 0;
  1129. border-radius: 3px;
  1130. font-size: 18px;
  1131. color: white;
  1132. box-shadow: 0px 1px 3px rgba(0, 0, 0, 0.25);
  1133. display: block;
  1134. text-align: center;
  1135. justify-content: center;
  1136. display: inline-flex;
  1137. -ms-flex-align: center;
  1138. align-items: center;
  1139. -moz-user-select: none;
  1140. user-select: none;
  1141. cursor: pointer;
  1142. margin-bottom: 10px;
  1143. padding: 0;
  1144. &.red {
  1145. background-color: $red;
  1146. }
  1147. &.green {
  1148. background-color: $green;
  1149. }
  1150. &.blue {
  1151. background-color: $musare-blue;
  1152. }
  1153. &.orange {
  1154. background-color: $light-orange;
  1155. }
  1156. &.yellow {
  1157. background-color: $yellow;
  1158. }
  1159. i {
  1160. font-size: 20px;
  1161. margin-right: 4px;
  1162. }
  1163. }
  1164. }
  1165. .col {
  1166. display: grid;
  1167. grid-column-gap: 16px;
  1168. }
  1169. .col-1 {
  1170. grid-template-columns: auto;
  1171. }
  1172. .col-2 {
  1173. grid-template-columns: auto auto;
  1174. }
  1175. .slide-down-enter-active {
  1176. transition: transform 0.25s;
  1177. }
  1178. .slide-down-enter {
  1179. transform: translateY(-10px);
  1180. }
  1181. #playlists {
  1182. overflow: auto;
  1183. }
  1184. .modal-card {
  1185. overflow: auto;
  1186. }
  1187. .modal-card-body {
  1188. overflow: unset;
  1189. }
  1190. </style>