StationHeader.vue 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. <template>
  2. <div>
  3. <nav class="nav">
  4. <div class="nav-left">
  5. <router-link class="nav-item is-brand" :to="{ path: '/' }">
  6. <img
  7. :src="`${this.siteSettings.logo}`"
  8. :alt="`${this.siteSettings.siteName}` || `Musare`"
  9. />
  10. </router-link>
  11. </div>
  12. <div class="nav-center stationDisplayName">
  13. {{ station.displayName }}
  14. </div>
  15. <span class="nav-toggle" v-on:click="controlBar = !controlBar">
  16. <span />
  17. <span />
  18. <span />
  19. </span>
  20. <div class="nav-right nav-menu" :class="{ 'is-active': isMobile }">
  21. <router-link
  22. v-if="role === 'admin'"
  23. class="nav-item is-tab admin"
  24. :to="{ path: '/admin' }"
  25. >
  26. <strong>Admin</strong>
  27. </router-link>
  28. <span v-if="loggedIn" class="grouped">
  29. <router-link
  30. class="nav-item is-tab"
  31. :to="{ path: '/u/' + username }"
  32. >Profile</router-link
  33. >
  34. <router-link class="nav-item is-tab" to="/settings"
  35. >Settings</router-link
  36. >
  37. <a class="nav-item is-tab" @click="logout()">Logout</a>
  38. </span>
  39. <span v-else class="grouped">
  40. <a
  41. class="nav-item"
  42. href="#"
  43. @click="openModal({ sector: 'header', modal: 'login' })"
  44. >Login</a
  45. >
  46. <a
  47. class="nav-item"
  48. href="#"
  49. @click="
  50. openModal({ sector: 'header', modal: 'register' })
  51. "
  52. >Register</a
  53. >
  54. </span>
  55. </div>
  56. </nav>
  57. <div class="control-sidebar" :class="{ 'show-controlBar': controlBar }">
  58. <div class="inner-wrapper">
  59. <div v-if="isOwner()">
  60. <a class="sidebar-item" href="#" @click="settings()">
  61. <span class="icon">
  62. <i class="material-icons">settings</i>
  63. </span>
  64. <span class="icon-purpose">Station settings</span>
  65. </a>
  66. <a
  67. v-if="!noSong"
  68. class="sidebar-item"
  69. href="#"
  70. @click="$parent.skipStation()"
  71. >
  72. <span class="icon">
  73. <i class="material-icons">skip_next</i>
  74. </span>
  75. <span class="icon-purpose">Skip current song</span>
  76. </a>
  77. <a
  78. v-if="paused"
  79. class="sidebar-item"
  80. href="#"
  81. @click="$parent.resumeStation()"
  82. >
  83. <span class="icon">
  84. <i class="material-icons">play_arrow</i>
  85. </span>
  86. <span class="icon-purpose">Resume station</span>
  87. </a>
  88. <a
  89. v-if="!paused && !noSong"
  90. class="sidebar-item"
  91. href="#"
  92. @click="$parent.pauseStation()"
  93. >
  94. <span class="icon">
  95. <i class="material-icons">pause</i>
  96. </span>
  97. <span class="icon-purpose">Pause station</span>
  98. </a>
  99. <hr />
  100. </div>
  101. <div v-if="loggedIn">
  102. <a
  103. v-if="station.type === 'official'"
  104. class="sidebar-item"
  105. href="#"
  106. @click="
  107. openModal({
  108. sector: 'station',
  109. modal: 'addSongToQueue'
  110. })
  111. "
  112. >
  113. <span class="icon">
  114. <i class="material-icons">queue</i>
  115. </span>
  116. <span class="icon-purpose">Add song to queue</span>
  117. </a>
  118. <a
  119. v-if="!isOwner() && !noSong"
  120. class="sidebar-item"
  121. href="#"
  122. @click="$parent.voteSkipStation()"
  123. >
  124. <span class="icon">
  125. <i class="material-icons">skip_next</i>
  126. </span>
  127. <span class="skip-votes">{{
  128. currentSong.skipVotes
  129. }}</span>
  130. <span class="icon-purpose">Skip current song</span>
  131. </a>
  132. <a
  133. v-if="!noSong && !currentSong.simpleSong"
  134. class="sidebar-item"
  135. href="#"
  136. @click="
  137. openModal({
  138. sector: 'station',
  139. modal: 'report'
  140. })
  141. "
  142. >
  143. <span class="icon">
  144. <i class="material-icons">report</i>
  145. </span>
  146. <span class="icon-purpose">Report a song</span>
  147. </a>
  148. <a
  149. v-if="!noSong"
  150. class="sidebar-item"
  151. href="#"
  152. @click="
  153. openModal({
  154. sector: 'station',
  155. modal: 'addSongToPlaylist'
  156. })
  157. "
  158. >
  159. <span class="icon">
  160. <i class="material-icons">playlist_add</i>
  161. </span>
  162. <span class="icon-purpose"
  163. >Add current song to playlist</span
  164. >
  165. </a>
  166. <hr v-if="!noSong" />
  167. </div>
  168. <a
  169. v-if="
  170. station.partyMode === true ||
  171. station.type === 'official'
  172. "
  173. class="sidebar-item"
  174. href="#"
  175. @click="$parent.toggleSidebar('songslist')"
  176. >
  177. <span class="icon">
  178. <i class="material-icons">queue_music</i>
  179. </span>
  180. <span class="icon-purpose">Show the station queue</span>
  181. </a>
  182. <a
  183. class="sidebar-item"
  184. href="#"
  185. @click="$parent.toggleSidebar('users')"
  186. >
  187. <span class="icon">
  188. <i class="material-icons">people</i>
  189. </span>
  190. <span class="icon-purpose"
  191. >Display users in the station</span
  192. >
  193. </a>
  194. <a
  195. v-if="loggedIn && station.type === 'community'"
  196. class="sidebar-item"
  197. href="#"
  198. @click="$parent.toggleSidebar('playlist')"
  199. >
  200. <span class="icon">
  201. <i class="material-icons">library_music</i>
  202. </span>
  203. <span class="icon-purpose">Show your playlists</span>
  204. </a>
  205. </div>
  206. </div>
  207. </div>
  208. </template>
  209. <script>
  210. import { mapState, mapActions } from "vuex";
  211. export default {
  212. data() {
  213. return {
  214. title: this.$route.params.id,
  215. isMobile: false,
  216. controlBar: false,
  217. frontendDomain: "",
  218. siteSettings: {
  219. logo: "",
  220. siteName: ""
  221. }
  222. };
  223. },
  224. computed: mapState({
  225. loggedIn: state => state.user.auth.loggedIn,
  226. userId: state => state.user.auth.userId,
  227. username: state => state.user.auth.username,
  228. role: state => state.user.auth.role,
  229. station: state => state.station.station,
  230. paused: state => state.station.paused,
  231. noSong: state => state.station.noSong,
  232. currentSong: state => state.station.currentSong
  233. }),
  234. mounted() {
  235. lofig.get("frontendDomain", res => {
  236. this.frontendDomain = res;
  237. return res;
  238. });
  239. lofig.get("siteSettings", res => {
  240. this.siteSettings = res;
  241. return res;
  242. });
  243. },
  244. methods: {
  245. isOwner() {
  246. return (
  247. this.loggedIn &&
  248. (this.role === "admin" || this.userId === this.station.owner)
  249. );
  250. },
  251. settings() {
  252. this.editStation({
  253. _id: this.station._id,
  254. name: this.station.name,
  255. type: this.station.type,
  256. partyMode: this.station.partyMode,
  257. description: this.station.description,
  258. privacy: this.station.privacy,
  259. displayName: this.station.displayName
  260. });
  261. this.openModal({
  262. sector: "station",
  263. modal: "editStation"
  264. });
  265. },
  266. ...mapActions("modals", ["openModal"]),
  267. ...mapActions("station", ["editStation"]),
  268. ...mapActions("user/auth", ["logout"])
  269. }
  270. };
  271. </script>
  272. <style lang="scss" scoped>
  273. @import "styles/global.scss";
  274. .nav {
  275. background-color: $primary-color;
  276. line-height: 64px;
  277. border-radius: 0% 0% 33% 33% / 0% 0% 7% 7%;
  278. .is-brand {
  279. font-size: 2.1rem !important;
  280. line-height: 38px !important;
  281. padding: 0 20px;
  282. color: $white;
  283. font-family: Pacifico, cursive;
  284. filter: brightness(0) invert(1);
  285. img {
  286. max-height: 38px;
  287. }
  288. }
  289. }
  290. a.nav-item {
  291. color: $white;
  292. font-size: 17px;
  293. &:hover {
  294. color: $white;
  295. }
  296. padding: 0 12px;
  297. .icon {
  298. height: 64px;
  299. i {
  300. font-size: 2rem;
  301. line-height: 64px;
  302. height: 64px;
  303. width: 34px;
  304. }
  305. }
  306. }
  307. a.nav-item.is-tab:hover {
  308. border-bottom: none;
  309. border-top: solid 1px $white;
  310. }
  311. .admin strong {
  312. color: $purple;
  313. }
  314. .grouped {
  315. margin: 0;
  316. display: flex;
  317. text-decoration: none;
  318. }
  319. .skip-votes {
  320. position: relative;
  321. left: 11px;
  322. }
  323. .nav-toggle {
  324. height: 64px;
  325. }
  326. @media screen and (max-width: 998px) {
  327. .nav-menu {
  328. background-color: $white;
  329. box-shadow: 0 4px 7px rgba(10, 10, 10, 0.1);
  330. left: 0;
  331. display: none;
  332. right: 0;
  333. top: 100%;
  334. position: absolute;
  335. }
  336. .nav-toggle {
  337. display: block;
  338. }
  339. }
  340. .logo {
  341. font-size: 2.1rem;
  342. line-height: 64px;
  343. padding-left: 20px !important;
  344. padding-right: 20px !important;
  345. }
  346. .nav-center {
  347. display: flex;
  348. align-items: center;
  349. color: $primary-color;
  350. font-size: 22px;
  351. position: absolute;
  352. margin: auto;
  353. top: 50%;
  354. left: 50%;
  355. transform: translate(-50%, -50%);
  356. }
  357. .nav-right.is-active .nav-item {
  358. background: $primary-color;
  359. border: 0;
  360. }
  361. .hidden {
  362. display: none;
  363. }
  364. .control-sidebar {
  365. position: fixed;
  366. z-index: 1;
  367. top: 0;
  368. left: 0;
  369. width: 64px;
  370. height: 100vh;
  371. background-color: $primary-color;
  372. box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.16),
  373. 0 2px 10px 0 rgba(0, 0, 0, 0.12);
  374. @media (max-width: 998px) {
  375. display: none;
  376. }
  377. .inner-wrapper {
  378. @media (min-width: 999px) {
  379. .mobile-only {
  380. display: none;
  381. }
  382. .desktop-only {
  383. display: flex;
  384. }
  385. }
  386. @media (max-width: 998px) {
  387. .mobile-only {
  388. display: flex;
  389. }
  390. .desktop-only {
  391. display: none;
  392. visibility: hidden;
  393. }
  394. }
  395. }
  396. }
  397. .show-controlBar {
  398. display: block;
  399. }
  400. .inner-wrapper {
  401. top: 64px;
  402. position: relative;
  403. }
  404. .control-sidebar .material-icons {
  405. width: 100%;
  406. font-size: 2rem;
  407. }
  408. .control-sidebar .sidebar-item {
  409. font-size: 2rem;
  410. height: 50px;
  411. color: $white;
  412. -webkit-box-align: center;
  413. -ms-flex-align: center;
  414. align-items: center;
  415. display: -webkit-box;
  416. display: -ms-flexbox;
  417. display: flex;
  418. -webkit-box-flex: 0;
  419. -ms-flex-positive: 0;
  420. flex-grow: 0;
  421. -ms-flex-negative: 0;
  422. flex-shrink: 0;
  423. -webkit-box-pack: center;
  424. -ms-flex-pack: center;
  425. justify-content: center;
  426. width: 100%;
  427. position: relative;
  428. }
  429. .control-sidebar .sidebar-top-hr {
  430. margin: 0 0 20px 0;
  431. }
  432. .sidebar-item .icon-purpose {
  433. visibility: hidden;
  434. width: 160px;
  435. font-size: 12px;
  436. background-color: rgba(3, 169, 244, 0.8);
  437. color: $white;
  438. text-align: center;
  439. border-radius: 6px;
  440. padding: 5px;
  441. position: absolute;
  442. z-index: 1;
  443. left: 115%;
  444. opacity: 0;
  445. transition: opacity 0.5s;
  446. display: none;
  447. }
  448. .sidebar-item .icon-purpose::after {
  449. content: "";
  450. position: absolute;
  451. top: 50%;
  452. right: 100%;
  453. margin-top: -5px;
  454. border-width: 5px;
  455. border-style: solid;
  456. border-color: transparent rgba(3, 169, 244, 0.8) transparent transparent;
  457. }
  458. .sidebar-item:hover .icon-purpose {
  459. visibility: visible;
  460. opacity: 1;
  461. display: block;
  462. }
  463. </style>