index.vue 24 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150
  1. <template>
  2. <div>
  3. <metadata title="Home" />
  4. <div class="app">
  5. <main-header
  6. :hide-logo="true"
  7. :transparent="true"
  8. :hide-logged-out="true"
  9. />
  10. <div class="header" :class="{ loggedIn }">
  11. <img class="background" src="/assets/homebg.jpeg" />
  12. <div class="overlay"></div>
  13. <div class="content-container">
  14. <div class="content">
  15. <img
  16. class="logo"
  17. src="/assets/white_wordmark.png"
  18. :alt="`${this.siteName}` || `Musare`"
  19. />
  20. <div v-if="!loggedIn" class="buttons">
  21. <button
  22. class="button login"
  23. @click="
  24. openModal({
  25. sector: 'header',
  26. modal: 'login'
  27. })
  28. "
  29. >
  30. Login
  31. </button>
  32. <button
  33. class="button register"
  34. @click="
  35. openModal({
  36. sector: 'header',
  37. modal: 'register'
  38. })
  39. "
  40. >
  41. Register
  42. </button>
  43. </div>
  44. </div>
  45. </div>
  46. </div>
  47. <div v-if="favoriteStations.length > 0" class="group">
  48. <div class="group-title">
  49. <div>
  50. <h2>My Favorites</h2>
  51. </div>
  52. </div>
  53. <router-link
  54. v-for="(station, index) in favoriteStations"
  55. :key="index"
  56. :to="{
  57. name: 'station',
  58. params: { id: station.name }
  59. }"
  60. class="card station-card"
  61. :class="{
  62. isPrivate: station.privacy === 'private',
  63. isMine: isOwner(station)
  64. }"
  65. :style="'--station-theme: var(--' + station.theme + ')'"
  66. >
  67. <div class="card-image">
  68. <figure class="image is-square">
  69. <div
  70. v-if="station.currentSong.ytThumbnail"
  71. class="ytThumbnailBg"
  72. :style="{
  73. 'background-image':
  74. 'url(' +
  75. station.currentSong.ytThumbnail +
  76. ')'
  77. }"
  78. ></div>
  79. <img
  80. v-if="station.currentSong.ytThumbnail"
  81. :src="station.currentSong.ytThumbnail"
  82. onerror="this.src='/assets/notes-transparent.png'"
  83. />
  84. <img
  85. v-else
  86. :src="station.currentSong.thumbnail"
  87. onerror="this.src='/assets/notes-transparent.png'"
  88. />
  89. </figure>
  90. </div>
  91. <div class="card-content">
  92. <div class="media">
  93. <div class="media-left displayName">
  94. <i
  95. v-if="loggedIn && !station.isFavorited"
  96. @click.prevent="favoriteStation(station)"
  97. class="favorite material-icons"
  98. >star_border</i
  99. >
  100. <i
  101. v-if="loggedIn && station.isFavorited"
  102. @click.prevent="unfavoriteStation(station)"
  103. class="favorite material-icons"
  104. >star</i
  105. >
  106. <h5>{{ station.displayName }}</h5>
  107. <i
  108. v-if="station.type === 'official'"
  109. class="material-icons verified-station"
  110. title="Verified station"
  111. >
  112. check_circle
  113. </i>
  114. </div>
  115. </div>
  116. <div class="content">
  117. {{ station.description }}
  118. </div>
  119. <div class="under-content">
  120. <p class="hostedBy">
  121. Hosted by
  122. <span class="host">
  123. <span
  124. v-if="station.type === 'official'"
  125. title="Musare"
  126. >Musare</span
  127. >
  128. <user-id-to-username
  129. v-else
  130. :user-id="station.owner"
  131. :link="true"
  132. />
  133. </span>
  134. </p>
  135. <div class="icons">
  136. <i
  137. v-if="
  138. station.type === 'community' &&
  139. isOwner(station)
  140. "
  141. class="homeIcon material-icons"
  142. title="This is your station."
  143. >home</i
  144. >
  145. <i
  146. v-if="station.privacy === 'private'"
  147. class="privateIcon material-icons"
  148. title="This station is not visible to other users."
  149. >lock</i
  150. >
  151. <i
  152. v-if="station.privacy === 'unlisted'"
  153. class="unlistedIcon material-icons"
  154. title="Unlisted Station"
  155. >link</i
  156. >
  157. </div>
  158. </div>
  159. </div>
  160. <div class="bottomBar">
  161. <i
  162. v-if="station.paused && station.currentSong.title"
  163. class="material-icons"
  164. title="Station Paused"
  165. >pause</i
  166. >
  167. <i
  168. v-else-if="station.currentSong.title"
  169. class="material-icons"
  170. >music_note</i
  171. >
  172. <i v-else class="material-icons">music_off</i>
  173. <span
  174. v-if="station.currentSong.title"
  175. class="songTitle"
  176. :title="
  177. station.currentSong.artists.length > 0
  178. ? 'Now Playing: ' +
  179. station.currentSong.title +
  180. ' by ' +
  181. station.currentSong.artists.join(',')
  182. : 'Now Playing: ' +
  183. station.currentSong.title
  184. "
  185. >{{ station.currentSong.title }}
  186. {{
  187. station.currentSong.artists.length > 0
  188. ? " by " +
  189. station.currentSong.artists.join(",")
  190. : ""
  191. }}</span
  192. >
  193. <span v-else class="songTitle">No Songs Playing</span>
  194. </div>
  195. </router-link>
  196. </div>
  197. <div class="group bottom">
  198. <div class="group-title">
  199. <div>
  200. <h1>Stations</h1>
  201. </div>
  202. </div>
  203. <a
  204. v-if="loggedIn"
  205. @click="
  206. openModal({
  207. sector: 'home',
  208. modal: 'createCommunityStation'
  209. })
  210. "
  211. class="card station-card createStation"
  212. >
  213. <div class="card-image">
  214. <figure class="image is-square">
  215. <i class="material-icons">radio</i>
  216. </figure>
  217. </div>
  218. <div class="card-content">
  219. <div class="media">
  220. <div class="media-left displayName">
  221. <h5>Create Station</h5>
  222. </div>
  223. </div>
  224. <div class="content">
  225. Click here to create your own station!
  226. </div>
  227. </div>
  228. <div class="bottomBar"></div>
  229. </a>
  230. <a v-else class="card station-card createStation">
  231. <div class="card-image">
  232. <figure class="image is-square">
  233. <i class="material-icons">radio</i>
  234. </figure>
  235. </div>
  236. <div class="card-content">
  237. <div class="media">
  238. <div class="media-left displayName">
  239. <h5>Create Station</h5>
  240. </div>
  241. </div>
  242. <div class="content">Login to create a station!</div>
  243. </div>
  244. <div class="bottomBar"></div>
  245. </a>
  246. <router-link
  247. v-for="(station, index) in filteredStations"
  248. :key="index"
  249. :to="{
  250. name: 'station',
  251. params: { id: station.name }
  252. }"
  253. class="card station-card"
  254. :class="{
  255. isPrivate: station.privacy === 'private',
  256. isMine: isOwner(station)
  257. }"
  258. :style="'--station-theme: var(--' + station.theme + ')'"
  259. >
  260. <div class="card-image">
  261. <figure class="image is-square">
  262. <div
  263. v-if="station.currentSong.ytThumbnail"
  264. class="ytThumbnailBg"
  265. :style="{
  266. 'background-image':
  267. 'url(' +
  268. station.currentSong.ytThumbnail +
  269. ')'
  270. }"
  271. ></div>
  272. <img
  273. v-if="station.currentSong.ytThumbnail"
  274. :src="station.currentSong.ytThumbnail"
  275. onerror="this.src='/assets/notes-transparent.png'"
  276. />
  277. <img
  278. v-else
  279. :src="station.currentSong.thumbnail"
  280. onerror="this.src='/assets/notes-transparent.png'"
  281. />
  282. </figure>
  283. </div>
  284. <div class="card-content">
  285. <div class="media">
  286. <div class="media-left displayName">
  287. <i
  288. v-if="loggedIn && !station.isFavorited"
  289. @click.prevent="favoriteStation(station)"
  290. class="favorite material-icons"
  291. >star_border</i
  292. >
  293. <i
  294. v-if="loggedIn && station.isFavorited"
  295. @click.prevent="unfavoriteStation(station)"
  296. class="favorite material-icons"
  297. >star</i
  298. >
  299. <h5>{{ station.displayName }}</h5>
  300. <i
  301. v-if="station.type === 'official'"
  302. class="material-icons verified-station"
  303. title="Verified station"
  304. >
  305. check_circle
  306. </i>
  307. </div>
  308. </div>
  309. <div class="content">
  310. {{ station.description }}
  311. </div>
  312. <div class="under-content">
  313. <p class="hostedBy">
  314. Hosted by
  315. <span class="host">
  316. <span
  317. v-if="station.type === 'official'"
  318. title="Musare"
  319. >Musare</span
  320. >
  321. <user-id-to-username
  322. v-else
  323. :user-id="station.owner"
  324. :link="true"
  325. />
  326. </span>
  327. </p>
  328. <div class="icons">
  329. <i
  330. v-if="
  331. station.type === 'community' &&
  332. isOwner(station)
  333. "
  334. class="homeIcon material-icons"
  335. title="This is your station."
  336. >home</i
  337. >
  338. <i
  339. v-if="station.privacy === 'private'"
  340. class="privateIcon material-icons"
  341. title="This station is not visible to other users."
  342. >lock</i
  343. >
  344. <i
  345. v-if="station.privacy === 'unlisted'"
  346. class="unlistedIcon material-icons"
  347. title="Unlisted Station"
  348. >link</i
  349. >
  350. </div>
  351. </div>
  352. </div>
  353. <div class="bottomBar">
  354. <i
  355. v-if="station.paused && station.currentSong.title"
  356. class="material-icons"
  357. title="Station Paused"
  358. >pause</i
  359. >
  360. <i
  361. v-else-if="station.currentSong.title"
  362. class="material-icons"
  363. >music_note</i
  364. >
  365. <i v-else class="material-icons">music_off</i>
  366. <span
  367. v-if="station.currentSong.title"
  368. class="songTitle"
  369. :title="
  370. station.currentSong.artists.length > 0
  371. ? 'Now Playing: ' +
  372. station.currentSong.title +
  373. ' by ' +
  374. station.currentSong.artists.join(',')
  375. : 'Now Playing: ' +
  376. station.currentSong.title
  377. "
  378. >{{ station.currentSong.title }}
  379. {{
  380. station.currentSong.artists.length > 0
  381. ? " by " +
  382. station.currentSong.artists.join(",")
  383. : ""
  384. }}</span
  385. >
  386. <span v-else class="songTitle">No Songs Playing</span>
  387. </div>
  388. </router-link>
  389. <h4 v-if="stations.length === 0">
  390. There are no stations to display
  391. </h4>
  392. </div>
  393. <main-footer />
  394. </div>
  395. <create-community-station v-if="modals.createCommunityStation" />
  396. </div>
  397. </template>
  398. <script>
  399. import { mapState, mapActions } from "vuex";
  400. import Toast from "toasters";
  401. import MainHeader from "../../components/layout/MainHeader.vue";
  402. import MainFooter from "../../components/layout/MainFooter.vue";
  403. import CreateCommunityStation from "./CreateCommunityStation.vue";
  404. import UserIdToUsername from "../../components/common/UserIdToUsername.vue";
  405. import io from "../../io";
  406. export default {
  407. components: {
  408. MainHeader,
  409. MainFooter,
  410. CreateCommunityStation,
  411. UserIdToUsername
  412. },
  413. data() {
  414. return {
  415. recaptcha: {
  416. key: ""
  417. },
  418. stations: [],
  419. searchQuery: "",
  420. siteName: "Musare"
  421. };
  422. },
  423. computed: {
  424. ...mapState({
  425. loggedIn: state => state.user.auth.loggedIn,
  426. userId: state => state.user.auth.userId,
  427. modals: state => state.modalVisibility.modals.home
  428. }),
  429. filteredStations() {
  430. const privacyOrder = ["public", "unlisted", "private"];
  431. return this.stations
  432. .filter(
  433. station =>
  434. JSON.stringify(Object.values(station)).indexOf(
  435. this.searchQuery
  436. ) !== -1
  437. )
  438. .sort(
  439. (a, b) =>
  440. this.isOwner(b) - this.isOwner(a) ||
  441. this.isPlaying(b) - this.isPlaying(a) ||
  442. a.paused - b.paused ||
  443. privacyOrder.indexOf(a.privacy) -
  444. privacyOrder.indexOf(b.privacy) ||
  445. b.userCount - a.userCount
  446. );
  447. },
  448. favoriteStations() {
  449. return this.filteredStations.filter(
  450. station => station.isFavorited === true
  451. );
  452. }
  453. },
  454. async mounted() {
  455. this.siteName = await lofig.get("siteSettings.siteName");
  456. io.getSocket(socket => {
  457. this.socket = socket;
  458. if (this.socket.connected) this.init();
  459. io.onConnect(() => this.init());
  460. this.socket.on("event:stations.created", res => {
  461. const station = res;
  462. if (
  463. this.stations.find(_station => _station._id === station._id)
  464. ) {
  465. this.stations.forEach(s => {
  466. const _station = s;
  467. if (_station._id === station._id) {
  468. _station.privacy = station.privacy;
  469. }
  470. });
  471. } else {
  472. if (!station.currentSong)
  473. station.currentSong = {
  474. thumbnail: "/assets/notes-transparent.png"
  475. };
  476. if (station.currentSong && !station.currentSong.thumbnail)
  477. station.currentSong.ytThumbnail = `https://img.youtube.com/vi/${station.currentSong.songId}/mqdefault.jpg`;
  478. this.stations.push(station);
  479. }
  480. });
  481. this.socket.on("event:station.removed", response => {
  482. const { stationId } = response;
  483. const station = this.stations.find(
  484. station => station._id === stationId
  485. );
  486. if (station) {
  487. const stationIndex = this.stations.indexOf(station);
  488. this.stations.splice(stationIndex, 1);
  489. }
  490. });
  491. this.socket.on(
  492. "event:userCount.updated",
  493. (stationId, userCount) => {
  494. this.stations.forEach(s => {
  495. const station = s;
  496. if (station._id === stationId) {
  497. station.userCount = userCount;
  498. }
  499. });
  500. }
  501. );
  502. this.socket.on("event:station.updatePrivacy", response => {
  503. const { stationId, privacy } = response;
  504. this.stations.forEach(s => {
  505. const station = s;
  506. if (station._id === stationId) {
  507. station.privacy = privacy;
  508. }
  509. });
  510. });
  511. this.socket.on("event:station.updateName", response => {
  512. const { stationId, name } = response;
  513. this.stations.forEach(s => {
  514. const station = s;
  515. if (station._id === stationId) {
  516. station.name = name;
  517. }
  518. });
  519. });
  520. this.socket.on("event:station.updateDisplayName", response => {
  521. const { stationId, displayName } = response;
  522. this.stations.forEach(s => {
  523. const station = s;
  524. if (station._id === stationId) {
  525. station.displayName = displayName;
  526. }
  527. });
  528. });
  529. this.socket.on("event:station.updateDescription", response => {
  530. const { stationId, description } = response;
  531. this.stations.forEach(s => {
  532. const station = s;
  533. if (station._id === stationId) {
  534. station.description = description;
  535. }
  536. });
  537. });
  538. this.socket.on("event:station.updateTheme", response => {
  539. const { stationId, theme } = response;
  540. this.stations.forEach(s => {
  541. const station = s;
  542. if (station._id === stationId) {
  543. station.theme = theme;
  544. }
  545. });
  546. });
  547. this.socket.on("event:station.nextSong", (stationId, song) => {
  548. let newSong = song;
  549. this.stations.forEach(s => {
  550. const station = s;
  551. if (station._id === stationId) {
  552. if (!newSong)
  553. newSong = {
  554. thumbnail: "/assets/notes-transparent.png"
  555. };
  556. if (newSong && !newSong.thumbnail)
  557. newSong.ytThumbnail = `https://img.youtube.com/vi/${newSong.songId}/mqdefault.jpg`;
  558. station.currentSong = newSong;
  559. }
  560. });
  561. });
  562. this.socket.on("event:station.pause", response => {
  563. const { stationId } = response;
  564. this.stations.forEach(s => {
  565. const station = s;
  566. if (station._id === stationId) {
  567. station.paused = true;
  568. }
  569. });
  570. });
  571. this.socket.on("event:station.resume", response => {
  572. const { stationId } = response;
  573. this.stations.forEach(s => {
  574. const station = s;
  575. if (station._id === stationId) {
  576. station.paused = false;
  577. }
  578. });
  579. });
  580. this.socket.on("event:user.favoritedStation", stationId => {
  581. this.stations.forEach(s => {
  582. const station = s;
  583. if (station._id === stationId) {
  584. station.isFavorited = true;
  585. }
  586. });
  587. });
  588. this.socket.on("event:user.unfavoritedStation", stationId => {
  589. this.stations.forEach(s => {
  590. const station = s;
  591. if (station._id === stationId) {
  592. station.isFavorited = false;
  593. }
  594. });
  595. });
  596. });
  597. },
  598. methods: {
  599. init() {
  600. this.socket.emit("stations.index", data => {
  601. this.stations = [];
  602. if (data.status === "success")
  603. data.stations.forEach(station => {
  604. const modifiableStation = station;
  605. if (!modifiableStation.currentSong)
  606. modifiableStation.currentSong = {
  607. thumbnail: "/assets/notes-transparent.png"
  608. };
  609. if (
  610. modifiableStation.currentSong &&
  611. !modifiableStation.currentSong.thumbnail
  612. )
  613. modifiableStation.currentSong.ytThumbnail = `https://img.youtube.com/vi/${station.currentSong.songId}/mqdefault.jpg`;
  614. this.stations.push(modifiableStation);
  615. });
  616. });
  617. this.socket.emit("apis.joinRoom", "home", () => {});
  618. },
  619. isOwner(station) {
  620. return station.owner === this.userId;
  621. },
  622. isPlaying(station) {
  623. return typeof station.currentSong.title !== "undefined";
  624. },
  625. favoriteStation(station) {
  626. this.socket.emit("stations.favoriteStation", station._id, res => {
  627. if (res.status === "success") {
  628. new Toast({
  629. content: "Successfully favorited station.",
  630. timeout: 4000
  631. });
  632. } else new Toast({ content: res.message, timeout: 8000 });
  633. });
  634. },
  635. unfavoriteStation(station) {
  636. this.socket.emit("stations.unfavoriteStation", station._id, res => {
  637. if (res.status === "success") {
  638. new Toast({
  639. content: "Successfully unfavorited station.",
  640. timeout: 4000
  641. });
  642. } else new Toast({ content: res.message, timeout: 8000 });
  643. });
  644. },
  645. ...mapActions("modalVisibility", ["openModal"]),
  646. ...mapActions("station", ["updateIfStationIsFavorited"])
  647. }
  648. };
  649. </script>
  650. <style lang="scss">
  651. * {
  652. box-sizing: border-box;
  653. }
  654. html {
  655. width: 100%;
  656. height: 100%;
  657. color: rgba(0, 0, 0, 0.87);
  658. body {
  659. width: 100%;
  660. height: 100%;
  661. margin: 0;
  662. padding: 0;
  663. }
  664. }
  665. .night-mode {
  666. .header .overlay {
  667. background: linear-gradient(
  668. 180deg,
  669. rgba(34, 34, 34, 0.8) 0%,
  670. rgba(34, 34, 34, 0.95) 31.25%,
  671. rgba(34, 34, 34, 0.9) 54.17%,
  672. rgba(34, 34, 34, 0.8) 100%
  673. );
  674. }
  675. .card,
  676. .card-content,
  677. .card-content div {
  678. background-color: var(--dark-grey-3);
  679. }
  680. .card-content .icons i,
  681. .group-title i {
  682. color: var(--light-grey-2);
  683. }
  684. .card-image .image {
  685. background-color: var(--dark-grey-2);
  686. }
  687. .card-content .under-content .hostedBy {
  688. color: var(--light-grey-2);
  689. }
  690. }
  691. @media only screen and (min-width: 1200px) {
  692. html {
  693. font-size: 15px;
  694. }
  695. }
  696. @media only screen and (min-width: 992px) {
  697. html {
  698. font-size: 14.5px;
  699. }
  700. }
  701. @media only screen and (min-width: 0) {
  702. html {
  703. font-size: 14px;
  704. }
  705. }
  706. .header {
  707. display: flex;
  708. height: 35vh;
  709. margin-top: -64px;
  710. border-radius: 0% 0% 33% 33% / 0% 0% 7% 7%;
  711. img.background {
  712. height: 35vh;
  713. width: 100%;
  714. object-fit: cover;
  715. object-position: center;
  716. filter: blur(1px);
  717. border-radius: 0% 0% 33% 33% / 0% 0% 7% 7%;
  718. overflow: hidden;
  719. }
  720. .overlay {
  721. background: linear-gradient(
  722. 180deg,
  723. rgba(3, 169, 244, 0.8) 0%,
  724. rgba(3, 169, 244, 0.95) 31.25%,
  725. rgba(3, 169, 244, 0.9) 54.17%,
  726. rgba(3, 169, 244, 0.8) 100%
  727. );
  728. position: absolute;
  729. height: 35vh;
  730. width: 100%;
  731. border-radius: 0% 0% 33% 33% / 0% 0% 7% 7%;
  732. overflow: hidden;
  733. }
  734. .content-container {
  735. position: absolute;
  736. left: 0;
  737. right: 0;
  738. margin-left: auto;
  739. margin-right: auto;
  740. text-align: center;
  741. height: 100%;
  742. height: 35vh;
  743. .content {
  744. position: absolute;
  745. top: 50%;
  746. left: 0;
  747. right: 0;
  748. transform: translateY(-50%);
  749. background-color: transparent !important;
  750. img.logo {
  751. max-height: 90px;
  752. font-size: 40px;
  753. color: var(--white);
  754. font-family: Pacifico, cursive;
  755. }
  756. .buttons {
  757. display: flex;
  758. justify-content: center;
  759. margin-top: 20px;
  760. flex-wrap: wrap;
  761. .login,
  762. .register {
  763. margin: 5px 10px;
  764. padding: 10px 15px;
  765. border-radius: 5px;
  766. font-size: 18px;
  767. width: 100%;
  768. max-width: 250px;
  769. font-weight: 600;
  770. border: 0;
  771. height: inherit;
  772. }
  773. .login {
  774. background: var(--white);
  775. color: var(--primary-color);
  776. }
  777. .register {
  778. background: var(--purple);
  779. color: var(--white);
  780. }
  781. }
  782. }
  783. }
  784. &.loggedIn {
  785. height: 20vh;
  786. .overlay,
  787. .content-container,
  788. img.background {
  789. height: 20vh;
  790. }
  791. }
  792. }
  793. @media only screen and (max-width: 550px) {
  794. .header {
  795. height: 45vh;
  796. .overlay,
  797. .content-container,
  798. img.background {
  799. height: 45vh;
  800. }
  801. }
  802. }
  803. .under-content {
  804. height: 20px;
  805. position: relative;
  806. line-height: 1;
  807. font-size: 24px;
  808. display: flex;
  809. align-items: center;
  810. text-align: left;
  811. margin-top: 10px;
  812. p {
  813. font-size: 15px;
  814. line-height: 15px;
  815. display: inline;
  816. }
  817. i {
  818. font-size: 20px;
  819. }
  820. * {
  821. z-index: 10;
  822. position: relative;
  823. }
  824. .icons {
  825. position: absolute;
  826. right: 0;
  827. .material-icons {
  828. font-size: 22px;
  829. }
  830. .material-icons:first-child {
  831. margin-left: 5px;
  832. }
  833. .unlistedIcon {
  834. color: var(--orange);
  835. }
  836. .privateIcon {
  837. color: var(--dark-pink);
  838. }
  839. .homeIcon {
  840. color: var(--light-purple);
  841. }
  842. }
  843. .hostedBy {
  844. font-weight: 400;
  845. font-size: 12px;
  846. color: var(--black);
  847. .host,
  848. .host a {
  849. font-weight: 400;
  850. color: var(--primary-color);
  851. &:hover,
  852. &:focus {
  853. filter: brightness(90%);
  854. }
  855. }
  856. }
  857. }
  858. .app {
  859. display: flex;
  860. flex-direction: column;
  861. }
  862. .users-count {
  863. font-size: 20px;
  864. position: relative;
  865. top: -4px;
  866. }
  867. .group {
  868. min-height: 64px;
  869. flex: 1 0 auto;
  870. }
  871. .station-card {
  872. display: inline-flex;
  873. flex-direction: row;
  874. overflow: hidden;
  875. margin: 10px;
  876. cursor: pointer;
  877. height: 150px;
  878. width: calc(100% - 30px);
  879. max-width: 400px;
  880. flex-wrap: wrap;
  881. border-radius: 5px;
  882. box-shadow: 0 2px 3px rgba(10, 10, 10, 0.1), 0 0 0 1px rgba(10, 10, 10, 0.1);
  883. transition: all ease-in-out 0.2s;
  884. --primary-color: var(--station-theme);
  885. .card-content {
  886. padding: 10px 10px 10px 15px;
  887. display: flex;
  888. flex-direction: column;
  889. flex-grow: 1;
  890. -webkit-line-clamp: 2;
  891. .media {
  892. display: flex;
  893. align-items: center;
  894. margin-bottom: 0;
  895. .displayName {
  896. display: flex;
  897. align-items: center;
  898. width: 100%;
  899. overflow: hidden;
  900. text-overflow: ellipsis;
  901. display: flex;
  902. line-height: 30px;
  903. max-height: 30px;
  904. .favorite {
  905. position: absolute;
  906. color: var(--yellow);
  907. right: 10px;
  908. top: 10px;
  909. font-size: 28px;
  910. }
  911. h5 {
  912. font-size: 20px;
  913. font-weight: 400;
  914. margin: 0;
  915. display: inline;
  916. margin-right: 6px;
  917. line-height: 30px;
  918. text-overflow: ellipsis;
  919. overflow: hidden;
  920. white-space: nowrap;
  921. max-width: 200px;
  922. }
  923. i {
  924. font-size: 22px;
  925. }
  926. .verified-station {
  927. color: var(--primary-color);
  928. }
  929. }
  930. }
  931. .content {
  932. word-wrap: break-word;
  933. overflow: hidden;
  934. text-overflow: ellipsis;
  935. display: -webkit-box;
  936. -webkit-box-orient: vertical;
  937. -webkit-line-clamp: 3;
  938. line-height: 20px;
  939. flex-grow: 1;
  940. text-align: left;
  941. word-wrap: break-word;
  942. margin-bottom: 0;
  943. }
  944. }
  945. .card-image {
  946. width: 120px;
  947. .image {
  948. box-shadow: 1px 0 3px rgba(100, 100, 100, 0.3);
  949. .ytThumbnailBg {
  950. background: url("/assets/notes-transparent.png") no-repeat
  951. center center;
  952. background-size: cover;
  953. height: 100%;
  954. width: 100%;
  955. position: absolute;
  956. top: 0;
  957. filter: blur(1px);
  958. }
  959. img {
  960. height: auto;
  961. width: 120px;
  962. top: 0;
  963. margin-top: auto;
  964. margin-bottom: auto;
  965. z-index: 1;
  966. }
  967. }
  968. }
  969. .bottomBar {
  970. position: relative;
  971. display: flex;
  972. align-items: center;
  973. background: var(--primary-color);
  974. // box-shadow: inset 0px 2px 4px rgba(100, 100, 100, 0.3);
  975. width: 100%;
  976. height: 30px;
  977. line-height: 30px;
  978. color: var(--white);
  979. font-weight: 400;
  980. font-size: 12px;
  981. padding: 0 5px;
  982. flex-basis: 100%;
  983. i.material-icons {
  984. vertical-align: middle;
  985. margin-left: 5px;
  986. font-size: 18px;
  987. }
  988. .songTitle {
  989. text-align: left;
  990. vertical-align: middle;
  991. margin-left: 5px;
  992. line-height: 30px;
  993. flex: 2 1 0;
  994. overflow: hidden;
  995. text-overflow: ellipsis;
  996. white-space: nowrap;
  997. }
  998. }
  999. &.createStation {
  1000. .card-image .image.is-square .material-icons {
  1001. position: absolute;
  1002. top: 25px;
  1003. bottom: 25px;
  1004. left: 0;
  1005. right: 0;
  1006. text-align: center;
  1007. font-size: 70px;
  1008. color: var(--primary-color);
  1009. }
  1010. .card-content {
  1011. .media {
  1012. margin-top: auto;
  1013. .displayName h5 {
  1014. font-weight: 600;
  1015. }
  1016. }
  1017. .content {
  1018. flex-grow: unset;
  1019. margin-bottom: auto;
  1020. }
  1021. }
  1022. }
  1023. }
  1024. .station-card:hover {
  1025. box-shadow: 0 2px 3px rgba(10, 10, 10, 0.3), 0 0 10px rgba(10, 10, 10, 0.3);
  1026. transition: all ease-in-out 0.2s;
  1027. }
  1028. .community-button {
  1029. cursor: pointer;
  1030. transition: 0.25s ease color;
  1031. font-size: 30px;
  1032. color: var(--dark-grey);
  1033. }
  1034. .community-button:hover {
  1035. color: var(--primary-color);
  1036. }
  1037. .station-privacy {
  1038. text-transform: capitalize;
  1039. }
  1040. .label {
  1041. display: flex;
  1042. }
  1043. .g-recaptcha {
  1044. display: flex;
  1045. justify-content: center;
  1046. margin-top: 20px;
  1047. }
  1048. .group {
  1049. text-align: center;
  1050. width: 100%;
  1051. margin: 10px 0;
  1052. .group-title {
  1053. display: flex;
  1054. align-items: center;
  1055. justify-content: center;
  1056. margin: 25px 0;
  1057. h1 {
  1058. display: inline-block;
  1059. font-size: 45px;
  1060. margin: 0;
  1061. }
  1062. h2 {
  1063. font-size: 35px;
  1064. margin: 0;
  1065. }
  1066. a {
  1067. display: flex;
  1068. margin-left: 8px;
  1069. }
  1070. }
  1071. &.bottom {
  1072. margin-bottom: 40px;
  1073. }
  1074. }
  1075. </style>