index.vue 54 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325
  1. <template>
  2. <div>
  3. <metadata v-if="exists && !loading" :title="`${station.displayName}`" />
  4. <metadata v-else-if="!exists && !loading" :title="`Not found`" />
  5. <div id="page-loader-container" v-if="loading">
  6. <content-loader
  7. width="1920"
  8. height="1080"
  9. :primary-color="nightmode ? '#222' : '#fff'"
  10. :secondary-color="nightmode ? '#444' : '#ddd'"
  11. preserve-aspect-ratio="none"
  12. id="page-loader-content"
  13. >
  14. <rect x="100" y="108" rx="5" ry="5" width="1048" height="672" />
  15. <rect x="100" y="810" rx="5" ry="5" width="1048" height="110" />
  16. <rect x="1190" y="110" rx="5" ry="5" width="630" height="149" />
  17. <rect x="1190" y="288" rx="5" ry="5" width="630" height="630" />
  18. </content-loader>
  19. <content-loader
  20. width="1920"
  21. height="1080"
  22. :primary-color="nightmode ? '#222' : '#fff'"
  23. :secondary-color="nightmode ? '#444' : '#ddd'"
  24. preserve-aspect-ratio="none"
  25. id="page-loader-layout"
  26. >
  27. <rect x="0" y="0" rx="0" ry="0" width="1920" height="64" />
  28. <rect x="0" y="980" rx="0" ry="0" width="1920" height="100" />
  29. </content-loader>
  30. </div>
  31. <!-- More simplistic loading animation for mobile users -->
  32. <div v-show="loading" id="mobile-progress-animation" />
  33. <div v-show="!loading">
  34. <main-header v-if="exists" />
  35. <div
  36. id="station-outer-container"
  37. :style="[!exists ? { margin: 0, padding: 0 } : {}]"
  38. >
  39. <div
  40. v-show="exists"
  41. id="station-inner-container"
  42. :class="{ 'nothing-here': noSong }"
  43. >
  44. <div id="station-left-column" class="column">
  45. <div class="player-container quadrant" v-show="!noSong">
  46. <div id="video-container">
  47. <div
  48. id="stationPlayer"
  49. style="
  50. width: 100%;
  51. height: 100%;
  52. min-height: 200px;
  53. "
  54. />
  55. <div
  56. class="player-cannot-autoplay"
  57. v-if="!canAutoplay"
  58. @click="
  59. increaseVolume() && decreaseVolume()
  60. "
  61. >
  62. <p>
  63. Please click anywhere on the screen for
  64. the video to start
  65. </p>
  66. </div>
  67. </div>
  68. <div id="seeker-bar-container">
  69. <div id="seeker-bar" style="width: 0%" />
  70. </div>
  71. <div id="control-bar-container">
  72. <div id="left-buttons">
  73. <!-- Debug Box -->
  74. <button
  75. :v-if="
  76. frontendDevMode === 'development'
  77. "
  78. class="button is-primary"
  79. @click="togglePlayerDebugBox()"
  80. @dblclick="resetPlayerDebugBox()"
  81. content="Debug"
  82. v-tippy
  83. >
  84. <i
  85. class="material-icons icon-with-button"
  86. >
  87. bug_report
  88. </i>
  89. </button>
  90. <!-- Local Pause/Resume Button -->
  91. <button
  92. class="button is-primary"
  93. @click="resumeLocalStation()"
  94. id="local-resume"
  95. v-if="localPaused"
  96. content="Unpause Playback"
  97. v-tippy
  98. >
  99. <i class="material-icons">play_arrow</i>
  100. </button>
  101. <button
  102. class="button is-primary"
  103. @click="pauseLocalStation()"
  104. id="local-pause"
  105. v-else
  106. content="Pause Playback"
  107. v-tippy
  108. >
  109. <i class="material-icons">pause</i>
  110. </button>
  111. <!-- Vote to Skip Button -->
  112. <button
  113. v-if="loggedIn"
  114. class="button is-primary"
  115. @click="voteSkipStation()"
  116. content="Vote to Skip Song"
  117. v-tippy
  118. >
  119. <i
  120. class="material-icons icon-with-button"
  121. >skip_next</i
  122. >
  123. {{ currentSong.skipVotes }}
  124. </button>
  125. <button
  126. v-else
  127. class="button is-primary disabled"
  128. content="Login to vote to skip songs"
  129. v-tippy
  130. >
  131. <i
  132. class="material-icons icon-with-button"
  133. >skip_next</i
  134. >
  135. {{ currentSong.skipVotes }}
  136. </button>
  137. </div>
  138. <div id="duration">
  139. <p>
  140. {{ timeElapsed }} /
  141. {{
  142. utils.formatTime(
  143. currentSong.duration
  144. )
  145. }}
  146. </p>
  147. </div>
  148. <p id="volume-control">
  149. <i
  150. v-if="muted"
  151. class="material-icons"
  152. @click="toggleMute()"
  153. content="Unmute"
  154. v-tippy
  155. >volume_mute</i
  156. >
  157. <i
  158. v-else
  159. class="material-icons"
  160. @click="toggleMute()"
  161. content="Mute"
  162. v-tippy
  163. >volume_down</i
  164. >
  165. <input
  166. v-model="volumeSliderValue"
  167. type="range"
  168. min="0"
  169. max="10000"
  170. class="volume-slider active"
  171. @change="changeVolume()"
  172. @input="changeVolume()"
  173. />
  174. <i
  175. class="material-icons"
  176. @click="increaseVolume()"
  177. content="Increase Volume"
  178. v-tippy
  179. >volume_up</i
  180. >
  181. </p>
  182. <div id="right-buttons" v-if="loggedIn">
  183. <!-- Ratings (Like/Dislike) Buttons -->
  184. <div
  185. id="ratings"
  186. :class="{
  187. liked: liked,
  188. disliked: disliked
  189. }"
  190. >
  191. <!-- Like Song Button -->
  192. <button
  193. class="button is-success like-song"
  194. id="like-song"
  195. @click="toggleLike()"
  196. content="Like Song"
  197. v-tippy
  198. >
  199. <i
  200. class="material-icons icon-with-button"
  201. :class="{ liked: liked }"
  202. >thumb_up_alt</i
  203. >{{ currentSong.likes }}
  204. </button>
  205. <!-- Dislike Song Button -->
  206. <button
  207. class="button is-danger dislike-song"
  208. id="dislike-song"
  209. @click="toggleDislike()"
  210. content="Dislike Song"
  211. v-tippy
  212. >
  213. <i
  214. class="material-icons icon-with-button"
  215. :class="{
  216. disliked: disliked
  217. }"
  218. >thumb_down_alt</i
  219. >{{ currentSong.dislikes }}
  220. </button>
  221. </div>
  222. <!-- Add Song To Playlist Button & Dropdown -->
  223. <add-to-playlist-dropdown
  224. :song="currentSong"
  225. placement="top-end"
  226. >
  227. <div
  228. slot="button"
  229. id="add-song-to-playlist"
  230. v-click-outside="
  231. () =>
  232. (this.showPlaylistDropdown = false)
  233. "
  234. content="Add Song to Playlist"
  235. v-tippy
  236. >
  237. <div class="control has-addons">
  238. <button
  239. class="button is-primary"
  240. @click="
  241. showPlaylistDropdown = !showPlaylistDropdown
  242. "
  243. >
  244. <i class="material-icons"
  245. >queue</i
  246. >
  247. </button>
  248. <button
  249. class="button"
  250. id="dropdown-toggle"
  251. @click="
  252. showPlaylistDropdown = !showPlaylistDropdown
  253. "
  254. >
  255. <i class="material-icons">
  256. {{
  257. showPlaylistDropdown
  258. ? "expand_more"
  259. : "expand_less"
  260. }}
  261. </i>
  262. </button>
  263. </div>
  264. </div>
  265. </add-to-playlist-dropdown>
  266. </div>
  267. <div id="right-buttons" v-else>
  268. <!-- Disabled Ratings (Like/Dislike) Buttons -->
  269. <div id="ratings">
  270. <!-- Disabled Like Song Button -->
  271. <button
  272. class="button is-success disabled"
  273. id="like-song"
  274. content="Login to like songs"
  275. v-tippy
  276. >
  277. <i
  278. class="material-icons icon-with-button"
  279. >thumb_up_alt</i
  280. >{{ currentSong.likes }}
  281. </button>
  282. <!-- Disabled Dislike Song Button -->
  283. <button
  284. class="button is-danger disabled"
  285. id="dislike-song"
  286. content="Login to dislike songs"
  287. v-tippy
  288. >
  289. <i
  290. class="material-icons icon-with-button"
  291. >thumb_down_alt</i
  292. >{{ currentSong.dislikes }}
  293. </button>
  294. </div>
  295. <!-- Disabled Add Song To Playlist Button & Dropdown -->
  296. <div id="add-song-to-playlist">
  297. <div class="control has-addons">
  298. <button
  299. class="button is-primary disabled"
  300. content="Login to add songs to playlist"
  301. v-tippy
  302. >
  303. <i class="material-icons"
  304. >queue</i
  305. >
  306. </button>
  307. </div>
  308. </div>
  309. </div>
  310. </div>
  311. </div>
  312. <p
  313. class="player-container nothing-here-text"
  314. v-if="noSong"
  315. >
  316. No song is currently playing
  317. </p>
  318. <div v-if="!noSong" id="current-next-row">
  319. <div
  320. id="currently-playing-container"
  321. class="quadrant"
  322. :class="{ 'no-currently-playing': noSong }"
  323. >
  324. <currently-playing :song="currentSong" />
  325. <!-- <p v-else class="nothing-here-text">
  326. No song is currently playing
  327. </p> -->
  328. </div>
  329. <div
  330. v-if="nextSong"
  331. id="next-up-container"
  332. class="quadrant"
  333. >
  334. <currently-playing
  335. type="next"
  336. :song="nextSong"
  337. />
  338. </div>
  339. </div>
  340. </div>
  341. <div id="station-right-column" class="column">
  342. <div id="about-station-container" class="quadrant">
  343. <div id="station-info">
  344. <div class="row" id="station-name">
  345. <h1>{{ station.displayName }}</h1>
  346. <a href="#">
  347. <!-- Favorite Station Button -->
  348. <i
  349. v-if="
  350. loggedIn && station.isFavorited
  351. "
  352. @click.prevent="unfavoriteStation()"
  353. content="Unfavorite Station"
  354. v-tippy
  355. class="material-icons"
  356. >star</i
  357. >
  358. <i
  359. v-if="
  360. loggedIn && !station.isFavorited
  361. "
  362. @click.prevent="favoriteStation()"
  363. class="material-icons"
  364. content="Favorite Station"
  365. v-tippy
  366. >star_border</i
  367. >
  368. </a>
  369. <i
  370. class="material-icons stationMode"
  371. :content="
  372. station.partyMode
  373. ? 'Station in Party mode'
  374. : 'Station in Playlist mode'
  375. "
  376. v-tippy
  377. >{{
  378. station.partyMode
  379. ? "emoji_people"
  380. : "playlist_play"
  381. }}</i
  382. >
  383. </div>
  384. <p>{{ station.description }}</p>
  385. </div>
  386. <div id="admin-buttons" v-if="isOwnerOrAdmin()">
  387. <!-- (Admin) Pause/Resume Button -->
  388. <button
  389. class="button is-danger"
  390. v-if="stationPaused"
  391. @click="resumeStation()"
  392. >
  393. <i class="material-icons icon-with-button"
  394. >play_arrow</i
  395. >
  396. <span class="optional-desktop-only-text">
  397. Resume Station
  398. </span>
  399. </button>
  400. <button
  401. class="button is-danger"
  402. @click="pauseStation()"
  403. v-else
  404. >
  405. <i class="material-icons icon-with-button"
  406. >pause</i
  407. >
  408. <span class="optional-desktop-only-text">
  409. Pause Station
  410. </span>
  411. </button>
  412. <!-- (Admin) Skip Button -->
  413. <button
  414. class="button is-danger"
  415. @click="skipStation()"
  416. >
  417. <i class="material-icons icon-with-button"
  418. >skip_next</i
  419. >
  420. <span class="optional-desktop-only-text">
  421. Force Skip
  422. </span>
  423. </button>
  424. <!-- (Admin) Station Settings Button -->
  425. <button
  426. class="button is-primary"
  427. @click="
  428. openModal({
  429. sector: 'station',
  430. modal: 'editStation'
  431. })
  432. "
  433. >
  434. <i class="material-icons icon-with-button"
  435. >settings</i
  436. >
  437. <span class="optional-desktop-only-text">
  438. Station settings
  439. </span>
  440. </button>
  441. </div>
  442. </div>
  443. <div id="sidebar-container" class="quadrant">
  444. <station-sidebar />
  445. </div>
  446. </div>
  447. </div>
  448. <song-queue v-if="modals.station.addSongToQueue" />
  449. <edit-playlist v-if="modals.station.editPlaylist" />
  450. <create-playlist v-if="modals.station.createPlaylist" />
  451. <edit-station
  452. v-if="modals.station.editStation"
  453. :station-id="station._id"
  454. sector="station"
  455. />
  456. <report v-if="modals.station.report" />
  457. </div>
  458. <main-footer v-if="exists" />
  459. </div>
  460. <edit-song
  461. v-if="modals.admin.editSong"
  462. song-type="songs"
  463. sector="station"
  464. />
  465. <floating-box id="player-debug-box" ref="playerDebugBox">
  466. <template #body>
  467. <span><b>YouTube id</b>: {{ currentSong.songId }}</span>
  468. <span><b>Duration</b>: {{ currentSong.duration }}</span>
  469. <span
  470. ><b>Skip duration</b>: {{ currentSong.skipDuration }}</span
  471. >
  472. <span><b>Can autoplay</b>: {{ canAutoplay }}</span>
  473. <span
  474. ><b>Attempts to play video</b>:
  475. {{ attemptsToPlayVideo }}</span
  476. >
  477. <span
  478. ><b>Last time requested if can autoplay</b>:
  479. {{ lastTimeRequestedIfCanAutoplay }}</span
  480. >
  481. <span><b>Loading</b>: {{ loading }}</span>
  482. <span><b>Playback rate</b>: {{ playbackRate }}</span>
  483. <span><b>Player ready</b>: {{ playerReady }}</span>
  484. <span><b>Ready</b>: {{ ready }}</span>
  485. <span><b>Seeking</b>: {{ seeking }}</span>
  486. <span><b>System difference</b>: {{ systemDifference }}</span>
  487. <span><b>Time before paused</b>: {{ timeBeforePause }}</span>
  488. <span><b>Time elapsed</b>: {{ timeElapsed }}</span>
  489. <span><b>Time paused</b>: {{ timePaused }}</span>
  490. <span><b>Volume slider value</b>: {{ volumeSliderValue }}</span>
  491. <span><b>Local paused</b>: {{ localPaused }}</span>
  492. <span><b>No song</b>: {{ noSong }}</span>
  493. <span
  494. ><b>Private playlist queue selected</b>:
  495. {{ privatePlaylistQueueSelected }}</span
  496. >
  497. <span><b>Station paused</b>: {{ stationPaused }}</span>
  498. <span
  499. ><b>Station Included Playlists</b>:
  500. {{ station.includedPlaylists.join(", ") }}</span
  501. >
  502. <span
  503. ><b>Station Excluded Playlists</b>:
  504. {{ station.excludedPlaylists.join(", ") }}</span
  505. >
  506. </template>
  507. </floating-box>
  508. <Z404 v-if="!exists"></Z404>
  509. </div>
  510. </template>
  511. <script>
  512. import { mapState, mapActions, mapGetters } from "vuex";
  513. import Toast from "toasters";
  514. import { ContentLoader } from "vue-content-loader";
  515. import MainHeader from "../../components/layout/MainHeader.vue";
  516. import MainFooter from "../../components/layout/MainFooter.vue";
  517. import Z404 from "../404.vue";
  518. import FloatingBox from "../../components/ui/FloatingBox.vue";
  519. import AddToPlaylistDropdown from "../../components/ui/AddToPlaylistDropdown.vue";
  520. import ws from "../../ws";
  521. import keyboardShortcuts from "../../keyboardShortcuts";
  522. import utils from "../../../js/utils";
  523. import CurrentlyPlaying from "./components/CurrentlyPlaying.vue";
  524. import StationSidebar from "./components/Sidebar/index.vue";
  525. export default {
  526. components: {
  527. ContentLoader,
  528. MainHeader,
  529. MainFooter,
  530. SongQueue: () => import("../../components/modals/AddSongToQueue.vue"),
  531. EditPlaylist: () =>
  532. import("../../components/modals/EditPlaylist/index.vue"),
  533. CreatePlaylist: () =>
  534. import("../../components/modals/CreatePlaylist.vue"),
  535. EditStation: () => import("../../components/modals/EditStation.vue"),
  536. Report: () => import("../../components/modals/Report.vue"),
  537. Z404,
  538. FloatingBox,
  539. CurrentlyPlaying,
  540. StationSidebar,
  541. AddToPlaylistDropdown,
  542. EditSong: () => import("../../components/modals/EditSong.vue")
  543. },
  544. data() {
  545. return {
  546. utils,
  547. title: "Station",
  548. loading: true,
  549. ready: false,
  550. exists: true,
  551. playerReady: false,
  552. player: undefined,
  553. timePaused: 0,
  554. muted: false,
  555. timeElapsed: "0:00",
  556. liked: false,
  557. disliked: false,
  558. timeBeforePause: 0,
  559. skipVotes: 0,
  560. automaticallyRequestedSongId: null,
  561. systemDifference: 0,
  562. attemptsToPlayVideo: 0,
  563. canAutoplay: true,
  564. lastTimeRequestedIfCanAutoplay: 0,
  565. seeking: false,
  566. playbackRate: 1,
  567. volumeSliderValue: 0,
  568. showPlaylistDropdown: false,
  569. theme: "var(--primary-color)"
  570. };
  571. },
  572. computed: {
  573. ...mapState("modalVisibility", {
  574. modals: state => state.modals
  575. }),
  576. ...mapState("station", {
  577. station: state => state.station,
  578. currentSong: state => state.currentSong,
  579. nextSong: state => state.nextSong,
  580. songsList: state => state.songsList,
  581. stationPaused: state => state.stationPaused,
  582. localPaused: state => state.localPaused,
  583. noSong: state => state.noSong,
  584. privatePlaylistQueueSelected: state =>
  585. state.privatePlaylistQueueSelected
  586. }),
  587. ...mapState({
  588. loggedIn: state => state.user.auth.loggedIn,
  589. userId: state => state.user.auth.userId,
  590. role: state => state.user.auth.role,
  591. nightmode: state => state.user.preferences.nightmode,
  592. autoSkipDisliked: state => state.user.preferences.autoSkipDisliked
  593. }),
  594. ...mapGetters({
  595. socket: "websockets/getSocket"
  596. })
  597. },
  598. mounted() {
  599. window.scrollTo(0, 0);
  600. Date.currently = () => {
  601. return new Date().getTime() + this.systemDifference;
  602. };
  603. this.stationIdentifier = this.$route.params.id;
  604. window.stationInterval = 0;
  605. if (this.socket.readyState === 1) this.join();
  606. ws.onConnect(() => this.join());
  607. this.socket.dispatch(
  608. "stations.existsByName",
  609. this.stationIdentifier,
  610. res => {
  611. if (res.status === "failure" || !res.exists) {
  612. // station identifier may be using stationid instead
  613. this.socket.dispatch(
  614. "stations.existsById",
  615. this.stationIdentifier,
  616. res => {
  617. if (res.status === "failure" || !res.exists) {
  618. this.loading = false;
  619. this.exists = false;
  620. }
  621. }
  622. );
  623. }
  624. }
  625. );
  626. this.socket.on("event:songs.next", data => {
  627. const previousSong = this.currentSong.songId
  628. ? this.currentSong
  629. : null;
  630. this.updatePreviousSong(previousSong);
  631. const { currentSong } = data;
  632. this.updateCurrentSong(currentSong || {});
  633. let nextSong = null;
  634. if (this.songsList[1]) {
  635. nextSong = this.songsList[1].songId ? this.songsList[1] : null;
  636. }
  637. this.updateNextSong(nextSong);
  638. this.startedAt = data.startedAt;
  639. this.updateStationPaused(data.paused);
  640. this.timePaused = data.timePaused;
  641. if (currentSong) {
  642. this.updateNoSong(false);
  643. if (!this.playerReady) this.youtubeReady();
  644. else this.playVideo();
  645. this.socket.dispatch(
  646. "songs.getOwnSongRatings",
  647. data.currentSong.songId,
  648. song => {
  649. if (this.currentSong.songId === song.songId) {
  650. this.liked = song.liked;
  651. this.disliked = song.disliked;
  652. if (
  653. this.autoSkipDisliked &&
  654. song.disliked === true
  655. ) {
  656. this.voteSkipStation();
  657. new Toast({
  658. content:
  659. "Automatically voted to skip disliked song.",
  660. timeout: 4000
  661. });
  662. }
  663. }
  664. }
  665. );
  666. } else {
  667. if (this.playerReady) this.player.pauseVideo();
  668. this.updateNoSong(true);
  669. }
  670. let isInQueue = false;
  671. this.songsList.forEach(queueSong => {
  672. if (queueSong.requestedBy === this.userId) isInQueue = true;
  673. });
  674. if (
  675. !isInQueue &&
  676. this.privatePlaylistQueueSelected &&
  677. (this.automaticallyRequestedSongId !==
  678. this.currentSong.songId ||
  679. !this.currentSong.songId)
  680. ) {
  681. this.addFirstPrivatePlaylistSongToQueue();
  682. }
  683. // if (this.station.type === "official") {
  684. // this.socket.dispatch(
  685. // "stations.getQueue",
  686. // this.station._id,
  687. // res => {
  688. // if (res.status === "success") {
  689. // this.updateSongsList(res.queue);
  690. // }
  691. // }
  692. // );
  693. // }
  694. // if (
  695. // !isInQueue &&
  696. // this.privatePlaylistQueueSelected &&
  697. // (this.automaticallyRequestedSongId !==
  698. // this.currentSong.songId ||
  699. // !this.currentSong.songId)
  700. // ) {
  701. // this.addFirstPrivatePlaylistSongToQueue();
  702. // }
  703. });
  704. this.socket.on("event:stations.pause", data => {
  705. this.pausedAt = data.pausedAt;
  706. this.updateStationPaused(true);
  707. this.pauseLocalPlayer();
  708. });
  709. this.socket.on("event:stations.resume", data => {
  710. this.timePaused = data.timePaused;
  711. this.updateStationPaused(false);
  712. if (!this.localPaused) this.resumeLocalPlayer();
  713. });
  714. this.socket.on("event:stations.remove", () => {
  715. window.location.href = "/";
  716. return true;
  717. });
  718. this.socket.on("event:song.like", data => {
  719. if (!this.noSong) {
  720. if (data.songId === this.currentSong.songId) {
  721. this.currentSong.dislikes = data.dislikes;
  722. this.currentSong.likes = data.likes;
  723. }
  724. }
  725. });
  726. this.socket.on("event:song.dislike", data => {
  727. if (!this.noSong) {
  728. if (data.songId === this.currentSong.songId) {
  729. this.currentSong.dislikes = data.dislikes;
  730. this.currentSong.likes = data.likes;
  731. }
  732. }
  733. });
  734. this.socket.on("event:song.unlike", data => {
  735. if (!this.noSong) {
  736. if (data.songId === this.currentSong.songId) {
  737. this.currentSong.dislikes = data.dislikes;
  738. this.currentSong.likes = data.likes;
  739. }
  740. }
  741. });
  742. this.socket.on("event:song.undislike", data => {
  743. if (!this.noSong) {
  744. if (data.songId === this.currentSong.songId) {
  745. this.currentSong.dislikes = data.dislikes;
  746. this.currentSong.likes = data.likes;
  747. }
  748. }
  749. });
  750. this.socket.on("event:song.newRatings", data => {
  751. if (!this.noSong) {
  752. if (data.songId === this.currentSong.songId) {
  753. this.liked = data.liked;
  754. this.disliked = data.disliked;
  755. }
  756. }
  757. });
  758. this.socket.on("event:queue.update", queue => {
  759. this.updateSongsList(queue);
  760. let nextSong = null;
  761. if (this.songsList[0]) {
  762. nextSong = this.songsList[0].songId ? this.songsList[0] : null;
  763. }
  764. this.updateNextSong(nextSong);
  765. });
  766. this.socket.on("event:song.voteSkipSong", () => {
  767. if (this.currentSong) this.currentSong.skipVotes += 1;
  768. });
  769. this.socket.on("event:privatePlaylist.selected", playlistId => {
  770. if (this.station.type === "community") {
  771. this.station.privatePlaylist = playlistId;
  772. }
  773. });
  774. this.socket.on("event:privatePlaylist.deselected", () => {
  775. if (this.station.type === "community") {
  776. this.station.privatePlaylist = null;
  777. }
  778. });
  779. this.socket.on("event:partyMode.updated", partyMode => {
  780. if (this.station.type === "community") {
  781. this.station.partyMode = partyMode;
  782. }
  783. });
  784. this.socket.on("event:station.themeUpdated", theme => {
  785. this.station.theme = theme;
  786. document.body.style.cssText = `--primary-color: var(--${theme})`;
  787. });
  788. this.socket.on("event:station.updateName", res => {
  789. this.station.name = res.name;
  790. // eslint-disable-next-line no-restricted-globals
  791. history.pushState(
  792. {},
  793. null,
  794. `${res.name}?${Object.keys(this.$route.query)
  795. .map(key => {
  796. return `${encodeURIComponent(key)}=${encodeURIComponent(
  797. this.$route.query[key]
  798. )}`;
  799. })
  800. .join("&")}`
  801. );
  802. });
  803. this.socket.on("event:station.updateDisplayName", res => {
  804. this.station.displayName = res.displayName;
  805. });
  806. this.socket.on("event:station.updateDescription", res => {
  807. this.station.description = res.description;
  808. });
  809. // this.socket.on("event:newOfficialPlaylist", playlist => {
  810. // if (this.station.type === "official")
  811. // this.updateSongsList(playlist);
  812. // });
  813. this.socket.on("event:users.updated", users => this.updateUsers(users));
  814. this.socket.on("event:userCount.updated", userCount =>
  815. this.updateUserCount(userCount)
  816. );
  817. this.socket.on("event:queueLockToggled", locked => {
  818. this.station.locked = locked;
  819. });
  820. this.socket.on("event:user.favoritedStation", stationId => {
  821. if (stationId === this.station._id)
  822. this.updateIfStationIsFavorited({ isFavorited: true });
  823. });
  824. this.socket.on("event:user.unfavoritedStation", stationId => {
  825. if (stationId === this.station._id)
  826. this.updateIfStationIsFavorited({ isFavorited: false });
  827. });
  828. if (JSON.parse(localStorage.getItem("muted"))) {
  829. this.muted = true;
  830. this.player.setVolume(0);
  831. this.volumeSliderValue = 0 * 100;
  832. } else {
  833. let volume = parseFloat(localStorage.getItem("volume"));
  834. volume =
  835. typeof volume === "number" && !Number.isNaN(volume)
  836. ? volume
  837. : 20;
  838. localStorage.setItem("volume", volume);
  839. this.volumeSliderValue = volume * 100;
  840. }
  841. this.frontendDevMode = lofig.get("mode");
  842. },
  843. beforeDestroy() {
  844. document.body.style.cssText = "";
  845. /** Reset Songslist */
  846. this.updateSongsList([]);
  847. const shortcutNames = [
  848. "station.pauseResume",
  849. "station.skipStation",
  850. "station.lowerVolumeLarge",
  851. "station.lowerVolumeSmall",
  852. "station.increaseVolumeLarge",
  853. "station.increaseVolumeSmall",
  854. "station.toggleDebug"
  855. ];
  856. shortcutNames.forEach(shortcutName => {
  857. keyboardShortcuts.unregisterShortcut(shortcutName);
  858. });
  859. },
  860. methods: {
  861. isOwnerOnly() {
  862. return this.loggedIn && this.userId === this.station.owner;
  863. },
  864. isAdminOnly() {
  865. return this.loggedIn && this.role === "admin";
  866. },
  867. isOwnerOrAdmin() {
  868. return this.isOwnerOnly() || this.isAdminOnly();
  869. },
  870. removeFromQueue(songId) {
  871. window.socket.dispatch(
  872. "stations.removeFromQueue",
  873. this.station._id,
  874. songId,
  875. res => {
  876. if (res.status === "success") {
  877. new Toast({
  878. content:
  879. "Successfully removed song from the queue.",
  880. timeout: 4000
  881. });
  882. } else new Toast({ content: res.message, timeout: 8000 });
  883. }
  884. );
  885. },
  886. youtubeReady() {
  887. if (!this.player) {
  888. this.player = new window.YT.Player("stationPlayer", {
  889. height: 270,
  890. width: 480,
  891. videoId: this.currentSong.songId,
  892. host: "https://www.youtube-nocookie.com",
  893. startSeconds:
  894. this.getTimeElapsed() / 1000 +
  895. this.currentSong.skipDuration,
  896. playerVars: {
  897. controls: 0,
  898. iv_load_policy: 3,
  899. rel: 0,
  900. showinfo: 0,
  901. disablekb: 1
  902. },
  903. events: {
  904. onReady: () => {
  905. this.playerReady = true;
  906. let volume = parseInt(
  907. localStorage.getItem("volume")
  908. );
  909. volume = typeof volume === "number" ? volume : 20;
  910. this.player.setVolume(volume);
  911. if (volume > 0) this.player.unMute();
  912. if (this.muted) this.player.mute();
  913. this.playVideo();
  914. },
  915. onError: err => {
  916. console.log("error with youtube video", err);
  917. if (err.data === 150 && this.loggedIn) {
  918. new Toast({
  919. content:
  920. "Automatically voted to skip as this song isn't available for you.",
  921. timeout: 4000
  922. });
  923. // automatically vote to skip
  924. this.voteSkipStation();
  925. // persistent message while song is playing
  926. const toastMessage =
  927. "This song is unavailable for you, but is playing for everyone else.";
  928. new Toast({
  929. content: toastMessage,
  930. persistant: true
  931. });
  932. // save current song id
  933. const erroredSongId = this.currentSong.songId;
  934. // remove persistent toast if video has finished
  935. window.isSongErroredInterval = setInterval(
  936. () => {
  937. if (
  938. this.currentSong.songId !==
  939. erroredSongId
  940. ) {
  941. document
  942. .getElementById(
  943. "toasts-content"
  944. )
  945. .childNodes.forEach(toast => {
  946. if (
  947. toast.innerHTML ===
  948. toastMessage
  949. )
  950. toast.remove();
  951. });
  952. clearInterval(
  953. window.isSongErroredInterval
  954. );
  955. }
  956. },
  957. 150
  958. );
  959. } else {
  960. new Toast({
  961. content:
  962. "There has been an error with the YouTube Embed",
  963. timeout: 8000
  964. });
  965. }
  966. },
  967. onStateChange: event => {
  968. if (
  969. event.data === window.YT.PlayerState.PLAYING &&
  970. this.videoLoading === true
  971. ) {
  972. this.videoLoading = false;
  973. this.player.seekTo(
  974. this.getTimeElapsed() / 1000 +
  975. this.currentSong.skipDuration,
  976. true
  977. );
  978. if (this.localPaused || this.stationPaused)
  979. this.player.pauseVideo();
  980. } else if (
  981. event.data === window.YT.PlayerState.PLAYING &&
  982. (this.localPaused || this.stationPaused)
  983. ) {
  984. this.player.seekTo(
  985. this.timeBeforePause / 1000,
  986. true
  987. );
  988. this.player.pauseVideo();
  989. } else if (
  990. event.data === window.YT.PlayerState.PLAYING &&
  991. this.seeking === true
  992. ) {
  993. this.seeking = false;
  994. }
  995. if (
  996. event.data === window.YT.PlayerState.PAUSED &&
  997. !this.localPaused &&
  998. !this.stationPaused &&
  999. !this.noSong &&
  1000. this.player.getDuration() / 1000 <
  1001. this.currentSong.duration
  1002. ) {
  1003. this.player.seekTo(
  1004. this.getTimeElapsed() / 1000 +
  1005. this.currentSong.skipDuration,
  1006. true
  1007. );
  1008. this.player.playVideo();
  1009. }
  1010. }
  1011. }
  1012. });
  1013. }
  1014. },
  1015. getTimeElapsed() {
  1016. if (this.currentSong) {
  1017. let { timePaused } = this;
  1018. if (this.stationPaused)
  1019. timePaused += Date.currently() - this.pausedAt;
  1020. return Date.currently() - this.startedAt - timePaused;
  1021. }
  1022. return 0;
  1023. },
  1024. playVideo() {
  1025. if (this.playerReady) {
  1026. this.videoLoading = true;
  1027. this.player.loadVideoById(
  1028. this.currentSong.songId,
  1029. this.getTimeElapsed() / 1000 + this.currentSong.skipDuration
  1030. );
  1031. if (window.stationInterval !== 0)
  1032. clearInterval(window.stationInterval);
  1033. window.stationInterval = setInterval(() => {
  1034. this.resizeSeekerbar();
  1035. this.calculateTimeElapsed();
  1036. }, 150);
  1037. }
  1038. },
  1039. resizeSeekerbar() {
  1040. if (!this.stationPaused) {
  1041. document.getElementById(
  1042. "seeker-bar"
  1043. ).style.width = `${parseFloat(
  1044. (this.getTimeElapsed() / 1000 / this.currentSong.duration) *
  1045. 100
  1046. )}%`;
  1047. }
  1048. },
  1049. calculateTimeElapsed() {
  1050. if (
  1051. this.playerReady &&
  1052. this.currentSong &&
  1053. this.player.getPlayerState() === -1
  1054. ) {
  1055. if (this.attemptsToPlayVideo >= 5) {
  1056. if (
  1057. Date.now() - this.lastTimeRequestedIfCanAutoplay >
  1058. 2000
  1059. ) {
  1060. this.lastTimeRequestedIfCanAutoplay = Date.now();
  1061. window.canAutoplay.video().then(({ result }) => {
  1062. if (result) {
  1063. this.attemptsToPlayVideo = 0;
  1064. this.canAutoplay = true;
  1065. } else {
  1066. this.canAutoplay = false;
  1067. }
  1068. });
  1069. }
  1070. } else {
  1071. this.player.playVideo();
  1072. this.attemptsToPlayVideo += 1;
  1073. }
  1074. }
  1075. if (!this.stationPaused && !this.localPaused) {
  1076. const timeElapsed = this.getTimeElapsed();
  1077. const currentPlayerTime =
  1078. Math.max(
  1079. this.player.getCurrentTime() -
  1080. this.currentSong.skipDuration,
  1081. 0
  1082. ) * 1000;
  1083. const difference = timeElapsed - currentPlayerTime;
  1084. // console.log(difference);
  1085. let playbackRate = 1;
  1086. if (difference < -2000) {
  1087. if (!this.seeking) {
  1088. this.seeking = true;
  1089. this.player.seekTo(
  1090. this.getTimeElapsed() / 1000 +
  1091. this.currentSong.skipDuration
  1092. );
  1093. }
  1094. } else if (difference < -200) {
  1095. // console.log("Difference0.8");
  1096. playbackRate = 0.8;
  1097. } else if (difference < -50) {
  1098. // console.log("Difference0.9");
  1099. playbackRate = 0.9;
  1100. } else if (difference < -25) {
  1101. // console.log("Difference0.99");
  1102. playbackRate = 0.95;
  1103. } else if (difference > 2000) {
  1104. if (!this.seeking) {
  1105. this.seeking = true;
  1106. this.player.seekTo(
  1107. this.getTimeElapsed() / 1000 +
  1108. this.currentSong.skipDuration
  1109. );
  1110. }
  1111. } else if (difference > 200) {
  1112. // console.log("Difference1.2");
  1113. playbackRate = 1.2;
  1114. } else if (difference > 50) {
  1115. // console.log("Difference1.1");
  1116. playbackRate = 1.1;
  1117. } else if (difference > 25) {
  1118. // console.log("Difference1.01");
  1119. playbackRate = 1.05;
  1120. } else if (this.player.getPlaybackRate !== 1.0) {
  1121. // console.log("NDifference1.0");
  1122. this.player.setPlaybackRate(1.0);
  1123. }
  1124. if (this.playbackRate !== playbackRate) {
  1125. this.player.setPlaybackRate(playbackRate);
  1126. this.playbackRate = playbackRate;
  1127. }
  1128. }
  1129. /* if (this.currentTime !== undefined && this.paused) {
  1130. this.timePaused += Date.currently() - this.currentTime;
  1131. this.currentTime = undefined;
  1132. } */
  1133. let { timePaused } = this;
  1134. if (this.stationPaused)
  1135. timePaused += Date.currently() - this.pausedAt;
  1136. const duration =
  1137. (Date.currently() - this.startedAt - timePaused) / 1000;
  1138. const songDuration = this.currentSong.duration;
  1139. if (songDuration <= duration) this.player.pauseVideo();
  1140. if (!this.stationPaused && duration <= songDuration)
  1141. this.timeElapsed = utils.formatTime(duration);
  1142. },
  1143. toggleLock() {
  1144. window.socket.dispatch(
  1145. "stations.toggleLock",
  1146. this.station._id,
  1147. res => {
  1148. if (res.status === "success") {
  1149. new Toast({
  1150. content: "Successfully toggled the queue lock.",
  1151. timeout: 4000
  1152. });
  1153. } else new Toast({ content: res.message, timeout: 8000 });
  1154. }
  1155. );
  1156. },
  1157. changeVolume() {
  1158. const volume = this.volumeSliderValue;
  1159. localStorage.setItem("volume", volume / 100);
  1160. if (this.playerReady) {
  1161. this.player.setVolume(volume / 100);
  1162. if (volume > 0) {
  1163. this.player.unMute();
  1164. localStorage.setItem("muted", false);
  1165. this.muted = false;
  1166. }
  1167. }
  1168. },
  1169. resumeLocalStation() {
  1170. this.updateLocalPaused(false);
  1171. if (!this.stationPaused) this.resumeLocalPlayer();
  1172. },
  1173. pauseLocalStation() {
  1174. this.updateLocalPaused(true);
  1175. this.pauseLocalPlayer();
  1176. },
  1177. resumeLocalPlayer() {
  1178. if (!this.noSong) {
  1179. if (this.playerReady) {
  1180. this.player.seekTo(
  1181. this.getTimeElapsed() / 1000 +
  1182. this.currentSong.skipDuration
  1183. );
  1184. this.player.playVideo();
  1185. }
  1186. }
  1187. },
  1188. pauseLocalPlayer() {
  1189. if (!this.noSong) {
  1190. this.timeBeforePause = this.getTimeElapsed();
  1191. if (this.playerReady) this.player.pauseVideo();
  1192. }
  1193. },
  1194. skipStation() {
  1195. this.socket.dispatch(
  1196. "stations.forceSkip",
  1197. this.station._id,
  1198. data => {
  1199. if (data.status !== "success")
  1200. new Toast({
  1201. content: `Error: ${data.message}`,
  1202. timeout: 8000
  1203. });
  1204. else
  1205. new Toast({
  1206. content:
  1207. "Successfully skipped the station's current song.",
  1208. timeout: 4000
  1209. });
  1210. }
  1211. );
  1212. },
  1213. voteSkipStation() {
  1214. this.socket.dispatch(
  1215. "stations.voteSkip",
  1216. this.station._id,
  1217. data => {
  1218. if (data.status !== "success")
  1219. new Toast({
  1220. content: `Error: ${data.message}`,
  1221. timeout: 8000
  1222. });
  1223. else
  1224. new Toast({
  1225. content:
  1226. "Successfully voted to skip the current song.",
  1227. timeout: 4000
  1228. });
  1229. }
  1230. );
  1231. },
  1232. resumeStation() {
  1233. this.socket.dispatch("stations.resume", this.station._id, data => {
  1234. if (data.status !== "success")
  1235. new Toast({
  1236. content: `Error: ${data.message}`,
  1237. timeout: 8000
  1238. });
  1239. else
  1240. new Toast({
  1241. content: "Successfully resumed the station.",
  1242. timeout: 4000
  1243. });
  1244. });
  1245. },
  1246. pauseStation() {
  1247. this.socket.dispatch("stations.pause", this.station._id, data => {
  1248. if (data.status !== "success")
  1249. new Toast({
  1250. content: `Error: ${data.message}`,
  1251. timeout: 8000
  1252. });
  1253. else
  1254. new Toast({
  1255. content: "Successfully paused the station.",
  1256. timeout: 4000
  1257. });
  1258. });
  1259. },
  1260. toggleMute() {
  1261. if (this.playerReady) {
  1262. const previousVolume = parseFloat(
  1263. localStorage.getItem("volume")
  1264. );
  1265. const volume =
  1266. this.player.getVolume() * 100 <= 0 ? previousVolume : 0;
  1267. this.muted = !this.muted;
  1268. localStorage.setItem("muted", this.muted);
  1269. this.volumeSliderValue = volume * 100;
  1270. this.player.setVolume(volume);
  1271. if (!this.muted) localStorage.setItem("volume", volume);
  1272. }
  1273. },
  1274. increaseVolume() {
  1275. if (this.playerReady) {
  1276. const previousVolume = parseInt(localStorage.getItem("volume"));
  1277. let volume = previousVolume + 5;
  1278. if (previousVolume === 0) {
  1279. this.muted = false;
  1280. localStorage.setItem("muted", false);
  1281. }
  1282. if (volume > 100) volume = 100;
  1283. this.volumeSliderValue = volume * 100;
  1284. this.player.setVolume(volume);
  1285. localStorage.setItem("volume", volume);
  1286. }
  1287. },
  1288. toggleLike() {
  1289. if (this.liked)
  1290. this.socket.dispatch(
  1291. "songs.unlike",
  1292. this.currentSong.songId,
  1293. data => {
  1294. if (data.status !== "success")
  1295. new Toast({
  1296. content: `Error: ${data.message}`,
  1297. timeout: 8000
  1298. });
  1299. }
  1300. );
  1301. else
  1302. this.socket.dispatch(
  1303. "songs.like",
  1304. this.currentSong.songId,
  1305. data => {
  1306. if (data.status !== "success")
  1307. new Toast({
  1308. content: `Error: ${data.message}`,
  1309. timeout: 8000
  1310. });
  1311. }
  1312. );
  1313. },
  1314. toggleDislike() {
  1315. if (this.disliked)
  1316. return this.socket.dispatch(
  1317. "songs.undislike",
  1318. this.currentSong.songId,
  1319. data => {
  1320. if (data.status !== "success")
  1321. new Toast({
  1322. content: `Error: ${data.message}`,
  1323. timeout: 8000
  1324. });
  1325. }
  1326. );
  1327. return this.socket.dispatch(
  1328. "songs.dislike",
  1329. this.currentSong.songId,
  1330. data => {
  1331. if (data.status !== "success")
  1332. new Toast({
  1333. content: `Error: ${data.message}`,
  1334. timeout: 8000
  1335. });
  1336. }
  1337. );
  1338. },
  1339. addFirstPrivatePlaylistSongToQueue() {
  1340. let isInQueue = false;
  1341. if (
  1342. this.station.type === "community" &&
  1343. this.station.partyMode === true
  1344. ) {
  1345. this.songsList.forEach(queueSong => {
  1346. if (queueSong.requestedBy === this.userId) isInQueue = true;
  1347. });
  1348. if (!isInQueue && this.privatePlaylistQueueSelected) {
  1349. this.socket.dispatch(
  1350. "playlists.getFirstSong",
  1351. this.privatePlaylistQueueSelected,
  1352. data => {
  1353. if (data.status === "success") {
  1354. if (data.song) {
  1355. if (data.song.duration < 15 * 60) {
  1356. this.automaticallyRequestedSongId =
  1357. data.song.songId;
  1358. this.socket.dispatch(
  1359. "stations.addToQueue",
  1360. this.station._id,
  1361. data.song.songId,
  1362. data2 => {
  1363. if (data2.status === "success")
  1364. this.socket.dispatch(
  1365. "playlists.moveSongToBottom",
  1366. this
  1367. .privatePlaylistQueueSelected,
  1368. data.song.songId
  1369. );
  1370. }
  1371. );
  1372. } else {
  1373. new Toast({
  1374. content: `Top song in playlist was too long to be added.`,
  1375. timeout: 3000
  1376. });
  1377. this.socket.dispatch(
  1378. "playlists.moveSongToBottom",
  1379. this.privatePlaylistQueueSelected,
  1380. data.song.songId,
  1381. data3 => {
  1382. if (data3.status === "success")
  1383. setTimeout(
  1384. () =>
  1385. this.addFirstPrivatePlaylistSongToQueue(),
  1386. 3000
  1387. );
  1388. }
  1389. );
  1390. }
  1391. } else
  1392. new Toast({
  1393. content: `Selected playlist has no songs.`,
  1394. timeout: 4000
  1395. });
  1396. }
  1397. }
  1398. );
  1399. }
  1400. }
  1401. },
  1402. togglePlayerDebugBox() {
  1403. this.$refs.playerDebugBox.toggleBox();
  1404. },
  1405. resetPlayerDebugBox() {
  1406. this.$refs.playerDebugBox.resetBox();
  1407. },
  1408. join() {
  1409. this.socket.dispatch(
  1410. "stations.join",
  1411. this.stationIdentifier,
  1412. res => {
  1413. if (res.status === "success") {
  1414. setTimeout(() => {
  1415. this.loading = false;
  1416. }, 1000); // prevents popping in of youtube embed etc.
  1417. const {
  1418. _id,
  1419. displayName,
  1420. name,
  1421. description,
  1422. privacy,
  1423. locked,
  1424. partyMode,
  1425. owner,
  1426. privatePlaylist,
  1427. includedPlaylists,
  1428. excludedPlaylists,
  1429. type,
  1430. genres,
  1431. blacklistedGenres,
  1432. isFavorited,
  1433. theme
  1434. } = res.data;
  1435. // change url to use station name instead of station id
  1436. if (name !== this.stationIdentifier) {
  1437. // eslint-disable-next-line no-restricted-globals
  1438. history.pushState({}, null, name);
  1439. }
  1440. this.joinStation({
  1441. _id,
  1442. name,
  1443. displayName,
  1444. description,
  1445. privacy,
  1446. locked,
  1447. partyMode,
  1448. owner,
  1449. privatePlaylist,
  1450. includedPlaylists,
  1451. excludedPlaylists,
  1452. type,
  1453. genres,
  1454. blacklistedGenres,
  1455. isFavorited,
  1456. theme
  1457. });
  1458. document.body.style.cssText = `--primary-color: var(--${res.data.theme})`;
  1459. const currentSong = res.data.currentSong
  1460. ? res.data.currentSong
  1461. : {};
  1462. this.updateCurrentSong(currentSong);
  1463. this.startedAt = res.data.startedAt;
  1464. this.updateStationPaused(res.data.paused);
  1465. this.timePaused = res.data.timePaused;
  1466. this.updateUserCount(res.data.userCount);
  1467. this.updateUsers(res.data.users);
  1468. this.pausedAt = res.data.pausedAt;
  1469. if (res.data.currentSong) {
  1470. this.updateNoSong(false);
  1471. this.youtubeReady();
  1472. this.playVideo();
  1473. this.socket.dispatch(
  1474. "songs.getOwnSongRatings",
  1475. res.data.currentSong.songId,
  1476. song => {
  1477. if (
  1478. this.currentSong.songId === song.songId
  1479. ) {
  1480. this.liked = song.liked;
  1481. this.disliked = song.disliked;
  1482. }
  1483. }
  1484. );
  1485. } else {
  1486. if (this.playerReady) this.player.pauseVideo();
  1487. this.updateNoSong(true);
  1488. }
  1489. this.socket.dispatch("stations.getQueue", _id, res => {
  1490. if (res.status === "success") {
  1491. this.updateSongsList(res.queue);
  1492. let nextSong = null;
  1493. if (this.songsList[0]) {
  1494. nextSong = this.songsList[0].songId
  1495. ? this.songsList[0]
  1496. : null;
  1497. }
  1498. this.updateNextSong(nextSong);
  1499. }
  1500. });
  1501. if (this.isOwnerOrAdmin()) {
  1502. keyboardShortcuts.registerShortcut(
  1503. "station.pauseResume",
  1504. {
  1505. keyCode: 32,
  1506. shift: false,
  1507. ctrl: true,
  1508. preventDefault: true,
  1509. handler: () => {
  1510. if (this.stationPaused)
  1511. this.resumeStation();
  1512. else this.pauseStation();
  1513. }
  1514. }
  1515. );
  1516. keyboardShortcuts.registerShortcut(
  1517. "station.skipStation",
  1518. {
  1519. keyCode: 39,
  1520. shift: false,
  1521. ctrl: true,
  1522. preventDefault: true,
  1523. handler: () => {
  1524. this.skipStation();
  1525. }
  1526. }
  1527. );
  1528. }
  1529. keyboardShortcuts.registerShortcut(
  1530. "station.lowerVolumeLarge",
  1531. {
  1532. keyCode: 40,
  1533. shift: false,
  1534. ctrl: true,
  1535. preventDefault: true,
  1536. handler: () => {
  1537. this.volumeSliderValue -= 1000;
  1538. this.changeVolume();
  1539. }
  1540. }
  1541. );
  1542. keyboardShortcuts.registerShortcut(
  1543. "station.lowerVolumeSmall",
  1544. {
  1545. keyCode: 40,
  1546. shift: true,
  1547. ctrl: true,
  1548. preventDefault: true,
  1549. handler: () => {
  1550. this.volumeSliderValue -= 100;
  1551. this.changeVolume();
  1552. }
  1553. }
  1554. );
  1555. keyboardShortcuts.registerShortcut(
  1556. "station.increaseVolumeLarge",
  1557. {
  1558. keyCode: 38,
  1559. shift: false,
  1560. ctrl: true,
  1561. preventDefault: true,
  1562. handler: () => {
  1563. this.volumeSliderValue += 1000;
  1564. this.changeVolume();
  1565. }
  1566. }
  1567. );
  1568. keyboardShortcuts.registerShortcut(
  1569. "station.increaseVolumeSmall",
  1570. {
  1571. keyCode: 38,
  1572. shift: true,
  1573. ctrl: true,
  1574. preventDefault: true,
  1575. handler: () => {
  1576. this.volumeSliderValue += 100;
  1577. this.changeVolume();
  1578. }
  1579. }
  1580. );
  1581. keyboardShortcuts.registerShortcut(
  1582. "station.toggleDebug",
  1583. {
  1584. keyCode: 68,
  1585. shift: false,
  1586. ctrl: true,
  1587. preventDefault: true,
  1588. handler: () => {
  1589. this.togglePlayerDebugBox();
  1590. }
  1591. }
  1592. );
  1593. // UNIX client time before ping
  1594. const beforePing = Date.now();
  1595. this.socket.dispatch("apis.ping", pong => {
  1596. // UNIX client time after ping
  1597. const afterPing = Date.now();
  1598. // Average time in MS it took between the server responding and the client receiving
  1599. const connectionLatency =
  1600. (afterPing - beforePing) / 2;
  1601. console.log(
  1602. connectionLatency,
  1603. beforePing - afterPing
  1604. );
  1605. // UNIX server time
  1606. const serverDate = pong.date;
  1607. // Difference between the server UNIX time and the client UNIX time after ping, with the connectionLatency added to the server UNIX time
  1608. const difference =
  1609. serverDate + connectionLatency - afterPing;
  1610. console.log("Difference: ", difference);
  1611. if (difference > 3000 || difference < -3000) {
  1612. console.log(
  1613. "System time difference is bigger than 3 seconds."
  1614. );
  1615. }
  1616. this.systemDifference = difference;
  1617. });
  1618. } else {
  1619. this.loading = false;
  1620. this.exists = false;
  1621. }
  1622. }
  1623. );
  1624. },
  1625. favoriteStation() {
  1626. this.socket.dispatch(
  1627. "stations.favoriteStation",
  1628. this.station._id,
  1629. res => {
  1630. if (res.status === "success") {
  1631. new Toast({
  1632. content: "Successfully favorited station.",
  1633. timeout: 4000
  1634. });
  1635. } else new Toast({ content: res.message, timeout: 8000 });
  1636. }
  1637. );
  1638. },
  1639. unfavoriteStation() {
  1640. this.socket.dispatch(
  1641. "stations.unfavoriteStation",
  1642. this.station._id,
  1643. res => {
  1644. if (res.status === "success") {
  1645. new Toast({
  1646. content: "Successfully unfavorited station.",
  1647. timeout: 4000
  1648. });
  1649. } else new Toast({ content: res.message, timeout: 8000 });
  1650. }
  1651. );
  1652. },
  1653. ...mapActions("modalVisibility", ["openModal"]),
  1654. ...mapActions("station", [
  1655. "joinStation",
  1656. "updateUserCount",
  1657. "updateUsers",
  1658. "updateCurrentSong",
  1659. "updatePreviousSong",
  1660. "updateNextSong",
  1661. "updateSongsList",
  1662. "updateStationPaused",
  1663. "updateLocalPaused",
  1664. "updateNoSong",
  1665. "updateIfStationIsFavorited"
  1666. ]),
  1667. ...mapActions("modals/editSong", ["stopVideo"])
  1668. }
  1669. };
  1670. </script>
  1671. <style lang="scss" scoped>
  1672. #page-loader-container {
  1673. height: inherit;
  1674. #page-loader-content {
  1675. height: inherit;
  1676. position: absolute;
  1677. max-width: 100%;
  1678. width: 1800px;
  1679. transform: translateX(-50%);
  1680. left: 50%;
  1681. }
  1682. #page-loader-layout {
  1683. height: inherit;
  1684. width: 100%;
  1685. }
  1686. }
  1687. #mobile-progress-animation {
  1688. width: 50px;
  1689. animation: rotate 0.8s infinite linear;
  1690. border: 8px solid var(--primary-color);
  1691. border-right-color: transparent;
  1692. border-radius: 50%;
  1693. height: 50px;
  1694. position: absolute;
  1695. top: 50%;
  1696. left: 50%;
  1697. display: none;
  1698. }
  1699. @keyframes rotate {
  1700. 0% {
  1701. transform: rotate(0deg);
  1702. }
  1703. 100% {
  1704. transform: rotate(360deg);
  1705. }
  1706. }
  1707. .nav,
  1708. .button.is-primary {
  1709. background-color: var(--primary-color) !important;
  1710. }
  1711. .button.is-primary:hover,
  1712. .button.is-primary:focus {
  1713. filter: brightness(90%);
  1714. }
  1715. #player-debug-box {
  1716. .box-body {
  1717. flex-direction: column;
  1718. b {
  1719. color: var(--black);
  1720. }
  1721. }
  1722. }
  1723. .night-mode {
  1724. #currently-playing-container,
  1725. #next-up-container,
  1726. #about-station-container,
  1727. #control-bar-container,
  1728. .player-container {
  1729. background-color: var(--dark-grey-3) !important;
  1730. }
  1731. #video-container,
  1732. #control-bar-container,
  1733. .quadrant:not(#sidebar-container),
  1734. .player-container {
  1735. border: 0 !important;
  1736. }
  1737. #seeker-bar-container {
  1738. background-color: var(--dark-grey-3) !important;
  1739. }
  1740. #dropdown-toggle {
  1741. background-color: var(--dark-grey-2) !important;
  1742. border: 0;
  1743. i {
  1744. color: var(--white);
  1745. }
  1746. }
  1747. }
  1748. #station-outer-container {
  1749. margin: 0 auto;
  1750. padding: 20px 40px;
  1751. height: 100%;
  1752. width: 100%;
  1753. max-width: 1800px;
  1754. display: flex;
  1755. #station-inner-container {
  1756. height: 100%;
  1757. width: 100%;
  1758. min-height: calc(100vh - 428px);
  1759. display: flex;
  1760. flex-direction: row;
  1761. flex-wrap: wrap;
  1762. .row {
  1763. display: flex;
  1764. flex-direction: row;
  1765. max-width: 100%;
  1766. }
  1767. .column {
  1768. display: flex;
  1769. flex-direction: column;
  1770. }
  1771. .quadrant {
  1772. border-radius: 5px;
  1773. margin: 10px;
  1774. flex-grow: 1;
  1775. }
  1776. .quadrant:not(#sidebar-container) {
  1777. background-color: var(--white);
  1778. border: 1px solid var(--light-grey-3);
  1779. }
  1780. #station-left-column,
  1781. #station-right-column {
  1782. padding: 0;
  1783. }
  1784. #about-station-container {
  1785. padding: 20px;
  1786. display: flex;
  1787. flex-direction: column;
  1788. flex-grow: unset;
  1789. #station-info {
  1790. #station-name {
  1791. flex-direction: row !important;
  1792. h1 {
  1793. margin: 0;
  1794. font-size: 36px;
  1795. line-height: 0.8;
  1796. }
  1797. i {
  1798. margin-left: 10px;
  1799. font-size: 30px;
  1800. color: var(--yellow);
  1801. &.stationMode {
  1802. padding-left: 10px;
  1803. margin-left: auto;
  1804. color: var(--primary-color);
  1805. }
  1806. }
  1807. }
  1808. p {
  1809. max-width: 700px;
  1810. margin-bottom: 10px;
  1811. }
  1812. }
  1813. #admin-buttons {
  1814. display: flex;
  1815. .button {
  1816. margin: 3px;
  1817. }
  1818. }
  1819. }
  1820. #current-next-row {
  1821. display: flex;
  1822. flex-direction: row;
  1823. max-width: calc(100vw - 40px);
  1824. #currently-playing-container,
  1825. #next-up-container {
  1826. overflow: hidden;
  1827. flex-basis: 50%;
  1828. .nothing-here-text {
  1829. height: 100%;
  1830. }
  1831. }
  1832. }
  1833. .player-container {
  1834. height: inherit;
  1835. background-color: var(--white);
  1836. display: flex;
  1837. flex-direction: column;
  1838. border: 1px solid var(--light-grey-3);
  1839. border-radius: 5px;
  1840. overflow: hidden;
  1841. flex-grow: 1;
  1842. &.nothing-here-text {
  1843. margin: 10px;
  1844. }
  1845. #video-container {
  1846. width: 100%;
  1847. height: 100%;
  1848. .player-cannot-autoplay {
  1849. position: relative;
  1850. width: 100%;
  1851. height: 100%;
  1852. bottom: calc(100% + 5px);
  1853. background: var(--primary-color);
  1854. display: flex;
  1855. align-items: center;
  1856. justify-content: center;
  1857. p {
  1858. color: var(--white);
  1859. font-size: 26px;
  1860. text-align: center;
  1861. }
  1862. }
  1863. }
  1864. #seeker-bar-container {
  1865. background-color: var(--white);
  1866. position: relative;
  1867. height: 7px;
  1868. display: block;
  1869. width: 100%;
  1870. overflow: hidden;
  1871. #seeker-bar {
  1872. background-color: var(--primary-color);
  1873. top: 0;
  1874. left: 0;
  1875. bottom: 0;
  1876. position: absolute;
  1877. }
  1878. }
  1879. #control-bar-container {
  1880. display: flex;
  1881. justify-content: space-around;
  1882. padding: 10px 0;
  1883. width: 100%;
  1884. background: var(--white);
  1885. flex-direction: column;
  1886. flex-flow: wrap;
  1887. .button:not(#dropdown-toggle) {
  1888. width: 75px;
  1889. }
  1890. #left-buttons,
  1891. #right-buttons {
  1892. margin: 3px;
  1893. }
  1894. #left-buttons {
  1895. display: flex;
  1896. .button:not(:first-of-type) {
  1897. margin-left: 5px;
  1898. }
  1899. .disabled {
  1900. filter: grayscale(0.4);
  1901. }
  1902. }
  1903. #duration {
  1904. margin: 3px;
  1905. display: flex;
  1906. align-items: center;
  1907. p {
  1908. font-size: 22px;
  1909. /** prevents duration width slightly varying and shifting other controls slightly */
  1910. width: 150px;
  1911. text-align: center;
  1912. }
  1913. }
  1914. #volume-control {
  1915. margin: 3px;
  1916. margin-top: 0;
  1917. display: flex;
  1918. align-items: center;
  1919. cursor: pointer;
  1920. .volume-slider {
  1921. width: 100%;
  1922. padding: 0 15px;
  1923. background: transparent;
  1924. min-width: 100px;
  1925. }
  1926. input[type="range"] {
  1927. -webkit-appearance: none;
  1928. margin: 7.3px 0;
  1929. }
  1930. input[type="range"]:focus {
  1931. outline: none;
  1932. }
  1933. input[type="range"]::-webkit-slider-runnable-track {
  1934. width: 100%;
  1935. height: 5.2px;
  1936. cursor: pointer;
  1937. box-shadow: 0;
  1938. background: var(--light-grey-3);
  1939. border-radius: 0;
  1940. border: 0;
  1941. }
  1942. input[type="range"]::-webkit-slider-thumb {
  1943. box-shadow: 0;
  1944. border: 0;
  1945. height: 19px;
  1946. width: 19px;
  1947. border-radius: 15px;
  1948. background: var(--primary-color);
  1949. cursor: pointer;
  1950. -webkit-appearance: none;
  1951. margin-top: -6.5px;
  1952. }
  1953. input[type="range"]::-moz-range-track {
  1954. width: 100%;
  1955. height: 5.2px;
  1956. cursor: pointer;
  1957. box-shadow: 0;
  1958. background: var(--light-grey-3);
  1959. border-radius: 0;
  1960. border: 0;
  1961. }
  1962. input[type="range"]::-moz-range-thumb {
  1963. box-shadow: 0;
  1964. border: 0;
  1965. height: 19px;
  1966. width: 19px;
  1967. border-radius: 15px;
  1968. background: var(--primary-color);
  1969. cursor: pointer;
  1970. -webkit-appearance: none;
  1971. margin-top: -6.5px;
  1972. }
  1973. input[type="range"]::-ms-track {
  1974. width: 100%;
  1975. height: 5.2px;
  1976. cursor: pointer;
  1977. box-shadow: 0;
  1978. background: var(--light-grey-3);
  1979. border-radius: 1.3px;
  1980. }
  1981. input[type="range"]::-ms-fill-lower {
  1982. background: var(--light-grey-3);
  1983. border: 0;
  1984. border-radius: 0;
  1985. box-shadow: 0;
  1986. }
  1987. input[type="range"]::-ms-fill-upper {
  1988. background: var(--light-grey-3);
  1989. border: 0;
  1990. border-radius: 0;
  1991. box-shadow: 0;
  1992. }
  1993. input[type="range"]::-ms-thumb {
  1994. box-shadow: 0;
  1995. border: 0;
  1996. height: 15px;
  1997. width: 15px;
  1998. border-radius: 15px;
  1999. background: var(--primary-color);
  2000. cursor: pointer;
  2001. -webkit-appearance: none;
  2002. margin-top: 1.5px;
  2003. }
  2004. }
  2005. #right-buttons {
  2006. display: flex;
  2007. #dropdown-toggle {
  2008. width: 35px;
  2009. }
  2010. #dislike-song,
  2011. #add-song-to-playlist .button:not(#dropdown-toggle) {
  2012. margin-left: 5px;
  2013. }
  2014. #ratings {
  2015. display: flex;
  2016. &.liked #dislike-song,
  2017. &.disliked #like-song {
  2018. background-color: var(--grey) !important;
  2019. }
  2020. #like-song.disabled,
  2021. #dislike-song.disabled {
  2022. filter: grayscale(0.4);
  2023. }
  2024. }
  2025. #add-song-to-playlist {
  2026. display: flex;
  2027. flex-direction: column-reverse;
  2028. #nav-dropdown {
  2029. position: absolute;
  2030. margin-left: 4px;
  2031. margin-bottom: 36px;
  2032. .nav-dropdown-items {
  2033. position: relative;
  2034. right: calc(100% - 110px);
  2035. }
  2036. }
  2037. .control {
  2038. width: fit-content;
  2039. margin-bottom: 0 !important;
  2040. button.disabled {
  2041. filter: grayscale(0.4);
  2042. border-radius: 3px;
  2043. &::after {
  2044. margin-right: 100%;
  2045. }
  2046. }
  2047. }
  2048. }
  2049. }
  2050. }
  2051. }
  2052. #sidebar-container {
  2053. border-top: 0;
  2054. position: relative;
  2055. height: inherit;
  2056. }
  2057. }
  2058. }
  2059. .footer {
  2060. margin-top: 30px;
  2061. }
  2062. /deep/ .nothing-here-text {
  2063. display: flex;
  2064. align-items: center;
  2065. justify-content: center;
  2066. }
  2067. @media (min-width: 1500px) {
  2068. #station-left-column {
  2069. max-width: calc(100% - 650px);
  2070. }
  2071. #station-right-column {
  2072. max-width: 650px;
  2073. }
  2074. }
  2075. @media (max-width: 950px) {
  2076. #mobile-progress-animation {
  2077. display: block;
  2078. }
  2079. #page-loader-container {
  2080. display: none;
  2081. }
  2082. #station-outer-container {
  2083. padding: 10px;
  2084. height: unset;
  2085. max-width: 700px;
  2086. #station-inner-container {
  2087. flex-direction: column;
  2088. #station-left-column {
  2089. #current-next-row {
  2090. flex-direction: column;
  2091. }
  2092. #control-bar-container {
  2093. #duration,
  2094. #volume-control,
  2095. #right-buttons,
  2096. #left-buttons {
  2097. margin-bottom: 5px;
  2098. justify-content: center;
  2099. }
  2100. #duration {
  2101. order: 1;
  2102. }
  2103. #volume-control {
  2104. order: 2;
  2105. max-width: 400px;
  2106. }
  2107. #right-buttons {
  2108. order: 3;
  2109. flex-wrap: wrap;
  2110. #ratings {
  2111. flex-wrap: wrap;
  2112. }
  2113. }
  2114. #left-buttons {
  2115. order: 4;
  2116. flex-wrap: wrap;
  2117. }
  2118. }
  2119. }
  2120. #station-right-column {
  2121. #about-station-container #admin-buttons {
  2122. flex-wrap: wrap;
  2123. }
  2124. #sidebar-container {
  2125. min-height: 350px;
  2126. }
  2127. }
  2128. }
  2129. }
  2130. }
  2131. </style>