index.html 84 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="apple-mobile-web-app-capable" content="yes" />
  6. <meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1, maximum-scale=1"/>
  7. <meta name="theme-color" content="#4b75ff">
  8. <link rel="stylesheet" href="../script/tocas/tocas.css">
  9. <link rel="stylesheet" href="./main.css">
  10. <script src="../script/tocas/tocas.js"></script>
  11. <script src="../script/jquery.min.js"></script>
  12. <script src="../script/ao_module.js"></script>
  13. <!-- Handle native playback on Chrome Andoird-->
  14. <script src="native.js"></script>
  15. <link rel="manifest" crossorigin="use-credentials" href="manifest.json">
  16. <title>AirMusic</title>
  17. </head>
  18. <body>
  19. <div id="mainMenu" class="ts basic fluid AMmenu menu bottomBlue" style="z-index:80;position:fixed;position:top:0px;left:0px;width:100%;">
  20. <button class="ts item icon noBorder AMmenu button" onClick="toggleLeftMenu();"><i class="content icon"></i></button>
  21. <div class="item noBorder AMmenu" style="padding-right:0px;padding-top:12px;"><i id="AMmenuIcon" class="music icon large whiteFont"></i></div>
  22. <div class="item noBorder AMmenu" style="padding-left:3px;padding-top:3px;padding-bottom:5px;min-width:50%;">
  23. <div class="ts header whiteFont">
  24. <div id="interfaceTitle" style="display:inline;">Music</div>
  25. <div id="interfaceDetails" class="sub header" style="font-size:80%;line-height:1em;">[0 songs]</div>
  26. </div>
  27. </div>
  28. <button class="ts right item icon AMmenu noBorder button" onClick="enterSearchMode();"><i class="search icon"></i></button>
  29. <!-- <button class="ts right item icon AMmenu noBorder button"><i class="ellipsis vertical icon"></i></button> -->
  30. </div>
  31. <div id="searchModeMenu" class="ts basic fluid AMmenu menu bottomBlue" style="z-index:85;position:fixed;position:top:0px;left:0px;width:100%;height:51px;display:none;">
  32. <button class="ts item icon noBorder AMmenu button" onClick="exitSearchMode();"><i class="arrow left icon"></i></button>
  33. <div class="ts fluid input" >
  34. <input id="searchInputBar" type="text" style="background-color:rgb(48, 48, 48);border:0px !important;border-bottom:2px solid rgb(60, 60, 60) !important;margin-bottom:3px;color:white;" placeholder="Search Music">
  35. </div>
  36. <button class="ts right item icon AMmenu noBorder button" onClick="searchSong();"><i class="search icon"></i></button>
  37. </div>
  38. <!-- The main list that display the current information-->
  39. <div id="mainList">
  40. </div>
  41. <!-- Advance functions config menu-->
  42. <div id="showmoreUIcover" style="position:fixed;left:0px;top:0px;width:100%;height:100%;background:rgba(30,30,30,0.7);z-index:95;display:none;" onClick="hideShowMoreMenu();"></div>
  43. <div id="showMoreUI" class="showMoreMenus" style="display:none;">
  44. <div class="ts large header whiteFont songTitle">N/A</div>
  45. <div class="showMoreMenuItem" onClick="playFromShowMoreMenu();">Play</div>
  46. <div class="showMoreMenuItem" onclick='addToPlaylistFromMoreMenu();'>Add to Playlist</div>
  47. <div class="showMoreMenuItem playlistonly" onclick="removeFromPlylistFromMoreMenu();">Remove from current Playlist</div>
  48. <div class="showMoreMenuItem" onClick="searchYoutubeViaShowMore();">Search on Youtube</div>
  49. <div class="showMoreMenuItem">Edit Tag</div>
  50. <div class="showMoreMenuItem">Share</div>
  51. <div class="showMoreMenuItem" onClick="startRelatedSearch();">Related Search</div>
  52. <div class="showMoreMenuItem" onClick="showFileInfo();">File Information</div>
  53. <div class="topRightCorner songID" align="center" style="margin-top:10px !important;padding-top:3px !important;">-1</div>
  54. </div>
  55. <div id="showFileInfo" class="showMoreMenus" style="display:none;">
  56. <div class="ts large header">
  57. <span class="filename whiteFont">Title</span>
  58. <div class="sub header rawname whiteFont" style="word-break: break-all !important; font-size:80%;">Storage Name</div>
  59. </div>
  60. <div class="ts list whiteFont">
  61. <div class="item whiteFont">Storage Location:   <span class="filepath" style="word-break:break-all;">N/A</span></div>
  62. <div class="item whiteFont">File Size:   <span class="filesize">N/A</span></div>
  63. <div class="item whiteFont">Last Modification Date:   <span class="filedate">N/A</span></div>
  64. </div>
  65. </div>
  66. <!-- Side bar -->
  67. <div id="sideBarCover" style="position:fixed;left:0px;top:0px;width:100%;height:100%;background:rgba(30,30,30,0.7);z-index:50;display:none;" onClick="toggleLeftMenu();"></div>
  68. <div id="leftSideBar" class="leftsb" style="max-width:60%;height:100%;z-index:90;display:none;">
  69. <div class="leftsbObject sidebarBanner">
  70. <img class="ts tiny middle aligned image" src="img/main_icon.png" style="max-width: 50px;"><h5 class="whiteFont" style="display:inline;padding-left:8px;">AirMusic</h5>
  71. </div>
  72. <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadSongList();setStorage('viewingTab','music');">
  73. <div class="ts header">
  74. <i class="music tiny icon whiteFont"></i>
  75. <div class="content whiteFont">
  76. Music
  77. </div>
  78. </div>
  79. </div>
  80. <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadFolderView();setStorage('viewingTab','folder');">
  81. <div class="ts header">
  82. <i class="folder tiny icon whiteFont"></i>
  83. <div class="content whiteFont">
  84. Folder
  85. </div>
  86. </div>
  87. </div>
  88. <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadPlaylistView(); setStorage('viewingTab','playlist');">
  89. <div class="ts header">
  90. <i class="align justify tiny icon whiteFont"></i>
  91. <div class="content whiteFont">
  92. Playlists
  93. </div>
  94. </div>
  95. </div>
  96. <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadNetworkView(); setStorage('viewingTab','network');">
  97. <div class="ts header">
  98. <i class="world tiny icon whiteFont"></i>
  99. <div class="content whiteFont">
  100. Network
  101. </div>
  102. </div>
  103. </div>
  104. <!--
  105. <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="">
  106. <div class="ts header">
  107. <i class="upload icon whiteFont"></i>
  108. <div class="content whiteFont">
  109. Upload
  110. </div>
  111. </div>
  112. </div>
  113. -->
  114. <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;border-top: 2px solid #4a4a4a;">
  115. <div class="ts header">
  116. <i class="setting tiny icon whiteFont"></i>
  117. <div class="content whiteFont">
  118. Settings
  119. </div>
  120. </div>
  121. </div>
  122. </div>
  123. <!-- Mini player controller put on the bottom side of the list interface-->
  124. <div id="miniPlayer" class="hidden">
  125. <div style="padding-left:20px;padding-top:5px;" >
  126. <div class="ts header selectable" style="margin:0px !important;width:300px;cursor:pointer;" onClick="showMainPlayerInterface();">
  127. <i class="music icon whiteFont" style="overflow:hidden;"></i>
  128. <div class="content whiteFont" style="padding-top:5px;line-height:1em;">
  129. <div id="miniPlayerDisplayName" style="text-overflow: ellipsis;overflow: hidden;max-width:200px; white-space: nowrap;">N/A</div>
  130. <div id="miniPlayerInformation" class="sub header" style="color: #c7c7c7;size:95%;">No information</div>
  131. </div>
  132. </div>
  133. </div>
  134. <div id="miniPlayerIDtab" class="miniPlayer minitab">
  135. 0/0
  136. </div>
  137. <div class="miniPlayerControls">
  138. <button class="miniPlayer selectable" style="font-size:90%;" onClick="previousSong();"><i class="step backward icon whiteFont"></i></button>
  139. <button class="miniPlayer selectable" style="font-size:120%;" onClick="togglePlayMode();"><i class="play icon whiteFont PlayButton"></i></button>
  140. <button class="miniPlayer selectable" style="font-size:90%;" onClick="nextSong();"><i class="step forward icon whiteFont "></i></button>
  141. </div>
  142. </div>
  143. <!-- Main interface for player and audio controller-->
  144. <div id="playerInterface" style="display:none;">
  145. <div id="infoBar" class="whiteFont" style="padding:20px;" align="center">
  146. <!-- Top Information Bar-->
  147. <div class="whiteFont songTitleWrapper" style="margin:0px;display:inline-block;">
  148. <div id="mainPlayerSongTitle" class="whiteFont" style="font-weight: bold;font-size:120%;">Song Title</div>
  149. <small id="mainPlayerSongDesc">This is some small text for display</small>
  150. </div>
  151. <div style="position:absolute;left:20px;top:30px;cursor:pointer;" onClick="hideMainPlayerInterface();">
  152. <i class="large chevron left icon"></i>
  153. </div>
  154. <div style="position:absolute;right:20px;top:30px;cursor:pointer;" onClick='$("#dropdownSonglist").slideDown();'>
  155. <i class="large content icon"></i>
  156. </div>
  157. </div>
  158. <div style="padding-left:20px; padding-right:20px;padding-bottom:5px;" align="center">
  159. <!-- Function Menu Bar-->
  160. <div class="ts grid">
  161. <div class="two wide column"><button style="cursor: pointer;" onclick="showPlaylistInterface();"><i class="large plus icon whiteFont"></i></button></div>
  162. <div class="two wide column"><button id="downloadBtn" style="cursor: pointer;" onClick="downloadPlayingSong();"><i class="large download icon whiteFont"></i></button></div>
  163. <div class="two wide column"><button style="cursor: pointer;" onClick="openInFileExplorer();"><i class="large folder open icon whiteFont"></i></button></div>
  164. <div class="two wide column desktopOnly"><button style="cursor: pointer;" onClick="mute();" ><i class="large volume off icon whiteFont"></i></button></div>
  165. <div class="two wide column desktopOnly"><button style="cursor: pointer;" onClick="addAudioVolume(-0.05);"><i class="large volume down icon whiteFont"></i></button></div>
  166. <div class="two wide column desktopOnly"><button style="cursor: pointer;" onClick="addAudioVolume(0.05);"><i class="large volume up icon whiteFont"></i></button></div>
  167. <div class="two wide column"><button style="cursor: pointer;" onClick="showTimerInterface();"><i class="large clock icon whiteFont"></i></button></div>
  168. <div class="two wide column"><button style="cursor: pointer;" onClick="showSettingInterface();"><i class="large setting icon whiteFont"></i></button></div>
  169. </div>
  170. <div id="volumeGUI">
  171. <i class="volume off large icon whiteFont" style="position:absolute;left:12%;top:10px;"></i>
  172. <div class="ts tiny progress">
  173. <div id="volBar" class="bar" style="width: 40%"></div>
  174. </div>
  175. <i class="volume up large icon whiteFont" style="position:absolute;right:12%;top:10px;"></i>
  176. </div>
  177. </div>
  178. <!-- Download progress bar-->
  179. <div id="downloadProgressBar" class="ts attached fluid tiny progress" style="background-color:rgba(20, 20, 20, 0.9);display:none;">
  180. <div class="bar" style="width: 0%;background-color:rgb(75, 117, 255) !important;"></div>
  181. </div>
  182. <div id="albumnArt" align="center">
  183. <img id="albumnArtImage" style="max-width: 80%;" src="img/default.png">
  184. </div>
  185. <div id="mainPlayerControlInterface">
  186. <!-- main control interface-->
  187. <div style="width:100%;" align="center">
  188. <div id="progressTime">
  189. 0:00
  190. </div>
  191. <div id="mainPlayerMiniTab" class="mainPlayer minitab">
  192. 0/0
  193. </div>
  194. <div id="remainTime">
  195. -0:00
  196. </div>
  197. </div>
  198. <div style="position:absolute;width:100%;left:0px;bottom:0px;">
  199. <div style="padding-left:10px;padding-right:10px;">
  200. <div class="ts small primary progress">
  201. <div id="audioProgressBar" class="bar" style="width: 100%; background: #4b75ff;"></div>
  202. </div>
  203. </div>
  204. <div class="mainControlButtons">
  205. <button style="font-size:90%;padding:15px;" onClick="previousSong();"><i class="step backward icon whiteFont"></i></button>
  206. <button style="font-size:130%;" onClick="togglePlayMode();"><i class="play icon whiteFont PlayButton"></i></button>
  207. <button style="font-size:90%;padding:15px;" onClick="nextSong();"><i class="step forward icon whiteFont "></i></button>
  208. </div>
  209. <div id="randomModeButton" style="position: absolute; left:30px;bottom:20px;" onClick="toggleRandomMode(this);">
  210. <i class="large random icon disabled"></i>
  211. </div>
  212. <div id="repeatModeButton" style="position: absolute; right:30px;bottom:20px;" onClick="toggleRepeatMode(this);">
  213. <i class="large retweet icon disabled"></i>
  214. <p class="whiteFont singleLoop" style="position:absolute;right:0px;bottom:0px;margin-bottom:-13px;margin-right:-2px;display:none;">1</p>
  215. </div>
  216. </div>
  217. </div>
  218. <audio id="mainAudioPlayer" style="display:none;" src=""></audio>
  219. <div id="dropdownSonglist" class="dropdownMusicList" style="display:none;">
  220. <div class="dropdownMusicListMiniMenu">
  221. <i class="icons" style="margin-right:8px;">
  222. <i class="content icon"></i>
  223. <i class="corner music icon" style='color:#333333'></i>
  224. </i>
  225. Now Playing
  226. <div style="position:absolute;top:3px;right:5px;">
  227. [ <span id="dropdownListSongCount">N/A</span> songs / <span id="dropdownListIDCount">N/A</span> MB ] <button onClick='$("#dropdownSonglist").slideUp();' style="cursor:pointer;"><i class="caret up icon whiteFont"></i></button>
  228. </div>
  229. </div>
  230. <div id="currentPlayingMainList" style="overflow-x:hidden;">
  231. </div>
  232. <div class="dropdownMusicListBottom" align="center" onClick='$("#dropdownSonglist").slideUp();'><i class="caret up icon"></i></div>
  233. </div>
  234. </div>
  235. <!-- Quick Menus and other tool windows-->
  236. <!-- Timer control interface-->
  237. <div id="timerInterface" class="quickMenu">
  238. <h4 class="whiteFont">Countdown Timer</h4>
  239. <br>
  240. <div id="timerSettingInterface" style="width:100%;" align="center">
  241. <div>
  242. <div class="ts statistic">
  243. <div class="label whiteFont">hours</div>
  244. <div id="timerHour" class="value whiteFont">0</div>
  245. </div>
  246. <div class="ts statistic">
  247. <div class="label whiteFont">minutes</div>
  248. <div id="timerMinute" class="value whiteFont">0</div>
  249. </div>
  250. </div>
  251. <div>
  252. <div class="ts icon mini buttons">
  253. <button class="ts inverted button" onClick="addTimer('h',1);"><i class="add icon"></i></button>
  254. <button class="ts inverted button" onClick="addTimer('h','r');"><i class="refresh icon"></i></button>
  255. <button class="ts inverted button" onClick="addTimer('h',-1);"><i class="minus icon"></i></button>
  256. </div>
  257. <div class="ts icon mini buttons">
  258. <button class="ts inverted button" onClick="addTimer('m',10);"><i class="add icon"></i></button>
  259. <button class="ts inverted button" onClick="addTimer('m','r');"><i class="refresh icon"></i></button>
  260. <button class="ts inverted button" onClick="addTimer('m',-1);"><i class="minus icon"></i></button>
  261. </div>
  262. </div>
  263. </div>
  264. <div id="timerCountingInterface" style="width:100%; display:none;" align="center">
  265. <div class="ts statistic">
  266. <div id="remainingUI" class="value whiteFont">0:00:00</div>
  267. <div class="label whiteFont">Remaining</div>
  268. </div>
  269. </div>
  270. <div class="ts divider"></div>
  271. <div>
  272. <div class="ts grid">
  273. <div class="eight wide column">
  274. Times up action
  275. </div>
  276. <div class="eight wide column">
  277. <select id="stopMode" class="ts basic inverted mini dropdown">
  278. <option>Fade Out & Stop</option>
  279. <option>Stop Playing</option>
  280. </select>
  281. </div>
  282. </div>
  283. </div>
  284. <div style="width:100%;position:absolute;left:0px;bottom:0px; margin-bottom: 10px;;" align="center">
  285. <div class="ts grid" >
  286. <div class="eight wide column" style="padding-right: 0px;" align="center"><button class="greenBtn" onClick='hideAllquickMenu();'>Close</button></div>
  287. <div class="eight wide column" style="padding-left: 0px;" align="center"><button class="greenBtn" onClick="toggleCountDown(this);">Start</button></div>
  288. </div>
  289. </div>
  290. </div>
  291. <!-- File properties interface-->
  292. <div id="filepropInterface" class="quickMenu">
  293. <div class="ts header">
  294. <span class="filename whiteFont">Title</span>
  295. <div class="sub header rawname whiteFont" style="word-break:break-all;">Storage Name</div>
  296. </div>
  297. <div class="ts list whiteFont">
  298. <div class="item whiteFont">Storage Location:   <span class="filepath" style="word-break:break-all;">N/A</span></div>
  299. <div class="item whiteFont">File Size:   <span class="filesize">N/A</span></div>
  300. <div class="item whiteFont">Last Modification Date:   <span class="filedate">N/A</span></div>
  301. </div>
  302. </div>
  303. <!-- Setting interface-->
  304. <div id="settingInterface" class="quickMenu">
  305. <div style="">
  306. <div class="ts relaxed list" style="padding:0px !important;">
  307. <div id="settingMenuTitle" class="item whiteFont" style="">N/A</div>
  308. <div class="item selectable whiteFont" onclick="showPlaylistInterface(); $('#settingInterface').fadeOut('fast');"><i class="add icon" style="margin-right:5px;"></i> Add to playlist</div>
  309. <div class="item selectable whiteFont" onClick="openInFileExplorer(); hideAllquickMenu();"><i class="folder open icon" style="margin-right:5px;"></i> Open in File Explorer</div>
  310. <div class="item selectable whiteFont" onClick="searchOnYoutube(); hideAllquickMenu();"><i class="youtube play icon" style="margin-right:5px;"></i>Search on Youtube</div>
  311. <div class="item selectable whiteFont" onClick="showFileInformation();"><i class="file icon" style="margin-right:5px;"></i> File Information</div>
  312. <div class="ts divider"></div>
  313. <div id="closeWebAppButton" class="item selectable whiteFont" onClick="ao_module_close();"><i class="remove icon" style="margin-right:5px;"></i> Exit Application</div>
  314. </div>
  315. </div>
  316. </div>
  317. <!-- Playlist interface-->
  318. <div id="playlistInterface" class="quickMenu">
  319. <div class="ts header">
  320. <span class="whiteFont">Add to Playlist</span>
  321. </div>
  322. <div class="playlist item selectable whiteFont" onclick="addToNewPlaylist();">
  323. <i class="add icon whiteFont"></i> New Playlist
  324. </div>
  325. <!-- playlist list-->
  326. <div id="existsingPlaylist" style="overflow-y: scroll; height: 50%; border: 1px solid white;">
  327. <div class="playlist item selectable whiteFont">
  328. <i class="list icon whiteFont"></i> Test Playlist
  329. </div>
  330. </div>
  331. <br>
  332. <div style="width:100%;position:absolute;left:0px;bottom:0px; margin-bottom: 10px;;" align="center">
  333. <button class="greenBtn" onClick='hideAllquickMenu();'>Close</button>
  334. </div>
  335. <div class="ts snackbar" id="succSnackBar">
  336. <div class="content">
  337. Action Completed
  338. </div>
  339. </div>
  340. </div>
  341. <!-- Utils Interfaces, usually hidden-->
  342. <div id="fadeReturnScreen" class="fadeScreen whiteFont" onClick="hideAllquickMenu();"></div>
  343. <script>
  344. var leftMenuShown = false;
  345. var currentPath = "";
  346. var currentMode = "music";
  347. var rootPaths = [];
  348. var totalMusicCount = 0;
  349. var currentPlaying = false;
  350. var audioElement = $("#mainAudioPlayer");
  351. var audioElementObject = document.getElementById("mainAudioPlayer"); //Directly expose the audio object
  352. var playingFileDetail = []; //[id, filepath]
  353. var displayList = []; //This is the list where the current UI is displaying.
  354. var playingList = []; //This is the list where the player last clicked on an item to play
  355. var randomMode = false;
  356. var repeatMode = 0; //Repeat mode 0 -> No repeat, 1 -> Repeat all 2 -> Repeat one
  357. var updateSystemVolume = true;
  358. var timerMode = false;
  359. var timerRemaining = 0;
  360. var playlistAddPendingFile = ""; //The filepath of the file that is pending to be added to playlist
  361. var timerEndMode = "fade"; //Fade or End, Provide volume fadeout or end immediately while times up
  362. var ua = navigator.userAgent.toLowerCase();
  363. var isAndroid = ua.indexOf("android") > -1; //&& ua.indexOf("mobile");
  364. var isChrome = /Chrome/.test(navigator.userAgent) && /Google Inc/.test(navigator.vendor);
  365. //Embed native mediaSession for Android
  366. if (isChrome){
  367. initNativeMediaPlayer();
  368. }
  369. //Update implementatio nof user agent detection
  370. if (typeof InstallTrigger !== 'undefined'){
  371. ua = "firefox";
  372. }
  373. if (!!window.chrome && (!!window.chrome.webstore || !!window.chrome.runtime)){
  374. ua = "chrome";
  375. }
  376. var isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
  377. if (isSafari == true){
  378. ua = "safari";
  379. }
  380. $(document).ready(function(){
  381. $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 300);
  382. //Update 7-8-2019, default loading tab is handled over to the returnToPreviousState() function.
  383. //loadSongList();
  384. //Initiate the module's volume to the system's
  385. syncSystemVol();
  386. initAudioListeners();
  387. resizeQuickAdjust(); //Adjust all the DOM items position according to window size
  388. restoreAllSettings(); //Restore all user prefered settings from storage
  389. hideMainPlayerInterface();//Hide the main interface
  390. setTimeout(function(){
  391. $("#playerInterface").show(); //Only show this interface after it has been out of the working area.
  392. },500);
  393. //Hide the exit application button if it is not in Virtual Desktop Mode
  394. if (!ao_module_virtualDesktop){
  395. $("#closeWebAppButton").hide();
  396. }
  397. returnToPreviousState(); //Try to return to previous state using the window hash value
  398. });
  399. function returnToPreviousState(){
  400. var hash = window.location.hash;
  401. if (hash != ""){
  402. //There are state to restore. Restore using the given hash value
  403. hash = decodeURIComponent(hash.substring(1));
  404. var previousState = JSON.parse(hash);
  405. console.log(previousState);
  406. var currentPath = previousState.path;
  407. var playingFileDetail = previousState.pfp;
  408. var initPauseState = previousState.pause;
  409. if (currentPath != ""){
  410. //Create a dummy folder and trick the system to load to the previous states
  411. var a = '<div></div>';
  412. a = $(a).attr("filepath",currentPath);
  413. openFolder(a[0],{playIDAfterOpen: playingFileDetail[0],startPaused: initPauseState});
  414. }else{
  415. //currentpath is empty, aka Music Mode
  416. loadSongList();
  417. }
  418. }else{
  419. //No state to restore. Restore Viewing Tab instead.
  420. var vt = loadStorage("viewingTab");
  421. if (vt != ""){
  422. if (vt == "music"){
  423. loadSongList();
  424. }else if(vt == "folder"){
  425. loadFolderView();
  426. }else if(vt == "playlist"){
  427. loadPlaylistView();
  428. }else if(vt == "network"){
  429. loadNetworkView();
  430. }
  431. }else{
  432. //Default action
  433. loadSongList();
  434. }
  435. }
  436. }
  437. function loadNetworkView(){
  438. currentMode = "network";
  439. }
  440. function initAudioListeners(){
  441. //Initiate audio progress bar and link it to the progress bar on the main player interface
  442. audioElement[0].addEventListener("timeupdate", function() {
  443. var currentTime = audioElement[0].currentTime;
  444. var duration = audioElement[0].duration;
  445. $('#audioProgressBar').css('width',((currentTime +.25)/duration)*100 + '%');
  446. updatePlaybackDisplayTime(currentTime,duration);
  447. });
  448. audioElement[0].onended = function() {
  449. nextSongHandler();
  450. };
  451. //Handle clicks on progress bar to skip through the playback
  452. $("#audioProgressBar").parent().on("click",function(e){
  453. var duration = audioElement[0].duration;
  454. var jumpTo = (e.offsetX / $(this).width()) * duration;
  455. if (!audioElement[0].paused){
  456. audioElement[0].pause();
  457. audioElement[0].currentTime = jumpTo;
  458. audioElement[0].play();
  459. }else{
  460. audioElement[0].currentTime = jumpTo;
  461. }
  462. });
  463. }
  464. function restoreAllSettings(){
  465. //Repeat mode setting
  466. var strRmode = loadStorage("repeatMode");
  467. if (strRmode != ""){
  468. repeatMode = parseInt(strRmode);
  469. setRepeatMode(repeatMode,$("#repeatModeButton")[0]);
  470. }
  471. //Random Mode setting
  472. var random = loadStorage("randomMode");
  473. if (random != ""){
  474. if (random == "true"){
  475. //random Mode is true when loaded from configuration. Toggle once to match the setting
  476. toggleRandomMode($("#randomModeButton")[0]);
  477. }
  478. }
  479. //Timer mode setting
  480. var TEM = loadStorage("timerEndMode");
  481. if (TEM != ""){
  482. if (TEM == "fade"){
  483. $("#stopMode").val("Fade Out & Stop");
  484. }else{
  485. $("#stopMode").val("Stop Playing");
  486. }
  487. }
  488. }
  489. //Setting menu function
  490. function searchOnYoutube(filename = ""){
  491. if (filename == ""){
  492. var filename = $("#mainPlayerSongTitle").text().trim();
  493. }
  494. var url = "https://www.youtube.com/results?search_query=" + filename ;
  495. window.open(url);
  496. }
  497. //Searching related functions
  498. var preSearchItemBuffer = "";
  499. var searchModeEnabled = false;
  500. function enterSearchMode(){
  501. if (searchModeEnabled){
  502. //Search mode already enabled
  503. return;
  504. }
  505. $("#searchModeMenu").show();
  506. $("#mainMenu").hide();
  507. preSearchItemBuffer = [$("#mainList").html(),$("#interfaceTitle").text(),$("#AMmenuIcon").attr("class")];
  508. searchModeEnabled = true;
  509. }
  510. function exitSearchMode(){
  511. $("#searchModeMenu").hide();
  512. $("#mainMenu").show();
  513. $("#searchInputBar").val("");
  514. if (preSearchItemBuffer != ""){
  515. //If previous record exists, recover the previous view from storage
  516. $("#mainList").html(preSearchItemBuffer[0]);
  517. $("#interfaceTitle").text(preSearchItemBuffer[1]);
  518. $("#AMmenuIcon").attr("class",preSearchItemBuffer[2])
  519. preSearchItemBuffer = "";
  520. }
  521. searchModeEnabled = false;
  522. }
  523. $("#searchInputBar").on("keyup",function(e){
  524. if(e.keyCode == 13){
  525. //Enter pressed. start searching
  526. searchSong();
  527. }
  528. });
  529. function searchSong(){
  530. var keyword = $("#searchInputBar").val();
  531. if (keyword.includes("?") || keyword.includes("&")){
  532. alert("Search keyword cannot contains & or ? symbol.");
  533. return;
  534. }
  535. //Clear the mainList, replace with a dummy loading screen
  536. $("#mainList").html('<div class="mainList item whiteFont">\
  537. <div class="ts header" style="margin:0px !important;padding-bottom:15px;padding-top:5px;">\
  538. <i class="loading spinner icon whiteFont" style="overflow:hidden;margin-top:5px;"></i>\
  539. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  540. Searching...\
  541. </div>\
  542. </div>\
  543. <div class="topRightCorner" align="center">\
  544. -1\
  545. </div>\
  546. </div>');
  547. ao_module_agirun("Music/functions/listSong.js", {
  548. listSong: "search:" + keyword
  549. }, function(data){
  550. currentPath = "";
  551. var template = '<div class="mainList item" filepath={filepath} id={id} rawname={rawname}>\
  552. <div class="ts header selectable" style="margin:0px !important;" onClick="playSong(this);">\
  553. <i class="music icon whiteFont" style="overflow:hidden;"></i>\
  554. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  555. {songtitle}\
  556. <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
  557. </div>\
  558. </div>\
  559. <div class="topRightCorner" align="center">\
  560. {id}\
  561. </div>\
  562. <div class="mainList rightFunctionBar" align="center" onClick="showMore(this);">\
  563. <i class="ellipsis vertical icon" style="top:30%;"></i>\
  564. </div>\
  565. </div>';
  566. if (data.length == 0){
  567. //No search results.
  568. $("#mainList").html("");
  569. $("#mainList").append('<div class="mainList item whiteFont">\
  570. <div class="ts header" style="margin:0px !important;padding-bottom:15px;padding-top:5px;">\
  571. <i class="remove icon whiteFont" style="overflow:hidden;margin-top:5px;"></i>\
  572. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  573. No result for keyword: ' + keyword + '.\
  574. </div>\
  575. </div>\
  576. <div class="topRightCorner" align="center">\
  577. -1\
  578. </div>\
  579. </div>');
  580. return;
  581. }
  582. $("#interfaceTitle").text("Search");
  583. $("#AMmenuIcon").attr("class","search icon large whiteFont")
  584. //Initialize the song list
  585. displayList = data;
  586. if (playingList == []){
  587. playingList = Array.from(displayList);
  588. }
  589. $("#mainList").html("");
  590. for ( var i =0; i < data.length; i++){
  591. var songInfo = data[i];
  592. var path = songInfo[0];
  593. var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
  594. var ext = songInfo[2];
  595. var size = songInfo[3];
  596. var box = template;
  597. box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
  598. box = replaceAll("{id}",i + 1,box); //User count from 1
  599. box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
  600. box = replaceAll("{songtitle}",displayname,box);
  601. box = replaceAll("{ext}",ext,box);
  602. box = replaceAll("{size}",size,box);
  603. $("#mainList").append(box);
  604. }
  605. totalMusicCount = data.length;
  606. $("#mainList").append("<br><br><br><br><br><br><br>");
  607. $("#interfaceDetails").text("[" + i + " songs]");
  608. highLightPlayingMusic();
  609. });
  610. }
  611. function enterRemotePlayID(object){
  612. var id = $(object).text().trim();
  613. $("#remotePlayID").val($("#remotePlayID").val() + id);
  614. }
  615. function ridOpr(opr){
  616. if (opr == 'backspace'){
  617. var currentText = $("#remotePlayID").val();
  618. var newText = currentText.substring(0,currentText.length - 1);
  619. $("#remotePlayID").val(newText);
  620. }else if (opr == 'reset'){
  621. $("#remotePlayID").val("");
  622. }
  623. }
  624. function openInFileExplorer(){
  625. var dirname = playingFileDetail[1].split("/");
  626. var fname = dirname.pop(); //Pop away the filename
  627. dirname = dirname.join("/") + "/";
  628. var path = dirname;
  629. if (dirname.substring(0,1) != "/"){
  630. //This is not absolute path. Start the path from airMusic
  631. path = "AirMusic/" + dirname;
  632. }
  633. path = path.replace("media?file=","");
  634. if (path.substring(0,1) == "/"){
  635. path = path.substring(1)
  636. }
  637. ao_module_openPath(path, fname);
  638. }
  639. //quickmenu Functions
  640. function hideAllquickMenu(){
  641. $(".quickMenu").fadeOut('fast');
  642. $("#fadeReturnScreen").fadeOut('fast');
  643. }
  644. //Setting related function
  645. function showSettingInterface(){
  646. $("#settingInterface").fadeIn('fast');
  647. $("#fadeReturnScreen").fadeIn('fast');
  648. //Update contents in the setting menu
  649. $("#settingMenuTitle").text($("#mainPlayerSongTitle").text());
  650. }
  651. //Timer related functions
  652. function showTimerInterface(){
  653. $("#timerInterface").fadeIn('fast');
  654. $("#fadeReturnScreen").fadeIn('fast');
  655. }
  656. function initPlaylistInterfaceList(){
  657. $("#existsingPlaylist").html("");
  658. //Get the playlist list from server side
  659. ao_module_agirun("Music/functions/playlist.js", {
  660. opr: "root"
  661. }, function(data){
  662. data.forEach(playlist => {
  663. $("#existsingPlaylist").append(`<div class="playlist item selectable whiteFont" name="${playlist.name}" onclick="addSongToSelectedPlaylist(this);">
  664. <i class="list icon whiteFont"></i> ${playlist.name} [${playlist.count} files]
  665. </div>`);
  666. })
  667. });
  668. }
  669. function showPlaylistInterface(){
  670. initPlaylistInterfaceList();
  671. $("#playlistInterface").fadeIn('fast');
  672. $("#fadeReturnScreen").fadeIn('fast');
  673. }
  674. function addToThisPlaylist(){
  675. }
  676. $("#stopMode").change(function () {
  677. var end = this.value;
  678. if (end.includes("Fade Out & Stop")){
  679. //Start fading out the volume
  680. timerEndMode = "fade";
  681. setStorage("timerEndMode","fade");
  682. }else{
  683. //End immediately
  684. timerEndMode = "end";
  685. setStorage("timerEndMode","end");
  686. }
  687. });
  688. var fadeOutStepping = 0.1; //The steps size for fading out during timer countdown
  689. var defaultVolumeBeforeFadeout = 0;
  690. function toggleCountDown(button){
  691. if (!timerMode){
  692. //Start count down
  693. timerMode = true;
  694. $(button).text("Stop");
  695. timerRemaining = ((parseInt($("#timerHour").text()) * 60) + parseInt($("#timerMinute").text())) * 60; //in seconds
  696. //Update the value on the remaining counter
  697. $("#remainingUI").text(parseTimer(timerRemaining));
  698. $("#timerSettingInterface").slideUp('fast');
  699. $("#timerCountingInterface").slideDown('fast');
  700. $("#stopMode").attr("disabled","");
  701. defaultVolumeBeforeFadeout = audioElement[0].volume;
  702. fadeOutStepping = (audioElement[0].volume) / timerRemaining;
  703. progressCounter();
  704. }else{
  705. //Stop count down
  706. timerMode = false;
  707. $(button).text("Start");
  708. timerRemaining = 0;
  709. $("#timerSettingInterface").slideDown('fast');
  710. $("#timerCountingInterface").slideUp('fast');
  711. $("#stopMode").removeAttr("disabled");
  712. }
  713. }
  714. function progressCounter(){
  715. //Update the timer
  716. timerRemaining = timerRemaining - 1;
  717. $("#remainingUI").text(parseTimer(timerRemaining));
  718. if (timerRemaining > 0){
  719. //Continues
  720. if (timerEndMode == "fade"){
  721. var newvol = audioElement[0].volume - fadeOutStepping;
  722. audioElement[0].volume = Math.max(0,newvol);
  723. }
  724. setTimeout(progressCounter,1000);
  725. }else{
  726. //Times up
  727. handleTimerEnd();
  728. }
  729. }
  730. function handleTimerEnd(){
  731. //Stop the playback
  732. originalVol = audioElement[0].volume;
  733. fadeAudio();
  734. setPlaying(false);
  735. audioElement[0].pause();
  736. setTimeout(function(){
  737. audioElement[0].volume = defaultVolumeBeforeFadeout;
  738. },500);
  739. }
  740. function parseTimer(remainingTime){
  741. //remainingTime in seconds, prase to HH:mm:ss
  742. var hours = Math.floor(remainingTime / 3600);
  743. var remainingTime = remainingTime % 3600;
  744. var minutes = Math.floor(remainingTime / 60);
  745. var seconds = remainingTime % 60;
  746. if (minutes < 10){
  747. minutes = "0" + minutes;
  748. }
  749. if (seconds < 10){
  750. seconds = "0" + seconds;
  751. }
  752. var formatted = hours + ":" + minutes + ":" + seconds;
  753. return formatted;
  754. }
  755. function addTimer(unit,value){
  756. if (unit == "h"){
  757. if (value == "r"){
  758. $("#timerHour").text("0");
  759. }else{
  760. $("#timerHour").text($("#timerHour").text() - -1 * value);
  761. }
  762. }else if (unit == "m"){
  763. if (value == "r"){
  764. $("#timerMinute").text("0");
  765. }else{
  766. $("#timerMinute").text($("#timerMinute").text() - -1 * value);
  767. }
  768. }
  769. if (parseInt($("#timerHour").text()) < 0){
  770. $("#timerHour").text("0");
  771. }
  772. if (parseInt($("#timerMinute").text()) < 0){
  773. if (parseInt($("#timerHour").text()) > 0){
  774. //subtract 1 from the hour
  775. $("#timerHour").text($("#timerHour").text() - 1);
  776. $("#timerMinute").text(parseInt($("#timerMinute").text()) + 60);
  777. }else{
  778. //There is no hours to subtract
  779. $("#timerMinute").text("0");
  780. }
  781. }
  782. if (parseInt($("#timerMinute").text()) > 59){
  783. $("#timerMinute").text(parseInt($("#timerMinute").text()) - 60);
  784. $("#timerHour").text($("#timerHour").text() - -1);
  785. }
  786. }
  787. //Download function
  788. function downloadPlayingSong(){
  789. var url = audioElement.attr("src");
  790. var ext = url.split(".").pop();
  791. var filename = $("#mainPlayerSongTitle").text().trim();
  792. generateDownloadElement(url,filename);
  793. /*
  794. if (url.includes("/media")){
  795. blobDownloadElement(url,filename);
  796. }else{
  797. generateDownloadElement(url,filename);
  798. }
  799. */
  800. }
  801. var downloadInProgress = false;
  802. function blobDownloadElement(filepath,filename){
  803. if (downloadInProgress){
  804. alert("Please wait until another download finished.");
  805. return;
  806. }
  807. downloadInProgress = true;
  808. $("#downloadBtn").find("i").addClass("disabled");
  809. $("#downloadProgressBar").find(".bar").css("width","0%");
  810. $("#downloadProgressBar").show();
  811. var xhr = new XMLHttpRequest();
  812. xhr.onreadystatechange = function(){
  813. if (this.readyState == 4 && this.status == 200){
  814. //handler(this.response);
  815. //console.log(this.response, typeof this.response);
  816. var url = window.URL || window.webkitURL;
  817. //Open the downloaded data in new window for testing
  818. //window.open(url.createObjectURL(this.response));
  819. generateDownloadElement(url.createObjectURL(this.response),filename);
  820. $("#downloadBtn").find("i").removeClass("disabled");
  821. $("#downloadProgressBar").hide();
  822. downloadInProgress = false;
  823. }
  824. }
  825. xhr.onprogress = function (event) {
  826. console.log("[AirMusic] Download Progress: " + event.loaded + " / " + event.total + "Bytes");
  827. $("#downloadProgressBar").find(".bar").css("width",(event.loaded / event.total) * 100 + "%");
  828. };
  829. xhr.open('GET', filepath);
  830. xhr.responseType = 'blob';
  831. xhr.send();
  832. }
  833. function generateDownloadElement(filepath, filename){
  834. var link = document.createElement('a');
  835. link.href = filepath + "&download=true";
  836. link.setAttribute('download', filename);
  837. document.getElementsByTagName("body")[0].appendChild(link);
  838. // Firefox
  839. if (document.createEvent) {
  840. var event = document.createEvent("MouseEvents");
  841. event.initEvent("click", true, true);
  842. link.dispatchEvent(event);
  843. }
  844. // IE
  845. else if (link.click) {
  846. link.click();
  847. }
  848. link.parentNode.removeChild(link);
  849. }
  850. //Volume control related features
  851. var previousAudioVolume = 0;
  852. function mute(){
  853. if (audioElement[0].volume != 0){
  854. //Set to mute
  855. previousAudioVolume = audioElement[0].volume;
  856. audioElement[0].volume = 0;
  857. }else{
  858. //Restore from mute
  859. audioElement[0].volume = previousAudioVolume;
  860. }
  861. $("#volBar").css("width",audioElement[0].volume * 100 + "%");
  862. }
  863. var volbarTimeoutEvt;
  864. function addAudioVolume(value){
  865. $("#volumeGUI").slideDown('fast');
  866. if (volbarTimeoutEvt !== undefined){
  867. clearTimeout(volbarTimeoutEvt);
  868. }
  869. volbarTimeoutEvt = setTimeout(function(){
  870. $("#volumeGUI").slideUp('fast');
  871. },3000);
  872. $("#volumeGUI")
  873. if (audioElement[0].volume + value >= 1){
  874. audioElement[0].volume = 1;
  875. }else if (audioElement[0].volume + value <= 0){
  876. audioElement[0].volume = 0;
  877. }else{
  878. audioElement[0].volume += value;
  879. }
  880. $("#volBar").css("width",Math.min(100,audioElement[0].volume * 100) + "%");
  881. }
  882. function updatePlaybackDisplayTime(currentTime,duration){
  883. let progress = Math.round(currentTime);
  884. let remainTime = Math.round(duration - currentTime);
  885. $("#progressTime").text(secondsToHms(progress));
  886. if (!isNaN(remainTime)){
  887. $("#remainTime").text("-" + secondsToHms(remainTime));
  888. }
  889. }
  890. function nextSongHandler(){
  891. console.log("Next song");
  892. if (repeatMode == 1){
  893. //Next song in the playlist
  894. if (randomMode){
  895. var randomTrackID = Math.floor(Math.random() * (playingList.length - 1));
  896. //console.log(randomTrackID, playingList[randomTrackID]);
  897. nextSong(randomTrackID,true);
  898. }else{
  899. var playingSongIndex = playingFileDetail[0];
  900. var nextSongIndex = playingSongIndex;
  901. if (nextSongIndex >= playingList.length){
  902. nextSongIndex = 0;
  903. }
  904. //console.log(playingSongIndex, nextSongIndex);
  905. nextSong(nextSongIndex,true);
  906. }
  907. }else if (repeatMode == 2){
  908. //Loop this song only
  909. audioElement[0].pause();
  910. audioElement[0].currentTime = 0;
  911. audioElement[0].play();
  912. }else if (repeatMode == 0){
  913. //Pause after finish this song
  914. setPlaying(false);
  915. audioElement[0].pause();
  916. audioElement[0].currentTime = 0;
  917. }
  918. }
  919. function mainPlayerShown(){
  920. if ($("#playerInterface").offset().left == 0){
  921. return true;
  922. }
  923. return false;
  924. }
  925. function toggleRandomMode(button){
  926. var btn = $(button).find("i");
  927. if (randomMode){
  928. //Set the random mode to false
  929. btn.removeClass("enabled").addClass("disabled");
  930. randomMode = false;
  931. }else{
  932. btn.removeClass("disabled").addClass("enabled");
  933. randomMode = true;
  934. }
  935. setStorage("randomMode",randomMode);
  936. }
  937. function toggleRepeatMode(button){
  938. if (repeatMode == 0){
  939. setRepeatMode(1,button);
  940. setStorage("repeatMode",1);
  941. }else if (repeatMode == 1){
  942. setRepeatMode(2,button);
  943. setStorage("repeatMode",2);
  944. }else if (repeatMode == 2){
  945. setRepeatMode(0,button);
  946. setStorage("repeatMode",0);
  947. }
  948. }
  949. function setRepeatMode(modeCode,button){
  950. if (modeCode == 1){
  951. //Set repeat mode to "Repeat all"
  952. $(button).find(".singleLoop").hide();
  953. $(button).find("i").removeClass("disabled").addClass("enabled");
  954. repeatMode = 1;
  955. }else if (modeCode == 2){
  956. //Set repeat mode to "Repeat one"
  957. $(button).find(".singleLoop").show();
  958. $(button).find("i").removeClass("disabled").addClass("enabled");
  959. repeatMode = 2;
  960. }else if (modeCode == 0){
  961. //Set repeat mode to "None"
  962. $(button).find(".singleLoop").hide();
  963. $(button).find("i").removeClass("enabled").addClass("disabled");
  964. repeatMode = 0;
  965. }
  966. }
  967. function hideMainPlayerInterface(){
  968. $("#playerInterface").animate({left: window.innerWidth},300);
  969. $("body").css("overflow-y","auto");
  970. }
  971. function showMainPlayerInterface(){
  972. $("#playerInterface").animate({left:0},300);
  973. $("body").css("overflow-y","hidden");
  974. }
  975. function syncSystemVol(){
  976. //Initiate the module's volume to the system's
  977. var globalvol = localStorage.getItem("global_volume");
  978. //console.log("Global Volume" + globalvol.toString());
  979. if (!globalvol){
  980. globalvol = 0.1;
  981. }
  982. //Check if it is mobile. If yes, always 100% and leave volume to system
  983. if (isMobile()){
  984. globalvol = 1;
  985. $(".desktopOnly").addClass("disabled");
  986. }
  987. audioElement[0].volume = parseFloat(globalvol);
  988. $("#volBar").css("width",audioElement[0].volume * 100 + "%");
  989. }
  990. function replaceAll(target, replace, original){
  991. return original.split(target).join(replace);
  992. };
  993. function playSong(object, startPaused = false){
  994. if ($(object).parent().hasClass("playingTrack")){
  995. //This is already the song that is currently playing. Show the main player interface instead.
  996. showMainPlayerInterface();
  997. resizeQuickAdjust();
  998. return;
  999. }
  1000. //$(".playingTrack").removeClass("playingTrack");
  1001. //$(object).parent().addClass("playingTrack");
  1002. var filepath = ao_module_utils.attrToObject($(object).parent().attr('filepath'));
  1003. var rawname = ao_module_utils.attrToObject($(object).parent().attr('rawname'));
  1004. var displayName = ao_module_codec.decodeUmFilename(rawname);
  1005. var info = $(object).parent().find(".fileinfo").text();
  1006. var id = $(object).parent().attr('id');
  1007. if (id === undefined){
  1008. //This might be a file from dropdown list. use listid instead
  1009. id = $(object).parent().attr('listid');
  1010. }
  1011. updateMiniPlayerUI(displayName,info,id);
  1012. updateMainPlayerUI(displayName,info,id);
  1013. loadAndPlayAudioFile(filepath, !startPaused);
  1014. if (!currentPlaying && !startPaused){
  1015. setPlaying(true);
  1016. }
  1017. playingFileDetail = [parseInt(id),filepath];
  1018. //If the user play this file, assume the next file to be added to playlist is this file
  1019. playlistAddPendingFile = filepath;
  1020. //Have a backup of the current list just incase the user open another directory while a list is playing
  1021. //Only backup when it is not call from dropdown list
  1022. if (!$(object).parent().hasClass("dropdownList")){
  1023. playingList = Array.from(displayList);
  1024. }
  1025. parsePlayingSongList();
  1026. if ($("#miniPlayer").hasClass("hidden")){
  1027. //This is the first song that has played after the UI init
  1028. $("#miniPlayer").css("display","none").removeClass("hidden").slideDown();
  1029. }
  1030. showMainPlayerInterface();
  1031. //Just in case the albumn image changed, albumn art css is also updated.
  1032. resizeQuickAdjust();
  1033. updateStateReferingURL();
  1034. highLightPlayingMusic();
  1035. }
  1036. function loadThumbnail(filepath){
  1037. //Load the thumbnail
  1038. var realVpath = filepath.split("=");
  1039. realVpath.shift();
  1040. realVpath = realVpath.join("=");
  1041. ao_module_agirun("Music/functions/getThumbnail.js", {
  1042. file: realVpath,
  1043. }, function(data){
  1044. if (data.error !== undefined){
  1045. console.log(data.error)
  1046. $("#albumnArtImage").attr("src","img/default.png");
  1047. if (isAndroid && navigator.mediaSession.metadata){
  1048. navigator.mediaSession.metadata.artwork = [
  1049. { src: "img/default.png", sizes: '512x512', type: 'image/png' }
  1050. ]
  1051. }
  1052. }else{
  1053. $("#albumnArtImage").attr("src","data:image/jpg;base64," + data);
  1054. if (isAndroid && navigator.mediaSession.metadata){
  1055. navigator.mediaSession.metadata.artwork = [
  1056. { src: "data:image/jpg;base64," + data, sizes: '480x480', type: 'image/jpg' }
  1057. ]
  1058. }
  1059. }
  1060. resizeQuickAdjust();
  1061. });
  1062. }
  1063. function nextSong(id = false,forcePlayEvenStopped = false){
  1064. //Check if the current playlist has more than one songs.
  1065. var currentPaused = audioElement[0].paused;
  1066. if (playingList.length > 1){
  1067. var nextSongIndex = playingFileDetail[0];
  1068. if (id != false){
  1069. //Allow manual overwrite for this function
  1070. nextSongIndex = id;
  1071. var nextSong = playingList[nextSongIndex];
  1072. }else{
  1073. if (playingFileDetail[0] >= playingList.length){
  1074. var nextSong = playingList[0]; //Back to the first song in list
  1075. nextSongIndex = 0;
  1076. }else{
  1077. var nextSong = playingList[nextSongIndex]; //Index of the playList + 1 as the index for the files is alrady i+1. (User count from one)
  1078. }
  1079. }
  1080. //The songs has to be play via background services instead of relying on DOM
  1081. var filepath = nextSong[0];
  1082. var rawname = nextSong[1];
  1083. var displayName = ao_module_codec.decodeUmFilename(rawname);
  1084. var info = nextSong[2] + " / " + nextSong[3];
  1085. var id = nextSongIndex + 1;
  1086. updateMiniPlayerUI(displayName,info,id);
  1087. updateMainPlayerUI(displayName,info,id);
  1088. if (forcePlayEvenStopped){
  1089. loadAndPlayAudioFile(filepath,true);
  1090. }else{
  1091. loadAndPlayAudioFile(filepath,!currentPaused);
  1092. }
  1093. if (currentPlaying){
  1094. setPlaying(true);
  1095. }
  1096. playingFileDetail = [id,filepath];
  1097. //Need not to update the playlist because it is the same
  1098. highLightPlayingMusic();
  1099. }else{
  1100. //This is the only song in playlist. Play again from start
  1101. audioElement[0].pause();
  1102. audioElement[0].currentTime = 0;
  1103. if (!currentPaused){
  1104. audioElement[0].play();
  1105. }
  1106. }
  1107. updateStateReferingURL();
  1108. }
  1109. function previousSong(){
  1110. var currentPaused = audioElement[0].paused;
  1111. if (playingList.length > 1){
  1112. var nextSongIndex = playingFileDetail[0] - 1;
  1113. if (nextSongIndex == 0){
  1114. nextSongIndex = playingList.length;
  1115. var nextSong = playingList[nextSongIndex - 1]; //Back to the last song in list
  1116. }else{
  1117. var nextSong = playingList[playingFileDetail[0] - 2]; //Index of the playList - 1 as the index for the files is alrady i+1. (User count from one)
  1118. }
  1119. //The songs has to be play via background services instead of relying on DOM
  1120. var filepath = nextSong[0];
  1121. var rawname = nextSong[1];
  1122. var displayName = ao_module_codec.decodeUmFilename(rawname);
  1123. var info = nextSong[2] + " / " + nextSong[3];
  1124. var id = nextSongIndex;
  1125. updateMiniPlayerUI(displayName,info,id);
  1126. updateMainPlayerUI(displayName,info,id);
  1127. loadAndPlayAudioFile(filepath,!currentPaused);
  1128. if (currentPlaying){
  1129. setPlaying(true);
  1130. }
  1131. playingFileDetail = [id,filepath];
  1132. //Need not to update the playlist because it is the same
  1133. highLightPlayingMusic();
  1134. }else{
  1135. //This is the only song in playlist. Play again from start
  1136. audioElement[0].pause();
  1137. audioElement[0].currentTime = 0;
  1138. audioElement[0].play();
  1139. }
  1140. updateStateReferingURL();
  1141. }
  1142. var originalVol = 0;
  1143. function togglePlayMode(){
  1144. if (currentPlaying){
  1145. originalVol = audioElement[0].volume;
  1146. fadeAudio();
  1147. setPlaying(false);
  1148. }else{
  1149. gainAudio();
  1150. setPlaying(true);
  1151. }
  1152. }
  1153. let audioTransitioning = false;
  1154. function fadeAudio(){
  1155. audioTransitioning = true;
  1156. if(audioElement[0].volume > 0){
  1157. var testval = audioElement[0].volume - 0.1;
  1158. audioElement[0].volume = Math.max(0,testval);
  1159. setTimeout(fadeAudio, 50);
  1160. }else{
  1161. audioElement[0].pause();
  1162. setTimeout(function(){
  1163. audioElement[0].volume = originalVol;
  1164. updateStateReferingURL();
  1165. audioTransitioning = false;
  1166. }, 100);
  1167. }
  1168. }
  1169. var targetVol = 0;
  1170. function gainAudio(){
  1171. targetVol = audioElement[0].volume;
  1172. audioElement[0].volume = 0;
  1173. setTimeout(function(){
  1174. audioElement[0].play();
  1175. recursiveGainAudio();
  1176. }, 100);
  1177. }
  1178. function recursiveGainAudio(){
  1179. audioTransitioning = true;
  1180. if(audioElement[0].volume < targetVol){
  1181. if (audioElement[0].volume + 0.1 < 1){
  1182. audioElement[0].volume += 0.1
  1183. }else{
  1184. //Audio volume larger than 1, make it 1
  1185. audioElement[0].volume = 1
  1186. }
  1187. setTimeout(recursiveGainAudio, 50);
  1188. }else{
  1189. setTimeout(function(){
  1190. audioElement[0].volume = targetVol;
  1191. updateStateReferingURL();
  1192. audioTransitioning = false;
  1193. }, 100);
  1194. }
  1195. }
  1196. //Update all icons on play and pause buttons
  1197. function setPlaying(playing){
  1198. if (playing == true){
  1199. $(".PlayButton").attr("class","pause icon whiteFont PlayButton");
  1200. currentPlaying = true;
  1201. }else{
  1202. $(".PlayButton").attr("class","play icon whiteFont PlayButton");
  1203. currentPlaying = false;
  1204. }
  1205. }
  1206. function loadAndPlayAudioFile(sourceURL,playAfterLoad = true){
  1207. var audio = audioElement;
  1208. sourceURL = "../" + sourceURL; //Convert absolute dir to relative
  1209. $("#mainAudioPlayer").attr("src", sourceURL);
  1210. //console.log(encodeURIComponent(sourceURL));
  1211. audio[0].pause();
  1212. audio[0].load();
  1213. if(playAfterLoad){
  1214. audio[0].oncanplaythrough = audio[0].play();
  1215. }else{
  1216. var currentTime = audioElement[0].currentTime;
  1217. var duration = audioElement[0].duration;
  1218. $('#audioProgressBar').css('width','0%');
  1219. updatePlaybackDisplayTime(currentTime,duration);
  1220. }
  1221. loadThumbnail(sourceURL);
  1222. }
  1223. function updateMiniPlayerUI(displayName, fileinfo,id){
  1224. $("#miniPlayerDisplayName").text(displayName);
  1225. $("#miniPlayerInformation").text(fileinfo);
  1226. $("#miniPlayerIDtab").text(id + "/" + totalMusicCount);
  1227. }
  1228. function updateMainPlayerUI(songname, fileinfo, id){
  1229. $("#mainPlayerSongTitle").text(songname);
  1230. $("#mainPlayerSongDesc").text(fileinfo);
  1231. $("#mainPlayerMiniTab").text(id + "/" + totalMusicCount);
  1232. if (isAndroid && 'mediaSession' in navigator){
  1233. var infoRewrite = fileinfo.split(" / ")
  1234. updateTitle(songname, infoRewrite[1] + " (" + infoRewrite[0] + ")", id + "/" + totalMusicCount);
  1235. }
  1236. }
  1237. /*
  1238. Playlist related functions
  1239. */
  1240. function renderPlaylistByName(listname){
  1241. $("#interfaceTitle").text(listname);
  1242. ao_module_agirun("Music/functions/playlist.js", {
  1243. opr: "list",
  1244. playlistname: listname
  1245. }, function(data){
  1246. $("#mainList").html("");
  1247. if (data.error !== undefined){
  1248. //This playlist no longer exists. Back to main
  1249. loadPlaylistView();
  1250. }else{
  1251. //Updat ethe global values
  1252. displayList = data;
  1253. totalMusicCount = data.length;
  1254. currentPath = listname;
  1255. //Add the back btn for back to playlist view
  1256. $("#mainList").append(`<div class="mainList item extrapadding" onClick="loadPlaylistView();">
  1257. <div class="ts header selectable" style="margin:0px !important;">
  1258. <i class="angle left icon whiteFont" style="overflow:hidden;"></i>
  1259. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
  1260. ../
  1261. </div>
  1262. </div>
  1263. </div>`);
  1264. //List the resulting song list
  1265. for (var i = 0; i < data.length; i++){
  1266. var songInfo = data[i];
  1267. $("#mainList").append(`<div class="mainList item" filepath=${ao_module_utils.objectToAttr(songInfo[0])} id=${i+1} rawname=${ao_module_utils.objectToAttr(songInfo[1])}>
  1268. <div class="ts header selectable" style="margin:0px !important;" onClick="playSong(this);">
  1269. <i class="music icon whiteFont" style="overflow:hidden;"></i>
  1270. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
  1271. ${songInfo[1]}
  1272. <div class="sub header fileinfo" style="color: #c7c7c7;">${songInfo[2]} / ${songInfo[3]}</div>
  1273. </div>
  1274. </div>\
  1275. <div class="topRightCorner" align="center">
  1276. ${i + 1}
  1277. </div>
  1278. <div class="mainList rightFunctionBar" type="file" align="center" onclick="showMore(this);">
  1279. <i class="ellipsis vertical icon" style="top:30%;"></i>
  1280. </div>
  1281. </div>`);
  1282. //Update the interface Detail
  1283. $("#interfaceDetails").text("[" + data.length + " files]");
  1284. };
  1285. //Fix some legacy css isseus
  1286. $("#mainList").append("<br><br><br><br><br><br><br>");
  1287. }
  1288. });
  1289. }
  1290. //Open a given playlist
  1291. function openPlaylist(object){
  1292. var listname = $(object).attr("listname");
  1293. renderPlaylistByName(listname);
  1294. }
  1295. function addToNewPlaylist(){
  1296. var playlistname = prompt("Enter new playlist name");
  1297. if (playlistname != null && playlistname != ""){
  1298. //Add to playlist
  1299. addSongToPlayList(playlistname);
  1300. }else{
  1301. $("#succSnackBar").find(".content").html(`<i class="remove icon"></i> Invalid playlist name`);
  1302. $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
  1303. }
  1304. }
  1305. function addToPlaylistFromMoreMenu(){
  1306. $(".showMoreMenus").fadeOut('fast');
  1307. $("#showmoreUIcover").fadeOut('fast');
  1308. showPlaylistInterface();
  1309. }
  1310. //Remove the curernt file from the current playlist
  1311. function removeFromPlylistFromMoreMenu(){
  1312. if (currentMode == "playlist"){
  1313. //Remvoe the playlistAddPendingFile
  1314. //In playlist mode, the currentPath is used to store the plylist name
  1315. ao_module_agirun("Music/functions/playlist.js", {
  1316. opr: "remove",
  1317. playlistname: currentPath,
  1318. musicpath: playlistAddPendingFile.replace("/media?file=","")
  1319. }, function(data){
  1320. if (data.error !== undefined){
  1321. alert(data.error);
  1322. }else{
  1323. //Removed. Reload playlist
  1324. renderPlaylistByName(currentPath);
  1325. //Hide the MoreMenu
  1326. hideShowMoreMenu();
  1327. }
  1328. });
  1329. }
  1330. }
  1331. function addSongToSelectedPlaylist(object){
  1332. var playlistName = $(object).attr("name").trim();
  1333. addSongToPlayList(playlistName);
  1334. }
  1335. function addSongToPlayList(listname){
  1336. //Get current music file name
  1337. var currentSongPath = "";
  1338. if (playlistAddPendingFile !== ""){
  1339. currentSongPath = playlistAddPendingFile.replace("/media?file=","");
  1340. }else{
  1341. $("#succSnackBar").find(".content").html(`<i class="remove icon"></i> No song selected`);
  1342. $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
  1343. return;
  1344. }
  1345. //Add to playlist
  1346. ao_module_agirun("Music/functions/playlist.js", {
  1347. opr: "add",
  1348. playlistname: listname,
  1349. musicpath: currentSongPath
  1350. },function(data){
  1351. //Show success
  1352. $("#succSnackBar").find(".content").html(`<i class="checkmark icon"></i> Added to playlist ${listname}`);
  1353. $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
  1354. //Reload the playlist
  1355. initPlaylistInterfaceList();
  1356. });
  1357. }
  1358. //List the number of playlist stored in this database
  1359. function loadPlaylistView(){
  1360. currentMode = "playlist";
  1361. //Clear the main list
  1362. $("#mainList").html("");
  1363. ao_module_agirun("Music/functions/playlist.js", {
  1364. opr: "root",
  1365. },function(data){
  1366. //Get the list of playlist
  1367. console.log(data);
  1368. //Render the elements
  1369. data.forEach(playlist => {
  1370. $("#mainList").append(`<div class="mainList item" listname="${playlist.name}" tag="playlist" onClick="openPlaylist(this);">
  1371. <div class="ts header selectable" style="margin:0px !important;" >
  1372. <i class="folder open icon whiteFont" style="overflow:hidden;"></i>
  1373. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
  1374. ${playlist.name}
  1375. <div class="sub header fileinfo" style="color: #c7c7c7;">[${playlist.count} files]</div>
  1376. </div>
  1377. </div>
  1378. <div class="mainList rightFunctionBar" tag="moreInfo" align="center" onClick="openPlaylist(this);">
  1379. <i class="chevron right icon" style="top:30%;"></i>
  1380. </div>
  1381. </div>`);
  1382. });
  1383. hideLeftMenu();
  1384. //Update the headers
  1385. $("#interfaceTitle").text("Playlist");
  1386. $("#AMmenuIcon").attr("class","list icon large whiteFont");
  1387. $("#interfaceDetails").text("[" + data.length + " playlist]");
  1388. });
  1389. }
  1390. function loadFolderView(){
  1391. currentMode = "folder";
  1392. currentPath = "";
  1393. var tempalte = '<div class="mainList item" filepath={filepath} id={id} tag="folder" onClick="openFolder(this);">\
  1394. <div class="ts header selectable" style="margin:0px !important;" >\
  1395. <i class="folder open icon whiteFont" style="overflow:hidden;"></i>\
  1396. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  1397. {foldername}\
  1398. <div class="sub header fileinfo" style="color: #c7c7c7;">{fileinfo}</div>\
  1399. </div>\
  1400. </div>\
  1401. <div class="mainList rightFunctionBar" tag="moreInfo" align="center" onClick="moreFolder(this);">\
  1402. <i class="chevron right icon" style="top:30%;"></i>\
  1403. </div>\
  1404. </div>';
  1405. $("#interfaceTitle").text("Storage");
  1406. $("#AMmenuIcon").attr("class","folder open icon large whiteFont");
  1407. ao_module_agirun("Music/functions/listSong.js", {
  1408. listdir: "root",
  1409. },function(data){
  1410. $("#mainList").html("");
  1411. for (var i =0; i < data.length; i++){
  1412. var folderName = data[i][0];
  1413. var folderPath = data[i][1];
  1414. var fileCount = data[i][2];
  1415. var folderCount = data[i][3];
  1416. var box = tempalte;
  1417. box = replaceAll("{filepath}",folderPath,box);
  1418. box = replaceAll("{id}",i + 1,box);
  1419. if (folderName == ".cache" || folderName == ".trash"){
  1420. //Hidden folders
  1421. continue;
  1422. }
  1423. if (folderPath.includes("/media")){
  1424. //This is from external storage devices. List its number as well.
  1425. var tmp = folderPath.split("/");
  1426. var extStoragePath = "/" + tmp[1] + "/" + tmp[2];
  1427. box = replaceAll("{foldername}",folderName + " ( " + extStoragePath + " )",box);
  1428. }else{
  1429. //This is directory inside normal folders.
  1430. box = replaceAll("{foldername}",folderName,box);
  1431. }
  1432. var fileinfo = "[" + fileCount + " files]"
  1433. if (folderCount > 0){
  1434. fileinfo = "[" + fileCount + " files, " + folderCount +" folders]"
  1435. }
  1436. box = replaceAll("{fileinfo}",fileinfo,box);
  1437. rootPaths.push(folderPath);
  1438. $("#mainList").append(box);
  1439. }
  1440. $("#interfaceDetails").text("[" + data.length + " folders]");
  1441. });
  1442. hideLeftMenu();
  1443. }
  1444. function isMobile(){
  1445. if( /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) ) {
  1446. return true
  1447. }
  1448. return false
  1449. }
  1450. function setWindowHash(hashValue){
  1451. hashValue = JSON.stringify(hashValue);
  1452. if (!isAndroid){
  1453. return;
  1454. }
  1455. if(history.pushState) {
  1456. window.history.pushState(null, null, '#' + hashValue);
  1457. }else {
  1458. location.hash = '#' + hashValue;
  1459. }
  1460. if(ao_module_virtualDesktop && !isMobile()){
  1461. //Update the iframe src as well
  1462. var newsrc = window.frameElement.getAttribute("src");
  1463. if (newsrc.includes("#")){
  1464. newsrc = newsrc.split("#")
  1465. newsrc.pop();
  1466. newsrc = newsrc.join("#");
  1467. }
  1468. newsrc = newsrc + "#" + hashValue;
  1469. $(window.frameElement).attr("src",newsrc);
  1470. //console.log(window.frameElement.getAttribute("src"));
  1471. }
  1472. }
  1473. function updateStateReferingURL(){
  1474. var paused = audioElement[0].paused; //Check the current playing state of the player
  1475. setWindowHash({path: currentPath, pfp:playingFileDetail,pause:paused});
  1476. }
  1477. function openFolder(object, playParameters=null){
  1478. var playIDAfterOpen = -1;
  1479. var startPaused = false;
  1480. if (playParameters !== null){
  1481. playIDAfterOpen = playParameters.playIDAfterOpen;
  1482. startPaused = playParameters.startPaused;
  1483. }
  1484. var backbtnTemplate = '<div class="mainList item extrapadding" filepath={filepath} id={id} onClick="openFolder(this);">\
  1485. <div class="ts header selectable" style="margin:0px !important;">\
  1486. <i class="angle left icon whiteFont" style="overflow:hidden;"></i>\
  1487. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  1488. ../\
  1489. </div>\
  1490. </div>\
  1491. </div>';
  1492. var folderTemplate = '<div class="mainList item" filepath={filepath} tag="folder" id={id} onClick="openFolder(this);">\
  1493. <div class="ts header selectable" style="margin:0px !important;">\
  1494. <i class="folder open icon whiteFont" style="overflow:hidden;"></i>\
  1495. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  1496. {foldername}\
  1497. <div class="sub header fileinfo" style="color: #c7c7c7;">{fileinfo}</div>\
  1498. </div>\
  1499. </div>\
  1500. <div class="mainList rightFunctionBar" type="folder" align="center">\
  1501. <i class="chevron right icon" style="top:30%;"></i>\
  1502. </div>\
  1503. </div>';
  1504. var fileTemplate = '<div class="mainList item" filepath={filepath} id={id} rawname={rawname}>\
  1505. <div class="ts header selectable" style="margin:0px !important;" onClick="playSong(this);">\
  1506. <i class="music icon whiteFont" style="overflow:hidden;"></i>\
  1507. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  1508. {songtitle}\
  1509. <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
  1510. </div>\
  1511. </div>\
  1512. <div class="topRightCorner" align="center">\
  1513. {id}\
  1514. </div>\
  1515. <div class="mainList rightFunctionBar" type="file" align="center">\
  1516. <i class="ellipsis vertical icon" style="top:30%;"></i>\
  1517. </div>\
  1518. </div>';
  1519. var targetPath = decodeURIComponent($(object).attr("filepath"));
  1520. if (targetPath == "root" || targetPath == "/"){
  1521. //User request to go back to storage root
  1522. loadFolderView();
  1523. return;
  1524. }
  1525. currentPath = targetPath.split("/./").join("/");
  1526. currentPath = currentPath.split("//").join("/");
  1527. $("#interfaceTitle").text(ao_module_codec.decodeHexFoldername(targetPath.substring(0,targetPath.length -1).split("/").pop()));
  1528. var parentDir = currentPath.split("/");
  1529. parentDir.pop();parentDir.pop();
  1530. parentDir = (parentDir.join("/") + "/");
  1531. //console.log(currentPath);
  1532. var backbtn;
  1533. if (rootPaths.includes(currentPath) == false){
  1534. //We are currently not at root
  1535. backbtn = backbtnTemplate;
  1536. backbtn = replaceAll("{filepath}",encodeURIComponent(parentDir),backbtn);
  1537. backbtn = replaceAll("{id}",0,backbtn);
  1538. }else{
  1539. //Create a back button to go back to root of storage
  1540. backbtn = backbtnTemplate;
  1541. backbtn = replaceAll("{filepath}","root",backbtn);
  1542. backbtn = replaceAll("{id}",0,backbtn);
  1543. }
  1544. //Request the server for a list of folders and file in this directory
  1545. ao_module_agirun("Music/functions/listSong.js", {
  1546. listdir: targetPath.split("//").join("/"),
  1547. },function(data){
  1548. $("#mainList").html("");
  1549. $("#mainList").append(backbtn);
  1550. if (Array.isArray(data)){
  1551. var folders = data[0];
  1552. var files = data[1];
  1553. //List all folders
  1554. for (var i =0; i < folders.length; i++){
  1555. var folderName = folders[i][0];
  1556. var folderPath = folders[i][1];
  1557. var fileCount = folders[i][2];
  1558. var folderCount = folders[i][3];
  1559. var box = folderTemplate;
  1560. if (folderName == ".cache" || folderName == ".trash"){
  1561. //Hidden folders
  1562. continue;
  1563. }
  1564. box = replaceAll("{filepath}",encodeURIComponent(folderPath),box);
  1565. box = replaceAll("{id}",i + 1,box);
  1566. box = replaceAll("{foldername}",ao_module_codec.decodeHexFoldername(folderName),box);
  1567. var fileinfo = "[" + fileCount + " files]"
  1568. if (folderCount > 0){
  1569. fileinfo = "[" + fileCount + " files, " + folderCount +" folders]"
  1570. }
  1571. box = replaceAll("{fileinfo}",fileinfo,box);
  1572. $("#mainList").append(box);
  1573. }
  1574. //List all files
  1575. for ( var i =0; i < files.length; i++){
  1576. var songInfo = files[i];
  1577. var path = songInfo[0];
  1578. var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
  1579. var ext = songInfo[2];
  1580. var size = songInfo[3];
  1581. var box = fileTemplate;
  1582. box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
  1583. box = replaceAll("{id}",i + 1,box); //User count from 1
  1584. box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
  1585. box = replaceAll("{songtitle}",displayname,box);
  1586. box = replaceAll("{ext}",ext,box);
  1587. box = replaceAll("{size}",size,box);
  1588. $("#mainList").append(box);
  1589. }
  1590. displayList = files;
  1591. totalMusicCount = files.length;
  1592. $("#mainList").append("<br><br><br><br><br><br><br>");
  1593. }else{
  1594. //Something went wrong
  1595. //console.log('[AirMusic] Something went wrong: ' + data);
  1596. loadFolderView();
  1597. return;
  1598. }
  1599. if (folders.length > 0 && files.length > 0){
  1600. $("#interfaceDetails").text("[" + files.length + " files, " + folders.length+ " folders]");
  1601. }else if (folders.length > 0){
  1602. $("#interfaceDetails").text("[" + folders.length + " folders]");
  1603. }else if (files.length > 0){
  1604. $("#interfaceDetails").text("[" + files.length + " files]");
  1605. }else{
  1606. $("#interfaceDetails").text("[0 files or folders]");
  1607. }
  1608. //Hook all the events for the moreinfo on folders
  1609. hookMoreFolderInfoClickEvent();
  1610. //Check if the current playing file is located inside this list.
  1611. highLightPlayingMusic();
  1612. //Check if auto playback is required. If yes, play it with the given filepath.
  1613. if (playIDAfterOpen != -1){
  1614. $(".mainList.item").each(function(){
  1615. if ($(this).attr("id") == playIDAfterOpen){
  1616. //This is the file that require to playback after the folder loaded
  1617. playSong($(this).find(".ts.header.selectable"),startPaused);
  1618. setTimeout(resizeQuickAdjust,500);
  1619. }
  1620. });
  1621. }
  1622. });
  1623. }
  1624. function moreInfo(object){
  1625. if ($(object).parent().attr('tag') == "folder"){
  1626. //This is a folder. Ask if play as playlist
  1627. alert("Folder playback work in progress");
  1628. }else{
  1629. showMore(object);
  1630. }
  1631. }
  1632. function hookMoreFolderInfoClickEvent(){
  1633. $(".mainList.rightFunctionBar").on("click", function(e){
  1634. e.stopPropagation();
  1635. moreInfo(this);
  1636. });
  1637. }
  1638. function highLightPlayingMusic(){
  1639. //This function is used for highlighting the current playing music if the music piece is found inside the current list
  1640. $(".mainList.item").removeClass("playingTrack");
  1641. $(".mainList.item").each(function(){
  1642. var id = parseInt($(this).attr("id"));
  1643. try{
  1644. var filepath = ao_module_utils.attrToObject($(this).attr("filepath"));
  1645. }catch{
  1646. //Use back the previous method of filepath storage for compatibility
  1647. var filepath = $(this).attr("filepath");
  1648. }
  1649. /*
  1650. if (id == playingFileDetail[0] && filepath == playingFileDetail[1]){
  1651. //This piece of music is playing in the current list
  1652. $(this).addClass("playingTrack");
  1653. }else if (playingFileDetail[0] == -1 && filepath == playingFileDetail[1]){
  1654. //This is for external accessed music playback. Music file might or might not exists in the current playlist. Try to highlight anyway
  1655. $(this).addClass("playingTrack");
  1656. }*/
  1657. //Id is ignored in the 9/9/2019 updates and only check if the path matches.
  1658. if (filepath == playingFileDetail[1]){
  1659. $(this).addClass("playingTrack");
  1660. }
  1661. //console.log([id,filepath], playingFileDetail);
  1662. });
  1663. //Update dropdownList as well if exists
  1664. $(".dropdownList.item").removeClass("playingTrack");
  1665. $(".dropdownList.item").each(function(){
  1666. try{
  1667. var filepath = ao_module_utils.attrToObject($(this).attr("filepath"));
  1668. }catch{
  1669. //Use back the previous method of filepath storage for compatibility
  1670. var filepath = $(this).attr("filepath");
  1671. }
  1672. //var filepath = $(this).attr("filepath");
  1673. if (filepath == playingFileDetail[1]){
  1674. $(this).addClass("playingTrack");
  1675. }
  1676. });
  1677. }
  1678. function monitorFileChangeForUpload(){
  1679. if ($("#interfaceTitle").text() == "Upload"){
  1680. //It is still in upload Mode. Continue monitor file change and move them if needed
  1681. $.post("listenUploadChange.php",{fileList: JSON.stringify(currentUploadTargetFileList),directory: currentUploadTargetFilepath}).done(function(data){
  1682. if (data.includes("ERROR")){
  1683. console.log(data);
  1684. }
  1685. });
  1686. setTimeout(monitorFileChangeForUpload,1000);
  1687. }
  1688. }
  1689. function hideShowMoreMenu(){
  1690. $(".showMoreMenus").fadeOut('fast');
  1691. $("#showmoreUIcover").fadeOut('fast');
  1692. //Return the playlistAddPendingFile object back to the playing one
  1693. if (playingFileDetail !== undefined){
  1694. playlistAddPendingFile = playingFileDetail[1];
  1695. }else{
  1696. //No song is being play back
  1697. playlistAddPendingFile = "";
  1698. }
  1699. }
  1700. function playFromShowMoreMenu(){
  1701. nextSong(showMoreOprPlayID - 1);
  1702. $(".showMoreMenus").fadeOut('fast');
  1703. }
  1704. function showFileInfo(){
  1705. $("#showFileInfo").show();
  1706. $("#showMoreUI").hide();
  1707. }
  1708. function startRelatedSearch(){
  1709. enterSearchMode();
  1710. $("#searchInputBar").val(showMoreoprDisplayName);
  1711. $("#searchInputBar").focus();
  1712. hideShowMoreMenu();
  1713. }
  1714. function searchYoutubeViaShowMore(){
  1715. if (showMoreoprDisplayName != "" || showMoreoprDisplayName != undefined){
  1716. var filename = showMoreoprDisplayName;
  1717. searchOnYoutube(filename);
  1718. }
  1719. }
  1720. var showMoreOprPlayID,showMoreOprFilepath, showMoreoprDisplayName;
  1721. function showMore(object){
  1722. var filepath = $(object).parent().attr("filepath");
  1723. filepath = JSON.parse(decodeURIComponent(filepath));
  1724. //User want more action on this file. assume this is the file to add
  1725. playlistAddPendingFile = filepath;
  1726. var id = $(object).parent().attr('id');
  1727. var rawname = $(object).parent().attr("rawname");
  1728. var displayName = JSON.parse(decodeURIComponent(ao_module_codec.decodeUmFilename(rawname)));
  1729. //Update global variable for quick operations
  1730. showMoreOprPlayID = id;
  1731. showMoreOprFilepath = filepath;
  1732. showMoreoprDisplayName = displayName;
  1733. $("#showMoreUI").find(".songTitle").text(displayName);
  1734. $("#showMoreUI").find(".songID").text(id);
  1735. $("#showFileInfo").find(".songTitle").text(displayName);
  1736. $("#showMoreUI").fadeIn('fast');
  1737. $("#showmoreUIcover").fadeIn('fast');
  1738. //Update fileinformation as well
  1739. if (filepath.substring(0, 12) == "/media?file="){
  1740. filepath = filepath.substring(12);
  1741. }
  1742. //Pre-render the file info
  1743. ao_module_agirun("Music/functions/getFileInfo.js", {
  1744. filepath: filepath,
  1745. }, function(data){
  1746. $("#showFileInfo").find(".filename").text(ao_module_codec.decodeUmFilename(data[0]));
  1747. $("#showFileInfo").find(".rawname").text(data[0]);
  1748. $("#showFileInfo").find(".filepath").text(data[1]);
  1749. $("#showFileInfo").find(".filesize").text(data[2] + " (" + data[3] + " Bytes)");
  1750. $("#showFileInfo").find(".filedate").text(data[4]);
  1751. });
  1752. if (currentMode != "playlist"){
  1753. $(".playlistonly").hide();
  1754. }else{
  1755. $(".playlistonly").show();
  1756. }
  1757. }
  1758. function playSongFromDropdownList(object){
  1759. //Play song from dropdown list. Hence, no need to update dropdownList
  1760. if ($(object).parent().hasClass("playingTrack")){
  1761. //This song already playing. Ignore play request.
  1762. return;
  1763. }
  1764. playSong(object);
  1765. //Check if the song is also in main list. If yes, highlight it as well
  1766. /*
  1767. $(".mainList.item").each(function(){
  1768. if ($(this).attr("filepath") == $(object).parent().attr("filepath")){
  1769. $(".mainList.item.playingTrack").removeClass("playingTrack");
  1770. $(this).addClass("playingTrack");
  1771. }
  1772. });
  1773. */
  1774. $("#dropdownSonglist").delay(500).slideUp();
  1775. highLightPlayingMusic();
  1776. }
  1777. //Move the playing song list into the dropdown song list. This list might not be the same as the one in the main list.
  1778. function parsePlayingSongList(){
  1779. var template = '<div class="dropdownList item" filepath={filepath} listid={id} rawname={rawname}>\
  1780. <div class="ts header selectable" style="margin:0px !important;" onClick="playSongFromDropdownList(this);">\
  1781. <i class="music icon whiteFont" style="overflow:hidden;"></i>\
  1782. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  1783. {songtitle}\
  1784. <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
  1785. </div>\
  1786. </div>\
  1787. <div class="topRightCorner" align="center">\
  1788. {id}\
  1789. </div>\
  1790. </div>';
  1791. $("#currentPlayingMainList").html("");
  1792. var counter = playingList.length;
  1793. var totalSize = 0.0;
  1794. for (var i = 0; i < playingList.length; i++){
  1795. var displayname = ao_module_codec.decodeUmFilename(playingList[i][1]);
  1796. var box = template;
  1797. box = replaceAll("{filepath}",ao_module_utils.objectToAttr(playingList[i][0]),box);
  1798. box = replaceAll("{id}",i + 1,box); //User count from 1
  1799. box = replaceAll("{rawname}",ao_module_utils.objectToAttr(playingList[i][1]),box);
  1800. box = replaceAll("{songtitle}",displayname,box);
  1801. box = replaceAll("{ext}",playingList[i][2],box);
  1802. box = replaceAll("{size}",playingList[i][3],box);
  1803. if (playingList[i][3].includes("MB")){
  1804. totalSize += parseFloat(playingList[i][3].split(" ")[0]);
  1805. }else if (playingList[i][3].includes("KB")){
  1806. totalSize += parseFloat(playingList[i][3].split(" ")[0]) / 1000;
  1807. }else if (playingList[i][3].includes("GB")){
  1808. totalSize += parseFloat(playingList[i][3].split(" ")[0]) * 1000;
  1809. }
  1810. $("#currentPlayingMainList").append(box);
  1811. $("#dropdownListSongCount").text(counter);
  1812. $("#dropdownListIDCount").text(parseInt(totalSize) + "");
  1813. }
  1814. $(".dropdownList.item").each(function(){
  1815. if ($(this).attr('filepath') == playingFileDetail[1]){
  1816. $(this).addClass("playingTrack");
  1817. }
  1818. });
  1819. }
  1820. function loadSongList(type = "all"){
  1821. currentPath = "";
  1822. currentMode = "music";
  1823. var template = '<div class="mainList item" filepath={filepath} id={id} rawname={rawname}>\
  1824. <div class="ts header selectable" style="margin:0px !important;" onClick="playSong(this);">\
  1825. <i class="music icon whiteFont" style="overflow:hidden;"></i>\
  1826. <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
  1827. {songtitle}\
  1828. <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
  1829. </div>\
  1830. </div>\
  1831. <div class="topRightCorner" align="center">\
  1832. {id}\
  1833. </div>\
  1834. <div class="mainList rightFunctionBar" align="center" onClick="showMore(this);">\
  1835. <i class="ellipsis vertical icon" style="top:30%;"></i>\
  1836. </div>\
  1837. </div>';
  1838. $("#interfaceTitle").text("Music");
  1839. $("#AMmenuIcon").attr("class","music icon large whiteFont")
  1840. ao_module_agirun("Music/functions/listSong.js", {
  1841. listSong: type,
  1842. },function(data){
  1843. //Initialize the song list
  1844. displayList = data;
  1845. if (type == "all"){
  1846. //Caching is used in here
  1847. displayList = data.list;
  1848. if (data.cached == true){
  1849. //This is a cached version of the music list.
  1850. console.log("Updating cached song list");
  1851. ao_module_agirun("Music/functions/buildCache.js", {}, function(data){
  1852. console.log("Cache updated: ", data);
  1853. });
  1854. }
  1855. }
  1856. console.log(displayList);
  1857. if (playingList == []){
  1858. playingList = Array.from(displayList);
  1859. }
  1860. $("#mainList").html("");
  1861. for ( var i =0; i < displayList.length; i++){
  1862. var songInfo = displayList[i];
  1863. var path = songInfo[0];
  1864. var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
  1865. var ext = songInfo[2];
  1866. var size = songInfo[3];
  1867. var box = template;
  1868. box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
  1869. box = replaceAll("{id}",i + 1,box); //User count from 1
  1870. box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
  1871. box = replaceAll("{songtitle}",displayname,box);
  1872. box = replaceAll("{ext}",ext,box);
  1873. box = replaceAll("{size}",size,box);
  1874. $("#mainList").append(box);
  1875. }
  1876. totalMusicCount = displayList.length;
  1877. $("#mainList").append("<br><br><br><br><br><br><br>");
  1878. $("#interfaceDetails").text("[" + i + " songs]");
  1879. highLightPlayingMusic();
  1880. }, function(){
  1881. alert("Failed to access listSong API with type: " + type)
  1882. });
  1883. hideLeftMenu();
  1884. }
  1885. function toggleLeftMenu(){
  1886. if (leftMenuShown){
  1887. $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 300,function(){
  1888. $("#leftSideBar").hide();
  1889. });
  1890. $("#sideBarCover").fadeOut();
  1891. leftMenuShown = false;
  1892. }else{
  1893. $("#leftSideBar").show();
  1894. $("#leftSideBar").animate({left:0}, 300);
  1895. $("#sideBarCover").fadeIn();
  1896. leftMenuShown = true;
  1897. }
  1898. }
  1899. function hideLeftMenu(){
  1900. if (leftMenuShown){
  1901. $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 300,function(){
  1902. $("#leftSideBar").hide();
  1903. });
  1904. $("#sideBarCover").fadeOut();
  1905. leftMenuShown = false;
  1906. }
  1907. }
  1908. function showFileInformation(){
  1909. ao_module_agirun("Music/functions/getFileInfo.js", {
  1910. filepath: playingFileDetail[1],
  1911. },function(data){
  1912. //console.log(data);
  1913. $("#settingInterface").hide();
  1914. $("#filepropInterface").show();
  1915. $("#filepropInterface").find(".filename").text(ao_module_codec.decodeUmFilename(data[0]));
  1916. $("#filepropInterface").find(".rawname").text(data[0]);
  1917. $("#filepropInterface").find(".filepath").text(data[1]);
  1918. $("#filepropInterface").find(".filesize").text(data[2] + " (" + data[3] + " Bytes)");
  1919. $("#filepropInterface").find(".filedate").text(data[4]);
  1920. });
  1921. }
  1922. function resizeQuickAdjust(){
  1923. //Resize the position of the leftMenu
  1924. if (!leftMenuShown){
  1925. $("#leftSideBar").css("left", $("#leftSideBar").width() * -1);
  1926. }
  1927. if ($("#playerInterface").offset().left != 0){
  1928. $("#playerInterface").css("left",window.innerWidth);
  1929. }
  1930. $("#albumnArt").css({
  1931. "height": window.innerHeight - 255,
  1932. "top": (window.innerHeight / 2 - $("#albumnArtImage").height()/2)
  1933. });
  1934. $("#albumnArtImage").css("max-height",window.innerHeight - 255);
  1935. //var imageTop = (window.innerHeight - 255 - $("#albumnArtImage").height()) / 2;
  1936. //$("#albumnArtImage").css("top",imageTop + "px");
  1937. //console.log(window.innerHeight,imageTop);
  1938. }
  1939. function setStorage(configName,configValue){
  1940. ao_module_storage.setStorage("AirMusic",configName,configValue);
  1941. /*
  1942. $.ajax({
  1943. type: 'GET',
  1944. url: "../system/file_system/preference",
  1945. data: {key: "AirMusic/" + configName,value:configValue},
  1946. success: function(data){},
  1947. async:true
  1948. });
  1949. */
  1950. return true;
  1951. }
  1952. function loadStorage(configName){
  1953. /*
  1954. var result = "";
  1955. $.ajax({
  1956. type: 'GET',
  1957. url: "../system/file_system/preference",
  1958. data: {key: "AirMusic/" + configName},
  1959. success: function(data){
  1960. if (data.error !== undefined){
  1961. result = "";
  1962. }else{
  1963. result = data;
  1964. }
  1965. },
  1966. error: function(data){result = "";},
  1967. async:false,
  1968. timeout: 3000
  1969. });
  1970. return result;
  1971. */
  1972. return ao_module_storage.loadStorage("AirMusic",configName);
  1973. }
  1974. //Wipe controller for mobile users
  1975. $("#albumnArt")[0].addEventListener('touchstart', handleTouchStart, false);
  1976. $("#albumnArt")[0].addEventListener('touchmove', handleTouchMove, false);
  1977. var xDown = null;
  1978. var yDown = null;
  1979. function getTouches(evt) {
  1980. return evt.touches || // browser API
  1981. evt.originalEvent.touches; // jQuery
  1982. }
  1983. function handleTouchStart(evt) {
  1984. const firstTouch = getTouches(evt)[0];
  1985. xDown = firstTouch.clientX;
  1986. yDown = firstTouch.clientY;
  1987. };
  1988. function handleTouchMove(evt) {
  1989. if ( ! xDown || ! yDown ) {
  1990. return;
  1991. }
  1992. var xUp = evt.touches[0].clientX;
  1993. var yUp = evt.touches[0].clientY;
  1994. var xDiff = xDown - xUp;
  1995. var yDiff = yDown - yUp;
  1996. if ( Math.abs( xDiff ) > Math.abs( yDiff ) ) {/*most significant*/
  1997. if ( xDiff > 0 ) {
  1998. /* left swipe */
  1999. //Going to the next song
  2000. nextSong();
  2001. } else {
  2002. /* right swipe */
  2003. //Going back one song
  2004. previousSong();
  2005. }
  2006. } else {
  2007. if ( yDiff > 0 ) {
  2008. /* up swipe */
  2009. } else {
  2010. /* down swipe */
  2011. }
  2012. }
  2013. /* reset values */
  2014. xDown = null;
  2015. yDown = null;
  2016. };
  2017. //Handle audio progress to time conversion
  2018. function secondsToHms(d) {
  2019. d = Number(d);
  2020. var h = Math.floor(d / 3600);
  2021. var m = Math.floor(d % 3600 / 60);
  2022. var s = Math.floor(d % 3600 % 60);
  2023. if (h > 0 && h < 10){
  2024. dh = "0" + h + ":";
  2025. }else if (h == 0){
  2026. dh = "";
  2027. }else{
  2028. dh = h + ":";
  2029. }
  2030. if (m > 0 && m < 10){
  2031. dm = "0" + m + ":";
  2032. }else if (m == 0){
  2033. if (h > 0){
  2034. dm = "00:";
  2035. }else{
  2036. dm = "0:";
  2037. }
  2038. }else{
  2039. dm = m + ":";
  2040. }
  2041. if (s > 0 && s < 10){
  2042. ds = "0" + s;
  2043. }else if (s == 0){
  2044. if (m > 0 || h > 0){
  2045. ds = "00";
  2046. }else{
  2047. ds = "00";
  2048. }
  2049. }else{
  2050. ds = s;
  2051. }
  2052. return dh + dm + ds;
  2053. }
  2054. //Handle window resize events
  2055. $( window ).resize(function() {
  2056. resizeQuickAdjust();
  2057. });
  2058. function AllQuickMenuHidden(){
  2059. var result = true;
  2060. $(".quickMenu").each(function(){
  2061. if ( $(this).is(':visible') ){
  2062. result = false;
  2063. }
  2064. });
  2065. return result;
  2066. }
  2067. function AllSubMenuHidden(){
  2068. var result = true;
  2069. $(".showMoreMenus").each(function(){
  2070. if ( $(this).is(':visible') ){
  2071. result = false;
  2072. }
  2073. });
  2074. return result;
  2075. }
  2076. //Handle back button press on mobile devices
  2077. function handleBackButton() {
  2078. if (!AllQuickMenuHidden()){
  2079. hideAllquickMenu();
  2080. window.history.pushState({}, '');
  2081. }else if (mainPlayerShown()){
  2082. //Close the main Menu
  2083. hideMainPlayerInterface();
  2084. window.history.pushState({}, '');
  2085. }else if (!AllSubMenuHidden()){
  2086. $(".showMoreMenus").fadeOut('fast');
  2087. $("#showmoreUIcover").fadeOut('fast');
  2088. window.history.pushState({}, '');
  2089. }else{
  2090. window.history.back();
  2091. }
  2092. }
  2093. window.addEventListener('popstate', handleBackButton);
  2094. window.history.pushState({}, '');
  2095. </script>
  2096. </body>
  2097. </html>