|
@@ -1,2625 +1,2731 @@
|
|
|
-<!DOCTYPE html>
|
|
|
-<html>
|
|
|
-<head>
|
|
|
- <meta charset="UTF-8">
|
|
|
- <meta name="apple-mobile-web-app-capable" content="yes" />
|
|
|
- <meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1, maximum-scale=1"/>
|
|
|
- <meta name="theme-color" content="#4b75ff">
|
|
|
- <link rel="stylesheet" href="../script/semantic/semantic.min.css">
|
|
|
- <link rel="stylesheet" href="./main.css">
|
|
|
- <script src="../script/jquery.min.js"></script>
|
|
|
- <script src="../script/ao_module.js"></script>
|
|
|
- <script src="../script/semantic/semantic.min.js"></script>
|
|
|
-
|
|
|
- <!-- Handle native playback on Chrome Andoird-->
|
|
|
- <script src="native.js"></script>
|
|
|
-
|
|
|
- <link rel="manifest" crossorigin="use-credentials" href="manifest.json">
|
|
|
- <title>AirMusic</title>
|
|
|
-</head>
|
|
|
-<body>
|
|
|
-<div id="mainMenu" class="ui basic fluid AMmenu menu bottomBlue" style="z-index:80;position:fixed;position:top:0px;left:0px;width:100%;">
|
|
|
- <button class="ui item icon noBorder AMmenu button" onClick="toggleLeftMenu();"><i class="content icon"></i></button>
|
|
|
- <div class="item noBorder AMmenu" style="padding-right:0px;padding-top:12px;"><i id="AMmenuIcon" class="music icon large whiteFont"></i></div>
|
|
|
- <div class="item noBorder AMmenu" style="padding-left:3px;padding-top:3px;padding-bottom:5px;min-width:50%;">
|
|
|
- <div class="ui header whiteFont">
|
|
|
- <div id="interfaceTitle" style="display:inline; font-weight: lighter;">Music</div>
|
|
|
- <div id="interfaceDetails" class="sub header" style="font-size:80%; line-height:1em; color: white;">[0 songs]</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <button class="ui right item icon AMmenu noBorder button" onClick="enterSearchMode();"><i class="search icon"></i></button>
|
|
|
- <!-- <button class="ui right item icon AMmenu noBorder button"><i class="ellipsis vertical icon"></i></button> -->
|
|
|
-</div>
|
|
|
-<div id="searchModeMenu" class="ui basic fluid AMmenu menu bottomBlue" style="z-index:85;position:fixed;top:0px;left:0px;width:100%;height:51px;display:none;margin-top: 0px;">
|
|
|
- <button class="ui item icon noBorder AMmenu button" onClick="exitSearchMode();"><i class="arrow left icon"></i></button>
|
|
|
- <div class="ui input" style="flex-grow: 1;">
|
|
|
- <input id="searchInputBar" onkeypress="//handleSearchInputEnter(event);" 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">
|
|
|
- </div>
|
|
|
- <button class="ui right item icon AMmenu noBorder button" onClick="searchSong();"><i class="search icon"></i></button>
|
|
|
-</div>
|
|
|
-<!-- The main list that display the current information-->
|
|
|
-<div id="mainList">
|
|
|
-
|
|
|
-</div>
|
|
|
-
|
|
|
-<!-- Advance functions config menu-->
|
|
|
-<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>
|
|
|
-<div id="showMoreUI" class="showMoreMenus" style="display:none;">
|
|
|
- <div class="ui large header whiteFont songTitle">N/A</div>
|
|
|
- <div class="showMoreMenuItem" onClick="playFromShowMoreMenu();">Play</div>
|
|
|
- <div class="showMoreMenuItem" onclick='addToPlaylistFromMoreMenu();'>Add to Playlist</div>
|
|
|
- <div class="showMoreMenuItem playlistonly" onclick="removeFromPlylistFromMoreMenu();">Remove from current Playlist</div>
|
|
|
- <div class="showMoreMenuItem" onClick="searchYoutubeViaShowMore();">Search on Youtube</div>
|
|
|
- <div class="showMoreMenuItem">Edit Tag</div>
|
|
|
- <div class="showMoreMenuItem">Share</div>
|
|
|
- <div class="showMoreMenuItem" onClick="startRelatedSearch();">Related Search</div>
|
|
|
- <div class="showMoreMenuItem" onClick="showFileInfo();">File Information</div>
|
|
|
-
|
|
|
- <div class="topRightCorner songID" align="center" style="margin-top:10px !important;padding-top:3px !important;">-1</div>
|
|
|
-</div>
|
|
|
-<div id="showFileInfo" class="showMoreMenus" style="display:none;">
|
|
|
- <div class="ui large header">
|
|
|
- <span class="filename whiteFont">Title</span>
|
|
|
- <div class="sub header rawname whiteFont" style="word-break: break-all !important; font-size:80%;">Storage Name</div>
|
|
|
- </div>
|
|
|
- <div class="ui list whiteFont">
|
|
|
- <div class="item whiteFont">Storage Location: <span class="filepath" style="word-break:break-all;">N/A</span></div>
|
|
|
- <div class="item whiteFont">File Size: <span class="filesize">N/A</span></div>
|
|
|
- <div class="item whiteFont">Last Modification Date: <span class="filedate">N/A</span></div>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-
|
|
|
-<!-- Side bar -->
|
|
|
-<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>
|
|
|
-<div id="leftSideBar" class="leftsb" style="max-width:60%;height:100%;z-index:90;display:none;">
|
|
|
- <div class="leftsbObject sidebarBanner">
|
|
|
- <img class="ui tiny middle aligned image" src="img/main_icon.png" style="max-width: 50px;"><h5 class="whiteFont" style="display:inline;padding-left:8px;">AirMusic</h5>
|
|
|
- </div>
|
|
|
- <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadSongList();setStorage('viewingTab','music');">
|
|
|
- <div class="ui header">
|
|
|
- <i class="music tiny icon whiteFont"></i>
|
|
|
- <div class="content whiteFont">
|
|
|
- Music
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadFolderView();setStorage('viewingTab','folder');">
|
|
|
- <div class="ui header">
|
|
|
- <i class="folder tiny icon whiteFont"></i>
|
|
|
- <div class="content whiteFont">
|
|
|
- Folder
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadPlaylistView(); setStorage('viewingTab','playlist');">
|
|
|
- <div class="ui header">
|
|
|
- <i class="align justify tiny icon whiteFont"></i>
|
|
|
- <div class="content whiteFont">
|
|
|
- Playlists
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadNetworkView(); setStorage('viewingTab','network');">
|
|
|
- <div class="ui header">
|
|
|
- <i class="world tiny icon whiteFont"></i>
|
|
|
- <div class="content whiteFont">
|
|
|
- Network
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <!--
|
|
|
- <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="">
|
|
|
- <div class="ui header">
|
|
|
- <i class="upload icon whiteFont"></i>
|
|
|
- <div class="content whiteFont">
|
|
|
- Upload
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- -->
|
|
|
- <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;border-top: 2px solid #4a4a4a;">
|
|
|
- <div class="ui header">
|
|
|
- <i class="setting tiny icon whiteFont"></i>
|
|
|
- <div class="content whiteFont">
|
|
|
- Settings
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-
|
|
|
-<!-- Mini player controller put on the bottom side of the list interface-->
|
|
|
-<div id="miniPlayer" class="hidden">
|
|
|
- <div style="padding-left:20px;padding-top:5px;" >
|
|
|
- <div class="ui header selectable" style="margin:0px !important;width:300px;cursor:pointer;" onClick="showMainPlayerInterface();">
|
|
|
- <img id="smallPlayerThumb" class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;">
|
|
|
- <div id="miniPlayerDisplayName" style="text-overflow: ellipsis;overflow: hidden;max-width:200px; white-space: nowrap;">N/A</div>
|
|
|
- <div id="miniPlayerInformation" class="sub header" style="color: #c7c7c7;size:95%;">No information</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div id="miniPlayerIDtab" class="miniPlayer minitab">
|
|
|
- 0/0
|
|
|
- </div>
|
|
|
- <div class="miniPlayerControls">
|
|
|
- <button class="miniPlayer selectable" style="font-size:90%; font-weight: lighter;" onClick="previousSong();"><i class="step backward icon whiteFont"></i></button>
|
|
|
- <button class="miniPlayer selectable" style="font-size:120%; font-weight: lighter;" onClick="togglePlayMode();"><i class="play icon whiteFont PlayButton"></i></button>
|
|
|
- <button class="miniPlayer selectable" style="font-size:90%; font-weight: lighter;" onClick="nextSong();"><i class="step forward icon whiteFont "></i></button>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-<!-- Main interface for player and audio controller-->
|
|
|
-<div id="playerInterface" style="display:none;">
|
|
|
- <div id="infoBar" class="whiteFont" style="padding:20px;" align="center">
|
|
|
- <!-- Top Information Bar-->
|
|
|
-
|
|
|
- <div class="whiteFont songTitleWrapper" style="margin:0px;display:inline-block;">
|
|
|
- <div id="mainPlayerSongTitle" class="whiteFont" style="font-weight: bold;font-size:120%;">Song Title</div>
|
|
|
- <small id="mainPlayerSongDesc">This is some small text for display</small>
|
|
|
- </div>
|
|
|
- <div style="position:absolute;left:20px;top:30px;cursor:pointer;" onClick="hideMainPlayerInterface();">
|
|
|
- <i class="large chevron left icon"></i>
|
|
|
- </div>
|
|
|
- <div style="position:absolute;right:20px;top:30px;cursor:pointer;" onClick='$("#dropdownSonglist").slideDown();'>
|
|
|
- <i class="large content icon"></i>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div style="padding-left:20px; padding-right:20px;padding-bottom:5px; position: relative;" align="center">
|
|
|
- <!-- Function Menu Bar-->
|
|
|
- <div class="ui grid">
|
|
|
- <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onclick="showPlaylistInterface();"><i class="large plus icon whiteFont"></i></button></div>
|
|
|
- <div class="two wide column"><button class="topPanelButtons" id="downloadBtn" style="cursor: pointer;" onClick="downloadPlayingSong();"><i class="large download icon whiteFont"></i></button></div>
|
|
|
- <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onClick="openInFileExplorer();"><i class="large folder open icon whiteFont"></i></button></div>
|
|
|
- <div class="two wide column desktopOnly" style="margin-top: -4px;"><button class="topPanelButtons" style="cursor: pointer; opacity: 0.6;" onClick="mute(this);" ><i class="icons"><i class="big red dont icon"></i><i style="margin-left: -3px;" class="large volume off icon whiteFont"></i></i></button></div>
|
|
|
- <div class="two wide column desktopOnly"><button class="topPanelButtons" id="voldownBtn" style="cursor: pointer;" onClick="addAudioVolume(-0.05);"><i class="large volume down icon whiteFont"></i></button></div>
|
|
|
- <div class="two wide column desktopOnly"><button class="topPanelButtons" id="volupBtn" style="cursor: pointer;" onClick="addAudioVolume(0.05);"><i class="large volume up icon whiteFont"></i></button></div>
|
|
|
- <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onClick="showTimerInterface();"><i class="large clock icon whiteFont"></i></button></div>
|
|
|
- <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onClick="showSettingInterface();"><i class="large setting icon whiteFont"></i></button></div>
|
|
|
- </div>
|
|
|
- <div id="volumeGUI" align="center">
|
|
|
- <i class="volume off large icon whiteFont" style="position:absolute;left:2em;top:1em;"></i>
|
|
|
- <div class="ui tiny progress" style=" margin-top: calc(2em - 5px); margin-bottom: 0; background-color: rgba(255,255,255,0.2);" align="left">
|
|
|
- <div id="volBar" class="bar" style="min-width: 0px; width: 40%; height: 10px; background-color: #4b75ff;"></div>
|
|
|
- </div>
|
|
|
- <i class="volume up large icon whiteFont" style="position:absolute;right:2em;top:1em;"></i>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <!-- Download progress bar-->
|
|
|
- <div id="downloadProgressBar" class="ui attached fluid tiny progress" style="background-color:rgba(20, 20, 20, 0.9);display:none;">
|
|
|
- <div class="bar" style="min-width: 0px; width: 0%;background-color:rgb(75, 117, 255) !important;"></div>
|
|
|
- </div>
|
|
|
-
|
|
|
- <div id="albumnArt" align="center">
|
|
|
- <img id="albumnArtImage" style="max-width: 80%; pointer-events: none; user-select: none;" src="img/default.png">
|
|
|
- </div>
|
|
|
- <div id="mainPlayerControlInterface">
|
|
|
- <!-- main control interface-->
|
|
|
- <div style="width:100%;" align="center">
|
|
|
- <div id="progressTime">
|
|
|
- 0:00
|
|
|
- </div>
|
|
|
- <div id="mainPlayerMiniTab" class="mainPlayer minitab">
|
|
|
- 0/0
|
|
|
- </div>
|
|
|
- <div id="remainTime">
|
|
|
- -0:00
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-
|
|
|
- <div style="position:absolute;width:100%;left:0px;bottom:0px;">
|
|
|
- <div style="padding-left:10px;padding-right:10px;">
|
|
|
- <div class="ui small primary progress" style="background-color: rgba(255,255,255,0.4);">
|
|
|
- <div id="audioProgressBar" class="bar" style="min-width: 0px; width: 100%; background: #4b75ff;"></div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="mainControlButtons">
|
|
|
- <button class="panelButtons" style="font-size:90%;padding:15px;" onClick="previousSong();"><i class="step backward icon whiteFont"></i></button>
|
|
|
- <button class="panelButtons" style="font-size:130%;" onClick="togglePlayMode();"><i class="play icon whiteFont PlayButton"></i></button>
|
|
|
- <button class="panelButtons" style="font-size:90%;padding:15px;" onClick="nextSong();"><i class="step forward icon whiteFont "></i></button>
|
|
|
- </div>
|
|
|
- <div id="randomModeButton" style="position: absolute; left:30px;bottom:20px;" onClick="toggleRandomMode(this);">
|
|
|
- <i class="large random icon disabled"></i>
|
|
|
- </div>
|
|
|
- <div id="repeatModeButton" style="position: absolute; right:30px;bottom:20px;" onClick="toggleRepeatMode(this);">
|
|
|
- <i class="large retweet icon disabled"></i>
|
|
|
- <p class="whiteFont singleLoop" style="position:absolute;right:0px;bottom:0px;margin-bottom:-13px;margin-right:-2px;display:none;">1</p>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <audio id="mainAudioPlayer" style="display:none;" src=""></audio>
|
|
|
-
|
|
|
- <div id="dropdownSonglist" class="dropdownMusicList" style="display:none;">
|
|
|
- <div class="dropdownMusicListMiniMenu">
|
|
|
- <i class="icons" style="margin-right:8px;">
|
|
|
- <i class="content icon"></i>
|
|
|
- <i class="corner music icon" style='color:#333333'></i>
|
|
|
- </i>
|
|
|
- Now Playing
|
|
|
- <div style="position:absolute;top:3px;right:5px;">
|
|
|
- [ <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>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div id="currentPlayingMainList" style="overflow-x:hidden;">
|
|
|
-
|
|
|
- </div>
|
|
|
- <div class="dropdownMusicListBottom" align="center" onClick='$("#dropdownSonglist").slideUp();'><i class="caret up icon"></i></div>
|
|
|
- </div>
|
|
|
-
|
|
|
-</div>
|
|
|
-
|
|
|
-<!-- Quick Menus and other tool windows-->
|
|
|
-
|
|
|
- <!-- Timer control interface-->
|
|
|
- <div id="timerInterface" class="quickMenu">
|
|
|
- <h4 class="whiteFont">Countdown Timer</h4>
|
|
|
- <br>
|
|
|
- <div id="timerSettingInterface" style="width:100%;" align="center">
|
|
|
- <div>
|
|
|
- <div class="ui statistic">
|
|
|
- <div class="label whiteFont">hours</div>
|
|
|
- <div id="timerHour" class="value whiteFont">0</div>
|
|
|
- </div>
|
|
|
- <div class="ui statistic">
|
|
|
- <div class="label whiteFont">minutes</div>
|
|
|
- <div id="timerMinute" class="value whiteFont">0</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div>
|
|
|
- <div class="ui icon mini buttons">
|
|
|
- <button class="ui secondary button" onClick="addTimer('h',1);"><i class="add icon"></i></button>
|
|
|
- <button class="ui secondary button" onClick="addTimer('h','r');"><i class="refresh icon"></i></button>
|
|
|
- <button class="ui secondary button" onClick="addTimer('h',-1);"><i class="minus icon"></i></button>
|
|
|
- </div>
|
|
|
- <div class="ui icon mini buttons">
|
|
|
- <button class="ui secondary button" onClick="addTimer('m',10);"><i class="add icon"></i></button>
|
|
|
- <button class="ui secondary button" onClick="addTimer('m','r');"><i class="refresh icon"></i></button>
|
|
|
- <button class="ui secondary button" onClick="addTimer('m',-1);"><i class="minus icon"></i></button>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div id="timerCountingInterface" style="width:100%; display:none;" align="center">
|
|
|
- <div class="ui statistic">
|
|
|
- <div id="remainingUI" class="value whiteFont">0:00:00</div>
|
|
|
- <div class="label whiteFont">Remaining</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="ui divider"></div>
|
|
|
- <div>
|
|
|
- <div class="ui grid">
|
|
|
- <div class="eight wide column">
|
|
|
- Times up action
|
|
|
- </div>
|
|
|
- <div class="eight wide column">
|
|
|
- <select id="stopMode" class="ui selection mini fluid dropdown">
|
|
|
- <option>Fade Out & Stop</option>
|
|
|
- <option>Stop Playing</option>
|
|
|
- </select>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div style="width:100%;position:absolute;left:0px;bottom:20px;" align="right">
|
|
|
- <div class="ui grid" >
|
|
|
- <div class="eight wide column" style="padding-right: 0px;" align="center"><button class="greenBtn fluid" onClick='hideAllquickMenu();'>Close</button></div>
|
|
|
- <div class="eight wide column" style="padding-left: 0px;" align="center"><button class="greenBtn fluid" onClick="toggleCountDown(this);">Start</button></div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-
|
|
|
-<!-- File properties interface-->
|
|
|
-<div id="filepropInterface" class="quickMenu">
|
|
|
- <div class="ui header">
|
|
|
- <span class="filename whiteFont">Title</span>
|
|
|
- <div class="sub header rawname whiteFont" style="word-break:break-all;">Storage Name</div>
|
|
|
- </div>
|
|
|
- <div class="ui list whiteFont">
|
|
|
- <div class="item whiteFont">Storage Location: <span class="filepath" style="word-break:break-all;">N/A</span></div>
|
|
|
- <div class="item whiteFont">File Size: <span class="filesize">N/A</span></div>
|
|
|
- <div class="item whiteFont">Last Modification Date: <span class="filedate">N/A</span></div>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-<!-- Setting interface-->
|
|
|
-<div id="settingInterface" class="quickMenu">
|
|
|
- <div style="">
|
|
|
- <div class="ui relaxed list" style="padding:0px !important;">
|
|
|
- <div id="settingMenuTitle" class="item whiteFont" style="">N/A</div>
|
|
|
- <div class="item selectable whiteFont" onclick="showPlaylistInterface(); $('#settingInterface').fadeOut('fast');"><i class="add icon" style="margin-right:5px;"></i> Add to playlist</div>
|
|
|
- <div class="item selectable whiteFont" onClick="openInFileExplorer(); hideAllquickMenu();"><i class="folder open icon" style="margin-right:5px;"></i> Open in File Explorer</div>
|
|
|
- <div class="item selectable whiteFont" onClick="searchOnYoutube(); hideAllquickMenu();"><i class="youtube play icon" style="margin-right:5px;"></i>Search on Youtube</div>
|
|
|
- <div class="item selectable whiteFont" onClick="showFileInformation();"><i class="file icon" style="margin-right:5px;"></i> File Information</div>
|
|
|
- <div class="ui divider"></div>
|
|
|
- <div id="closeWebAppButton" class="item selectable whiteFont" onClick="ao_module_close();"><i class="remove icon" style="margin-right:5px;"></i> Exit Application</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-
|
|
|
-
|
|
|
-<!-- Playlist interface-->
|
|
|
-<div id="playlistInterface" class="quickMenu">
|
|
|
- <div class="ui header">
|
|
|
- <span class="whiteFont">Add to Playlist</span>
|
|
|
- </div>
|
|
|
- <div class="playlist item selectable whiteFont" onclick="addToNewPlaylist();">
|
|
|
- <i class="add icon whiteFont"></i> New Playlist
|
|
|
- </div>
|
|
|
- <!-- playlist list-->
|
|
|
- <div id="existsingPlaylist" style="overflow-y: scroll; height: 50%; border: 1px solid white;">
|
|
|
- <div class="playlist item selectable whiteFont">
|
|
|
- <i class="list icon whiteFont"></i> Test Playlist
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <br>
|
|
|
- <div style="width:100%;position:absolute;left:0px;bottom:0px; margin-bottom: 10px;" align="center">
|
|
|
- <button class="greenBtn" onClick='hideAllquickMenu();'>Close</button>
|
|
|
- </div>
|
|
|
- <div class="ui snackbar" id="succSnackBar">
|
|
|
- <div class="content">
|
|
|
- Action Completed
|
|
|
- </div>
|
|
|
- </div>
|
|
|
-</div>
|
|
|
-
|
|
|
-<!-- Utils Interfaces, usually hidden-->
|
|
|
-<div id="fadeReturnScreen" class="fadeScreen whiteFont" onClick="hideAllquickMenu();"></div>
|
|
|
-
|
|
|
-<script>
|
|
|
- var leftMenuShown = false;
|
|
|
- var currentPath = "";
|
|
|
- var currentMode = "music";
|
|
|
- var rootPaths = [];
|
|
|
- var totalMusicCount = 0;
|
|
|
- var currentPlaying = false;
|
|
|
- var audioElement = $("#mainAudioPlayer");
|
|
|
- var audioElementObject = document.getElementById("mainAudioPlayer"); //Directly expose the audio object
|
|
|
- var playingFileDetail = []; //[id, filepath]
|
|
|
- var displayList = []; //This is the list where the current UI is displaying.
|
|
|
- var playingList = []; //This is the list where the player last clicked on an item to play
|
|
|
- var pagingEnabled = false; //Enable this when there are too many songs in list
|
|
|
- var pagingEnableForPlayingList = false;
|
|
|
- var pagingCutoffCount = 100; //No. of songs to start paging
|
|
|
- var currentPage = 0;
|
|
|
- var currentPlayingPage = 0;
|
|
|
- var randomMode = false;
|
|
|
- var repeatMode = 0; //Repeat mode 0 -> No repeat, 1 -> Repeat all 2 -> Repeat one
|
|
|
- var updateSystemVolume = true;
|
|
|
- var timerMode = false;
|
|
|
- var timerRemaining = 0;
|
|
|
- var playlistAddPendingFile = ""; //The filepath of the file that is pending to be added to playlist
|
|
|
- var timerEndMode = "fade"; //Fade or End, Provide volume fadeout or end immediately while times up
|
|
|
- var ua = navigator.userAgent.toLowerCase();
|
|
|
- var isAndroid = ua.indexOf("android") > -1; //&& ua.indexOf("mobile");
|
|
|
- var isChrome = /Chrome/.test(navigator.userAgent) && /Google Inc/.test(navigator.vendor);
|
|
|
-
|
|
|
- //Embed native mediaSession for any device that supports it
|
|
|
- initNativeMediaPlayer();
|
|
|
-
|
|
|
- //Update implementatio nof user agent detection
|
|
|
- if (typeof InstallTrigger !== 'undefined'){
|
|
|
- ua = "firefox";
|
|
|
- }
|
|
|
-
|
|
|
- if (!!window.chrome && (!!window.chrome.webstore || !!window.chrome.runtime)){
|
|
|
- ua = "chrome";
|
|
|
- }
|
|
|
-
|
|
|
- var isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
|
|
|
- if (isSafari == true){
|
|
|
- ua = "safari";
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- $(document).ready(function(){
|
|
|
- $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 300);
|
|
|
- //Update 7-8-2019, default loading tab is handled over to the returnToPreviousState() function.
|
|
|
- //loadSongList();
|
|
|
- //Initiate the module's volume to the system's
|
|
|
- syncSystemVol();
|
|
|
- initAudioListeners();
|
|
|
- resizeQuickAdjust(); //Adjust all the DOM items position according to window size
|
|
|
- restoreAllSettings(); //Restore all user prefered settings from storage
|
|
|
- hideMainPlayerInterface();//Hide the main interface
|
|
|
- setTimeout(function(){
|
|
|
- $("#playerInterface").show(); //Only show this interface after it has been out of the working area.
|
|
|
- },500);
|
|
|
-
|
|
|
- //Hide the exit application button if it is not in Virtual Desktop Mode
|
|
|
- if (!ao_module_virtualDesktop){
|
|
|
- $("#closeWebAppButton").hide();
|
|
|
- }
|
|
|
- returnToPreviousState(); //Try to return to previous state using the window hash value
|
|
|
- });
|
|
|
-
|
|
|
- function returnToPreviousState(){
|
|
|
- var hash = window.location.hash;
|
|
|
- if (hash != ""){
|
|
|
- //There are state to restore. Restore using the given hash value
|
|
|
- hash = decodeURIComponent(hash.substring(1));
|
|
|
- var previousState = JSON.parse(hash);
|
|
|
- console.log(previousState);
|
|
|
- var currentPath = previousState.path;
|
|
|
- var playingFileDetail = previousState.pfp;
|
|
|
- var initPauseState = previousState.pause;
|
|
|
- if (currentPath != ""){
|
|
|
- //Create a dummy folder and trick the system to load to the previous states
|
|
|
- var a = '<div></div>';
|
|
|
- a = $(a).attr("filepath",currentPath);
|
|
|
- openFolder(a[0],{playIDAfterOpen: playingFileDetail[0],startPaused: initPauseState});
|
|
|
- }else{
|
|
|
- //currentpath is empty, aka Music Mode
|
|
|
- loadSongList();
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- }else{
|
|
|
- //No state to restore. Restore Viewing Tab instead.
|
|
|
- var vt = loadStorage("viewingTab");
|
|
|
- if (vt != ""){
|
|
|
- if (vt == "music"){
|
|
|
- loadSongList();
|
|
|
-
|
|
|
- }else if(vt == "folder"){
|
|
|
- loadFolderView();
|
|
|
-
|
|
|
- }else if(vt == "playlist"){
|
|
|
- loadPlaylistView();
|
|
|
-
|
|
|
- }else if(vt == "network"){
|
|
|
- loadNetworkView();
|
|
|
-
|
|
|
- }
|
|
|
- }else{
|
|
|
- //Default action
|
|
|
- loadSongList();
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function loadNetworkView(){
|
|
|
- currentMode = "network";
|
|
|
- togglePagingMode(false);
|
|
|
- }
|
|
|
-
|
|
|
- function togglePagingMode(enabled=false){
|
|
|
- if (pagingEnabled != enabled){
|
|
|
- currentPage = 0;
|
|
|
- }
|
|
|
- pagingEnabled = enabled;
|
|
|
- }
|
|
|
-
|
|
|
- function initAudioListeners(){
|
|
|
- //Initiate audio progress bar and link it to the progress bar on the main player interface
|
|
|
- audioElement[0].addEventListener("timeupdate", function() {
|
|
|
- var currentTime = audioElement[0].currentTime;
|
|
|
- var duration = audioElement[0].duration;
|
|
|
- $('#audioProgressBar').css('width',((currentTime +.25)/duration)*100 + '%');
|
|
|
- updatePlaybackDisplayTime(currentTime,duration);
|
|
|
- });
|
|
|
- audioElement[0].onended = function() {
|
|
|
- nextSongHandler();
|
|
|
- };
|
|
|
-
|
|
|
- //Handle clicks on progress bar to skip through the playback
|
|
|
- $("#audioProgressBar").parent().on("click",function(e){
|
|
|
- var duration = audioElement[0].duration;
|
|
|
- var jumpTo = (e.offsetX / $(this).width()) * duration;
|
|
|
- if (!audioElement[0].paused){
|
|
|
- audioElement[0].pause();
|
|
|
- audioElement[0].currentTime = jumpTo;
|
|
|
- audioElement[0].play();
|
|
|
- }else{
|
|
|
- audioElement[0].currentTime = jumpTo;
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function restoreAllSettings(){
|
|
|
- //Repeat mode setting
|
|
|
- var strRmode = loadStorage("repeatMode");
|
|
|
- if (strRmode != ""){
|
|
|
- repeatMode = parseInt(strRmode);
|
|
|
- setRepeatMode(repeatMode,$("#repeatModeButton")[0]);
|
|
|
- }
|
|
|
- //Random Mode setting
|
|
|
- var random = loadStorage("randomMode");
|
|
|
- if (random != ""){
|
|
|
- if (random == "true"){
|
|
|
- //random Mode is true when loaded from configuration. Toggle once to match the setting
|
|
|
- toggleRandomMode($("#randomModeButton")[0]);
|
|
|
- }
|
|
|
- }
|
|
|
- //Timer mode setting
|
|
|
- var TEM = loadStorage("timerEndMode");
|
|
|
- if (TEM != ""){
|
|
|
- if (TEM == "fade"){
|
|
|
- $("#stopMode").val("Fade Out & Stop");
|
|
|
- }else{
|
|
|
- $("#stopMode").val("Stop Playing");
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- //Setting menu function
|
|
|
- function searchOnYoutube(filename = ""){
|
|
|
- if (filename == ""){
|
|
|
- var filename = $("#mainPlayerSongTitle").text().trim();
|
|
|
- }
|
|
|
-
|
|
|
- var url = "https://www.youtube.com/results?search_query=" + filename ;
|
|
|
- window.open(url);
|
|
|
- }
|
|
|
-
|
|
|
- //Searching related functions
|
|
|
- var preSearchItemBuffer = "";
|
|
|
- var searchModeEnabled = false;
|
|
|
- var previousPagingEnabled = false;
|
|
|
- function enterSearchMode(){
|
|
|
- if (searchModeEnabled){
|
|
|
- //Search mode already enabled
|
|
|
- return;
|
|
|
- }
|
|
|
- previousPagingEnabled = pagingEnabled;
|
|
|
- togglePagingMode(false);
|
|
|
- $("#searchModeMenu").show();
|
|
|
- $("#mainMenu").hide();
|
|
|
- let previousDisplayList = Array.from(displayList);
|
|
|
- preSearchItemBuffer = [$("#mainList").html(),$("#interfaceTitle").text(),$("#AMmenuIcon").attr("class"), $("#interfaceDetails").html(), previousDisplayList];
|
|
|
- searchModeEnabled = true;
|
|
|
- }
|
|
|
-
|
|
|
- function exitSearchMode(){
|
|
|
- $("#searchModeMenu").hide();
|
|
|
- $("#mainMenu").show();
|
|
|
- $("#searchInputBar").val("");
|
|
|
- if (preSearchItemBuffer != ""){
|
|
|
- //If previous record exists, recover the previous view from storage
|
|
|
- $("#mainList").html(preSearchItemBuffer[0]);
|
|
|
- $("#interfaceTitle").text(preSearchItemBuffer[1]);
|
|
|
- $("#AMmenuIcon").attr("class",preSearchItemBuffer[2]);
|
|
|
- $("#interfaceDetails").html(preSearchItemBuffer[3]);
|
|
|
- displayList = preSearchItemBuffer[4];
|
|
|
- preSearchItemBuffer = "";
|
|
|
- }
|
|
|
-
|
|
|
- if (previousPagingEnabled){
|
|
|
- togglePagingMode(true);
|
|
|
- }
|
|
|
- searchModeEnabled = false;
|
|
|
- }
|
|
|
-
|
|
|
- document.getElementById("searchInputBar").addEventListener("keypress", function(event){
|
|
|
- console.log(event, event.keyCode);
|
|
|
- if(event.keyCode == 13){
|
|
|
- //Enter pressed. start searching
|
|
|
- searchSong();
|
|
|
- }
|
|
|
- });
|
|
|
-
|
|
|
- /*
|
|
|
- function handleSearchInputEnter(e){
|
|
|
- if(e.keyCode == 13){
|
|
|
- //Enter pressed. start searching
|
|
|
- searchSong();
|
|
|
- }
|
|
|
- }
|
|
|
- */
|
|
|
-
|
|
|
- function searchSong(){
|
|
|
- var keyword = $("#searchInputBar").val();
|
|
|
- if (keyword.includes("?") || keyword.includes("&")){
|
|
|
- alert("Search keyword cannot contains & or ? symbol.");
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- if (keyword == ""){
|
|
|
- //Empty keyword. Return nothing
|
|
|
- return;
|
|
|
- }
|
|
|
- //Clear the mainList, replace with a dummy loading screen
|
|
|
- $("#mainList").html('<div class="mainList item whiteFont">\
|
|
|
- <div class="ui header" style="margin:0px !important;padding-bottom:15px;padding-top:5px;">\
|
|
|
- <i class="loading spinner icon whiteFont" style="overflow:hidden;margin-top:5px;"></i>\
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
|
|
|
- Searching...\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- <div class="topRightCorner" align="center">\
|
|
|
- -1\
|
|
|
- </div>\
|
|
|
- </div>');
|
|
|
-
|
|
|
- ao_module_agirun("Music/functions/listSong.js", {
|
|
|
- listSong: "search:" + keyword
|
|
|
- }, function(data){
|
|
|
- currentPath = "";
|
|
|
- var template = '<div class="mainList file item" filepath={filepath} id={id} rawname={rawname}>\
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">\
|
|
|
- <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>\
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
|
|
|
- {songtitle}\
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- <div class="topRightCorner" align="center">\
|
|
|
- {id}\
|
|
|
- </div>\
|
|
|
- <div class="mainList rightFunctionBar" align="center" onClick="showMore(this);">\
|
|
|
- <i class="ellipsis vertical icon" style="top: 2em;"></i>\
|
|
|
- </div>\
|
|
|
- </div>';
|
|
|
-
|
|
|
- if ($("#searchModeMenu").is(":hidden")){
|
|
|
- //The user exited search mode before the search results come back
|
|
|
- //Dispose the search result
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- if (data.length == 0){
|
|
|
- //No search results.
|
|
|
- $("#mainList").html("");
|
|
|
- $("#mainList").append('<div class="mainList item whiteFont">\
|
|
|
- <div class="ui header" style="margin:0px !important;padding-bottom:15px;padding-top:5px;">\
|
|
|
- <i class="remove icon whiteFont" style="overflow:hidden;margin-top:5px;"></i>\
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
|
|
|
- No result for keyword: ' + keyword + '.\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- <div class="topRightCorner" align="center">\
|
|
|
- -1\
|
|
|
- </div>\
|
|
|
- </div>');
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- $("#interfaceTitle").text("Search");
|
|
|
- $("#AMmenuIcon").attr("class","search icon large whiteFont")
|
|
|
- //Initialize the song list
|
|
|
- displayList = data;
|
|
|
- if (playingList == []){
|
|
|
- playingList = Array.from(displayList);
|
|
|
- pagingEnableForPlayingList = pagingEnabled;
|
|
|
- currentPlayingPage = currentPage;
|
|
|
- }
|
|
|
- $("#mainList").html("");
|
|
|
- for ( var i =0; i < data.length; i++){
|
|
|
- var songInfo = data[i];
|
|
|
- var path = songInfo[0];
|
|
|
- var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
|
|
|
- var ext = songInfo[2];
|
|
|
- var size = songInfo[3];
|
|
|
- var box = template;
|
|
|
- box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
|
|
|
- box = replaceAll("{id}",i + 1,box); //User count from 1
|
|
|
- box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
|
|
|
- box = replaceAll("{songtitle}",displayname,box);
|
|
|
- box = replaceAll("{ext}",ext,box);
|
|
|
- box = replaceAll("{size}",size,box);
|
|
|
- $("#mainList").append(box);
|
|
|
- }
|
|
|
- totalMusicCount = data.length;
|
|
|
- $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
- $("#interfaceDetails").text("[" + i + " songs]");
|
|
|
- highLightPlayingMusic();
|
|
|
- loadThumbnailToMusicList(".mainList.file.item");
|
|
|
- });
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function enterRemotePlayID(object){
|
|
|
- var id = $(object).text().trim();
|
|
|
- $("#remotePlayID").val($("#remotePlayID").val() + id);
|
|
|
- }
|
|
|
-
|
|
|
- function ridOpr(opr){
|
|
|
- if (opr == 'backspace'){
|
|
|
- var currentText = $("#remotePlayID").val();
|
|
|
- var newText = currentText.substring(0,currentText.length - 1);
|
|
|
- $("#remotePlayID").val(newText);
|
|
|
- }else if (opr == 'reset'){
|
|
|
- $("#remotePlayID").val("");
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function openInFileExplorer(){
|
|
|
- var dirname = playingFileDetail[1].split("/");
|
|
|
- var fname = dirname.pop(); //Pop away the filename
|
|
|
- dirname = dirname.join("/") + "/";
|
|
|
- var path = dirname;
|
|
|
- if (dirname.substring(0,1) != "/"){
|
|
|
- //This is not absolute path. Start the path from airMusic
|
|
|
- path = "AirMusic/" + dirname;
|
|
|
- }
|
|
|
-
|
|
|
- path = path.replace("media?file=","");
|
|
|
- if (path.substring(0,1) == "/"){
|
|
|
- path = path.substring(1)
|
|
|
- }
|
|
|
- ao_module_openPath(path, fname);
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- //quickmenu Functions
|
|
|
- function hideAllquickMenu(){
|
|
|
- $(".quickMenu").fadeOut('fast');
|
|
|
- $("#fadeReturnScreen").fadeOut('fast');
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
- //Setting related function
|
|
|
- function showSettingInterface(){
|
|
|
- $("#settingInterface").fadeIn('fast');
|
|
|
- $("#fadeReturnScreen").fadeIn('fast');
|
|
|
- //Update contents in the setting menu
|
|
|
- $("#settingMenuTitle").text($("#mainPlayerSongTitle").text());
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- //Timer related functions
|
|
|
- function showTimerInterface(){
|
|
|
- $("#timerInterface").fadeIn('fast');
|
|
|
- $("#fadeReturnScreen").fadeIn('fast');
|
|
|
- }
|
|
|
-
|
|
|
- function initPlaylistInterfaceList(){
|
|
|
- $("#existsingPlaylist").html("");
|
|
|
- //Get the playlist list from server side
|
|
|
- ao_module_agirun("Music/functions/playlist.js", {
|
|
|
- opr: "root"
|
|
|
- }, function(data){
|
|
|
- data.forEach(playlist => {
|
|
|
- $("#existsingPlaylist").append(`<div class="playlist item selectable whiteFont" name="${playlist.name}" onclick="addSongToSelectedPlaylist(this);">
|
|
|
- <i class="list icon whiteFont"></i> ${playlist.name} [${playlist.count} files]
|
|
|
- </div>`);
|
|
|
- })
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function showPlaylistInterface(){
|
|
|
- initPlaylistInterfaceList();
|
|
|
- $("#playlistInterface").fadeIn('fast');
|
|
|
- $("#fadeReturnScreen").fadeIn('fast');
|
|
|
- }
|
|
|
-
|
|
|
- function addToThisPlaylist(){
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- $("#stopMode").change(function () {
|
|
|
- var end = this.value;
|
|
|
- if (end.includes("Fade Out & Stop")){
|
|
|
- //Start fading out the volume
|
|
|
- timerEndMode = "fade";
|
|
|
- setStorage("timerEndMode","fade");
|
|
|
- }else{
|
|
|
- //End immediately
|
|
|
- timerEndMode = "end";
|
|
|
- setStorage("timerEndMode","end");
|
|
|
- }
|
|
|
- });
|
|
|
-
|
|
|
- var fadeOutStepping = 0.1; //The steps size for fading out during timer countdown
|
|
|
- var defaultVolumeBeforeFadeout = 0;
|
|
|
- function toggleCountDown(button){
|
|
|
- if (!timerMode){
|
|
|
- //Start count down
|
|
|
- timerMode = true;
|
|
|
- $(button).text("Stop");
|
|
|
- timerRemaining = ((parseInt($("#timerHour").text()) * 60) + parseInt($("#timerMinute").text())) * 60; //in seconds
|
|
|
- if (timerRemaining <= 0){
|
|
|
- //Start with 0
|
|
|
- timerMode = false;
|
|
|
- $(button).text("Start");
|
|
|
- return;
|
|
|
- }
|
|
|
- //Update the value on the remaining counter
|
|
|
- $("#remainingUI").text(parseTimer(timerRemaining));
|
|
|
- $("#timerSettingInterface").slideUp('fast');
|
|
|
- $("#timerCountingInterface").slideDown('fast');
|
|
|
- $("#stopMode").attr("disabled","");
|
|
|
- defaultVolumeBeforeFadeout = audioElement[0].volume;
|
|
|
- fadeOutStepping = (audioElement[0].volume) / timerRemaining;
|
|
|
- progressCounter();
|
|
|
- }else{
|
|
|
- //Stop count down
|
|
|
- timerMode = false;
|
|
|
- $(button).text("Start");
|
|
|
- timerRemaining = 0;
|
|
|
- $("#timerSettingInterface").slideDown('fast');
|
|
|
- $("#timerCountingInterface").slideUp('fast');
|
|
|
- $("#stopMode").removeAttr("disabled");
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function progressCounter(){
|
|
|
- //Update the timer
|
|
|
- timerRemaining = timerRemaining - 1;
|
|
|
- $("#remainingUI").text(parseTimer(timerRemaining));
|
|
|
- if (timerRemaining > 0){
|
|
|
- //Continues
|
|
|
- if (timerEndMode == "fade"){
|
|
|
- var newvol = audioElement[0].volume - fadeOutStepping;
|
|
|
- audioElement[0].volume = Math.max(0,newvol);
|
|
|
- }
|
|
|
- setTimeout(progressCounter,1000);
|
|
|
- }else{
|
|
|
- //Times up
|
|
|
- handleTimerEnd();
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function handleTimerEnd(){
|
|
|
- //Stop the playback
|
|
|
- originalVol = audioElement[0].volume;
|
|
|
- fadeAudio();
|
|
|
- setPlaying(false);
|
|
|
- audioElement[0].pause();
|
|
|
- setTimeout(function(){
|
|
|
- audioElement[0].volume = defaultVolumeBeforeFadeout;
|
|
|
- },500);
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function parseTimer(remainingTime){
|
|
|
- //remainingTime in seconds, prase to HH:mm:ss
|
|
|
- var hours = Math.floor(remainingTime / 3600);
|
|
|
- var remainingTime = remainingTime % 3600;
|
|
|
- var minutes = Math.floor(remainingTime / 60);
|
|
|
- var seconds = remainingTime % 60;
|
|
|
- if (minutes < 10){
|
|
|
- minutes = "0" + minutes;
|
|
|
- }
|
|
|
- if (seconds < 10){
|
|
|
- seconds = "0" + seconds;
|
|
|
- }
|
|
|
- var formatted = hours + ":" + minutes + ":" + seconds;
|
|
|
- return formatted;
|
|
|
- }
|
|
|
-
|
|
|
- function addTimer(unit,value){
|
|
|
- if (unit == "h"){
|
|
|
- if (value == "r"){
|
|
|
- $("#timerHour").text("0");
|
|
|
- }else{
|
|
|
- $("#timerHour").text($("#timerHour").text() - -1 * value);
|
|
|
- }
|
|
|
-
|
|
|
- }else if (unit == "m"){
|
|
|
- if (value == "r"){
|
|
|
- $("#timerMinute").text("0");
|
|
|
- }else{
|
|
|
- $("#timerMinute").text($("#timerMinute").text() - -1 * value);
|
|
|
- }
|
|
|
- }
|
|
|
- if (parseInt($("#timerHour").text()) < 0){
|
|
|
- $("#timerHour").text("0");
|
|
|
- }
|
|
|
- if (parseInt($("#timerMinute").text()) < 0){
|
|
|
- if (parseInt($("#timerHour").text()) > 0){
|
|
|
- //subtract 1 from the hour
|
|
|
- $("#timerHour").text($("#timerHour").text() - 1);
|
|
|
- $("#timerMinute").text(parseInt($("#timerMinute").text()) + 60);
|
|
|
- }else{
|
|
|
- //There is no hours to subtract
|
|
|
- $("#timerMinute").text("0");
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
- if (parseInt($("#timerMinute").text()) > 59){
|
|
|
- $("#timerMinute").text(parseInt($("#timerMinute").text()) - 60);
|
|
|
- $("#timerHour").text($("#timerHour").text() - -1);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- //Download function
|
|
|
- function downloadPlayingSong(){
|
|
|
- var url = audioElement.attr("src");
|
|
|
- var ext = url.split(".").pop();
|
|
|
- var filename = $("#mainPlayerSongTitle").text().trim();
|
|
|
- generateDownloadElement(url,filename);
|
|
|
- /*
|
|
|
- if (url.includes("/media")){
|
|
|
- blobDownloadElement(url,filename);
|
|
|
- }else{
|
|
|
- generateDownloadElement(url,filename);
|
|
|
- }
|
|
|
- */
|
|
|
- }
|
|
|
-
|
|
|
- var downloadInProgress = false;
|
|
|
- function blobDownloadElement(filepath,filename){
|
|
|
- if (downloadInProgress){
|
|
|
- alert("Please wait until another download finished.");
|
|
|
- return;
|
|
|
- }
|
|
|
- downloadInProgress = true;
|
|
|
- $("#downloadBtn").find("i").addClass("disabled");
|
|
|
- $("#downloadProgressBar").find(".bar").css("width","0%");
|
|
|
- $("#downloadProgressBar").show();
|
|
|
- var xhr = new XMLHttpRequest();
|
|
|
- xhr.onreadystatechange = function(){
|
|
|
- if (this.readyState == 4 && this.status == 200){
|
|
|
- //handler(this.response);
|
|
|
- //console.log(this.response, typeof this.response);
|
|
|
- var url = window.URL || window.webkitURL;
|
|
|
- //Open the downloaded data in new window for testing
|
|
|
- //window.open(url.createObjectURL(this.response));
|
|
|
- generateDownloadElement(url.createObjectURL(this.response),filename);
|
|
|
- $("#downloadBtn").find("i").removeClass("disabled");
|
|
|
- $("#downloadProgressBar").hide();
|
|
|
- downloadInProgress = false;
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
- xhr.onprogress = function (event) {
|
|
|
- console.log("[AirMusic] Download Progress: " + event.loaded + " / " + event.total + "Bytes");
|
|
|
- $("#downloadProgressBar").find(".bar").css("width",(event.loaded / event.total) * 100 + "%");
|
|
|
- };
|
|
|
- xhr.open('GET', filepath);
|
|
|
- xhr.responseType = 'blob';
|
|
|
- xhr.send();
|
|
|
- }
|
|
|
-
|
|
|
- function generateDownloadElement(filepath, filename){
|
|
|
- var link = document.createElement('a');
|
|
|
-
|
|
|
- //Clean the filepath
|
|
|
- filepath = filepath.split("//").join("/");
|
|
|
- //Convert the filepath to download path
|
|
|
- link.href = filepath.replace("/media?file=", "/media/download/?file=");
|
|
|
-
|
|
|
- //Generate the download element
|
|
|
- link.setAttribute('download', filename);
|
|
|
- document.getElementsByTagName("body")[0].appendChild(link);
|
|
|
- // Firefox
|
|
|
- if (document.createEvent) {
|
|
|
- var event = document.createEvent("MouseEvents");
|
|
|
- event.initEvent("click", true, true);
|
|
|
- link.dispatchEvent(event);
|
|
|
- }
|
|
|
- // IE
|
|
|
- else if (link.click) {
|
|
|
- link.click();
|
|
|
- }
|
|
|
- link.parentNode.removeChild(link);
|
|
|
- }
|
|
|
-
|
|
|
- //Volume control related features
|
|
|
- var previousAudioVolume = 0;
|
|
|
- function mute(button){
|
|
|
- if (audioElement[0].volume != 0){
|
|
|
- //Set to mute
|
|
|
- previousAudioVolume = audioElement[0].volume;
|
|
|
- audioElement[0].volume = 0;
|
|
|
- $(button).css("opacity", "1");
|
|
|
- $("#voldownBtn").addClass("disabled");
|
|
|
- $("#volupBtn").addClass("disabled");
|
|
|
- }else{
|
|
|
- //Restore from mute
|
|
|
- audioElement[0].volume = previousAudioVolume;
|
|
|
- $(button).css("opacity", "0.6");
|
|
|
- $("#voldownBtn").removeClass("disabled");
|
|
|
- $("#volupBtn").removeClass("disabled");
|
|
|
- }
|
|
|
- $("#volBar").css("width",audioElement[0].volume * 100 + "%");
|
|
|
- }
|
|
|
- var volbarTimeoutEvt;
|
|
|
- function addAudioVolume(value){
|
|
|
- $("#volumeGUI").slideDown('fast');
|
|
|
- if (volbarTimeoutEvt !== undefined){
|
|
|
- clearTimeout(volbarTimeoutEvt);
|
|
|
- }
|
|
|
- volbarTimeoutEvt = setTimeout(function(){
|
|
|
- $("#volumeGUI").slideUp('fast');
|
|
|
- },3000);
|
|
|
- $("#volumeGUI")
|
|
|
- if (audioElement[0].volume + value >= 1){
|
|
|
- audioElement[0].volume = 1;
|
|
|
- }else if (audioElement[0].volume + value <= 0){
|
|
|
- audioElement[0].volume = 0;
|
|
|
- }else{
|
|
|
- audioElement[0].volume += value;
|
|
|
- }
|
|
|
-
|
|
|
- $("#volBar").css("width",Math.min(100,audioElement[0].volume * 100) + "%");
|
|
|
- }
|
|
|
-
|
|
|
- function updatePlaybackDisplayTime(currentTime,duration){
|
|
|
- let progress = Math.round(currentTime);
|
|
|
- let remainTime = Math.round(duration - currentTime);
|
|
|
- $("#progressTime").text(secondsToHms(progress));
|
|
|
- if (!isNaN(remainTime)){
|
|
|
- $("#remainTime").text("-" + secondsToHms(remainTime));
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function nextSongHandler(){
|
|
|
- console.log("Next song");
|
|
|
- if (repeatMode == 1){
|
|
|
- //Next song in the playlist
|
|
|
- if (randomMode){
|
|
|
- var randomTrackID = Math.floor(Math.random() * (playingList.length - 1));
|
|
|
- //console.log(randomTrackID, playingList[randomTrackID]);
|
|
|
- nextSong(randomTrackID,true);
|
|
|
- }else{
|
|
|
- var playingSongIndex = playingFileDetail[0];
|
|
|
- var nextSongIndex = playingSongIndex;
|
|
|
- if (nextSongIndex >= playingList.length){
|
|
|
- nextSongIndex = 0;
|
|
|
- }
|
|
|
- //console.log(playingSongIndex, nextSongIndex);
|
|
|
- nextSong(nextSongIndex,true);
|
|
|
- }
|
|
|
- }else if (repeatMode == 2){
|
|
|
- //Loop this song only
|
|
|
- audioElement[0].pause();
|
|
|
- audioElement[0].currentTime = 0;
|
|
|
- audioElement[0].play();
|
|
|
- }else if (repeatMode == 0){
|
|
|
- //Pause after finish this song
|
|
|
- setPlaying(false);
|
|
|
- audioElement[0].pause();
|
|
|
- audioElement[0].currentTime = 0;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function mainPlayerShown(){
|
|
|
- if ($("#playerInterface").offset().left == 0){
|
|
|
- return true;
|
|
|
- }
|
|
|
- return false;
|
|
|
- }
|
|
|
-
|
|
|
- function toggleRandomMode(button){
|
|
|
- var btn = $(button).find("i");
|
|
|
- if (randomMode){
|
|
|
- //Set the random mode to false
|
|
|
- btn.removeClass("enabled").addClass("disabled");
|
|
|
- randomMode = false;
|
|
|
- }else{
|
|
|
- btn.removeClass("disabled").addClass("enabled");
|
|
|
- randomMode = true;
|
|
|
- }
|
|
|
- setStorage("randomMode",randomMode);
|
|
|
- }
|
|
|
-
|
|
|
- function toggleRepeatMode(button){
|
|
|
- if (repeatMode == 0){
|
|
|
- setRepeatMode(1,button);
|
|
|
- setStorage("repeatMode",1);
|
|
|
- }else if (repeatMode == 1){
|
|
|
- setRepeatMode(2,button);
|
|
|
- setStorage("repeatMode",2);
|
|
|
- }else if (repeatMode == 2){
|
|
|
- setRepeatMode(0,button);
|
|
|
- setStorage("repeatMode",0);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function setRepeatMode(modeCode,button){
|
|
|
- if (modeCode == 1){
|
|
|
- //Set repeat mode to "Repeat all"
|
|
|
- $(button).find(".singleLoop").hide();
|
|
|
- $(button).find("i").removeClass("disabled").addClass("enabled");
|
|
|
- repeatMode = 1;
|
|
|
- }else if (modeCode == 2){
|
|
|
- //Set repeat mode to "Repeat one"
|
|
|
- $(button).find(".singleLoop").show();
|
|
|
- $(button).find("i").removeClass("disabled").addClass("enabled");
|
|
|
- repeatMode = 2;
|
|
|
- }else if (modeCode == 0){
|
|
|
- //Set repeat mode to "None"
|
|
|
- $(button).find(".singleLoop").hide();
|
|
|
- $(button).find("i").removeClass("enabled").addClass("disabled");
|
|
|
- repeatMode = 0;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- function hideMainPlayerInterface(){
|
|
|
- $("#playerInterface").animate({left: window.innerWidth},300);
|
|
|
- $("body").css("overflow-y","auto");
|
|
|
- }
|
|
|
-
|
|
|
- function showMainPlayerInterface(){
|
|
|
- $("#playerInterface").animate({left:0},300);
|
|
|
- $("body").css("overflow-y","hidden");
|
|
|
- }
|
|
|
-
|
|
|
- function syncSystemVol(){
|
|
|
- //Initiate the module's volume to the system's
|
|
|
- var globalvol = localStorage.getItem("global_volume");
|
|
|
- //console.log("Global Volume" + globalvol.toString());
|
|
|
- if (!globalvol){
|
|
|
- globalvol = 0.1;
|
|
|
- }
|
|
|
-
|
|
|
- //Check if it is mobile. If yes, always 100% and leave volume to system
|
|
|
- if (isMobile()){
|
|
|
- globalvol = 1;
|
|
|
- //$(".desktopOnly").addClass("disabled");
|
|
|
- }
|
|
|
-
|
|
|
- audioElement[0].volume = parseFloat(globalvol);
|
|
|
- $("#volBar").css("width",audioElement[0].volume * 100 + "%");
|
|
|
- }
|
|
|
-
|
|
|
- function replaceAll(target, replace, original){
|
|
|
- return original.split(target).join(replace);
|
|
|
- };
|
|
|
-
|
|
|
- function playSong(object, startPaused = false){
|
|
|
- if ($(object).parent().hasClass("playingTrack")){
|
|
|
- //This is already the song that is currently playing. Show the main player interface instead.
|
|
|
- showMainPlayerInterface();
|
|
|
- resizeQuickAdjust();
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- //$(".playingTrack").removeClass("playingTrack");
|
|
|
- //$(object).parent().addClass("playingTrack");
|
|
|
- var filepath = ao_module_utils.attrToObject($(object).parent().attr('filepath'));
|
|
|
- var rawname = ao_module_utils.attrToObject($(object).parent().attr('rawname'));
|
|
|
- var displayName = ao_module_codec.decodeUmFilename(rawname);
|
|
|
- var info = $(object).parent().find(".fileinfo").text();
|
|
|
- var id = $(object).parent().attr('id');
|
|
|
- if (id === undefined){
|
|
|
- //This might be a file from dropdown list. use listid instead
|
|
|
- id = $(object).parent().attr('listid');
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- updateMiniPlayerUI(displayName,info,id);
|
|
|
-
|
|
|
- //Check if page mode is enabled. If yes, do auto tab switch
|
|
|
- if (pagingEnabled){
|
|
|
- var targetPageNumber = getPageNumberByPlaybackId(parseInt(id));
|
|
|
- switchToPage(targetPageNumber);
|
|
|
- }
|
|
|
-
|
|
|
- //Seperate out the filepath without media server path
|
|
|
- var fileVpath = filepath.split("=")[1];
|
|
|
- updateMainPlayerUI(displayName,info,id, fileVpath);
|
|
|
- loadAndPlayAudioFile(filepath, !startPaused);
|
|
|
- if (!currentPlaying && !startPaused){
|
|
|
- setPlaying(true);
|
|
|
- }
|
|
|
- playingFileDetail = [parseInt(id),filepath];
|
|
|
-
|
|
|
- //If the user play this file, assume the next file to be added to playlist is this file
|
|
|
- playlistAddPendingFile = filepath;
|
|
|
- //Have a backup of the current list just incase the user open another directory while a list is playing
|
|
|
- //Only backup when it is not call from dropdown list
|
|
|
- if (!$(object).parent().hasClass("dropdownList")){
|
|
|
- playingList = Array.from(displayList);
|
|
|
- pagingEnableForPlayingList = pagingEnabled;
|
|
|
- currentPlayingPage = currentPage;
|
|
|
- }
|
|
|
- parsePlayingSongList();
|
|
|
- if ($("#miniPlayer").hasClass("hidden")){
|
|
|
- //This is the first song that has played after the UI init
|
|
|
- $("#miniPlayer").css("display","none").removeClass("hidden").slideDown();
|
|
|
- }
|
|
|
- showMainPlayerInterface();
|
|
|
-
|
|
|
- //Just in case the albumn image changed, albumn art css is also updated.
|
|
|
- resizeQuickAdjust();
|
|
|
-
|
|
|
- updateStateReferingURL();
|
|
|
- highLightPlayingMusic();
|
|
|
- }
|
|
|
-
|
|
|
- function loadThumbnailToMusicList(selector = ".mainList.file.item"){
|
|
|
- $(selector).each(function(){
|
|
|
- let filepath = JSON.parse(decodeURIComponent($(this).attr("filepath")));
|
|
|
- let thisFileItemID = $(this).attr("id");
|
|
|
- let rawname = JSON.parse(decodeURIComponent($(this).attr("rawname")));
|
|
|
-
|
|
|
- let realVpath = filepath.split("=");
|
|
|
- realVpath.shift();
|
|
|
- realVpath = realVpath.join("=");
|
|
|
- //console.log(thisFileItemID, realVpath, rawname);
|
|
|
- //Load the thumbnail for this item in list
|
|
|
- fetch("../system/file_system/loadThumbnail?vpath=" + encodeURIComponent(realVpath))
|
|
|
- .then((response) => response.json())
|
|
|
- .then((data) => {
|
|
|
- if (data.error !== undefined || data.trim() == ""){
|
|
|
- return;
|
|
|
- }
|
|
|
- $("#" + thisFileItemID).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
- $(".dropdownList.file.item").each(function(){
|
|
|
- var thisFilepath = JSON.parse(decodeURIComponent($(this).attr("filepath")));
|
|
|
- if ($(this).attr("listid") == thisFileItemID && thisFilepath == filepath){
|
|
|
- $(this).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
- }
|
|
|
- });
|
|
|
- });
|
|
|
- /*
|
|
|
- ao_module_agirun("Music/functions/getThumbnail.js", {
|
|
|
- file: realVpath,
|
|
|
- }, function(data){
|
|
|
- if (data.error !== undefined){
|
|
|
-
|
|
|
- }else{
|
|
|
- $("#" + thisFileItemID).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
- $(".dropdownList.file.item").each(function(){
|
|
|
- var thisFilepath = JSON.parse(decodeURIComponent($(this).attr("filepath")));
|
|
|
- if ($(this).attr("listid") == thisFileItemID && thisFilepath == filepath){
|
|
|
- $(this).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
- }
|
|
|
- })
|
|
|
- }
|
|
|
- });
|
|
|
- */
|
|
|
- })
|
|
|
- }
|
|
|
-
|
|
|
- function loadThumbnail(filepath){
|
|
|
- //Load the thumbnail
|
|
|
- var realVpath = filepath.split("=");
|
|
|
- realVpath.shift();
|
|
|
- realVpath = realVpath.join("=");
|
|
|
- ao_module_agirun("Music/functions/getThumbnail.js", {
|
|
|
- file: realVpath,
|
|
|
- }, function(data){
|
|
|
- if (data.error !== undefined){
|
|
|
- console.log(data.error)
|
|
|
- $("#albumnArtImage").attr("src","img/default.png");
|
|
|
- $("#smallPlayerThumb").attr("src","img/default.png");
|
|
|
- if (navigator.mediaSession.metadata){
|
|
|
- navigator.mediaSession.metadata.artwork = [
|
|
|
- { src: "img/default.png", sizes: '512x512', type: 'image/png' }
|
|
|
- ]
|
|
|
- }
|
|
|
- }else{
|
|
|
- $("#albumnArtImage").attr("src","data:image/jpg;base64," + data);
|
|
|
- $("#smallPlayerThumb").attr("src","data:image/jpg;base64," + data);
|
|
|
- if (isAndroid && navigator.mediaSession.metadata){
|
|
|
- //Android
|
|
|
- navigator.mediaSession.metadata.artwork = [
|
|
|
- { src: "data:image/jpg;base64," + data, sizes: '480x480', type: 'image/jpg' }
|
|
|
- ]
|
|
|
- }
|
|
|
- }
|
|
|
- resizeQuickAdjust();
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function nextSong(id = false,forcePlayEvenStopped = false){
|
|
|
- //Check if the current playlist has more than one songs.
|
|
|
- var currentPaused = audioElement[0].paused;
|
|
|
- if (playingList.length > 1){
|
|
|
- var nextSongIndex = playingFileDetail[0];
|
|
|
- if (id != false){
|
|
|
- //Allow manual overwrite for this function
|
|
|
- nextSongIndex = id;
|
|
|
- var nextSong = playingList[nextSongIndex];
|
|
|
- }else{
|
|
|
- if (playingFileDetail[0] >= playingList.length){
|
|
|
- var nextSong = playingList[0]; //Back to the first song in list
|
|
|
- nextSongIndex = 0;
|
|
|
- }else{
|
|
|
- var nextSong = playingList[nextSongIndex]; //Index of the playList + 1 as the index for the files is alrady i+1. (User count from one)
|
|
|
- }
|
|
|
- }
|
|
|
- //The songs has to be play via background services instead of relying on DOM
|
|
|
- var filepath = nextSong[0];
|
|
|
- var rawname = nextSong[1];
|
|
|
- var displayName = ao_module_codec.decodeUmFilename(rawname);
|
|
|
- var info = nextSong[2] + " / " + nextSong[3];
|
|
|
- var id = nextSongIndex + 1;
|
|
|
- updateMiniPlayerUI(displayName,info,id);
|
|
|
-
|
|
|
- //Seperate out the filepath without media server path
|
|
|
- var fileVpath = filepath.split("=")[1];
|
|
|
- updateMainPlayerUI(displayName,info,id, fileVpath);
|
|
|
- if (forcePlayEvenStopped){
|
|
|
- loadAndPlayAudioFile(filepath,true);
|
|
|
- }else{
|
|
|
- loadAndPlayAudioFile(filepath,!currentPaused);
|
|
|
- }
|
|
|
-
|
|
|
- if (currentPlaying){
|
|
|
- setPlaying(true);
|
|
|
- }
|
|
|
- playingFileDetail = [id,filepath];
|
|
|
- //Need not to update the playlist because it is the same
|
|
|
- if (pagingEnabled){
|
|
|
- var targetPageNumber = getPageNumberByPlaybackId(parseInt(id));
|
|
|
- switchToPage(targetPageNumber, function(){
|
|
|
- parsePlayingSongList();
|
|
|
- highLightPlayingMusic();
|
|
|
- });
|
|
|
- }else{
|
|
|
- highLightPlayingMusic();
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- }else{
|
|
|
- //This is the only song in playlist. Play again from start
|
|
|
- audioElement[0].pause();
|
|
|
- audioElement[0].currentTime = 0;
|
|
|
- if (!currentPaused){
|
|
|
- audioElement[0].play();
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
- updateStateReferingURL();
|
|
|
- }
|
|
|
-
|
|
|
- function previousSong(){
|
|
|
- var currentPaused = audioElement[0].paused;
|
|
|
- if (playingList.length > 1){
|
|
|
- var nextSongIndex = playingFileDetail[0] - 1;
|
|
|
- if (nextSongIndex == 0){
|
|
|
- nextSongIndex = playingList.length;
|
|
|
- var nextSong = playingList[nextSongIndex - 1]; //Back to the last song in list
|
|
|
- }else{
|
|
|
- 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)
|
|
|
- }
|
|
|
- //The songs has to be play via background services instead of relying on DOM
|
|
|
- var filepath = nextSong[0];
|
|
|
- var rawname = nextSong[1];
|
|
|
- var displayName = ao_module_codec.decodeUmFilename(rawname);
|
|
|
- var info = nextSong[2] + " / " + nextSong[3];
|
|
|
- var id = nextSongIndex;
|
|
|
- updateMiniPlayerUI(displayName,info,id);
|
|
|
-
|
|
|
- //Seperate out the filepath without media server path
|
|
|
- var fileVpath = filepath.split("=")[1];
|
|
|
- updateMainPlayerUI(displayName,info,id, fileVpath);
|
|
|
- loadAndPlayAudioFile(filepath,!currentPaused);
|
|
|
- if (currentPlaying){
|
|
|
- setPlaying(true);
|
|
|
- }
|
|
|
- playingFileDetail = [id,filepath];
|
|
|
- //Need not to update the playlist because it is the same
|
|
|
- if (pagingEnabled){
|
|
|
- var targetPageNumber = getPageNumberByPlaybackId(parseInt(id));
|
|
|
- switchToPage(targetPageNumber, function(){
|
|
|
- parsePlayingSongList();
|
|
|
- highLightPlayingMusic();
|
|
|
-
|
|
|
- });
|
|
|
- }else{
|
|
|
- highLightPlayingMusic();
|
|
|
- }
|
|
|
- }else{
|
|
|
- //This is the only song in playlist. Play again from start
|
|
|
- audioElement[0].pause();
|
|
|
- audioElement[0].currentTime = 0;
|
|
|
- audioElement[0].play();
|
|
|
- }
|
|
|
- updateStateReferingURL();
|
|
|
- }
|
|
|
-
|
|
|
- var originalVol = 0;
|
|
|
- function togglePlayMode(){
|
|
|
- if (currentPlaying){
|
|
|
- originalVol = audioElement[0].volume;
|
|
|
- fadeAudio();
|
|
|
- setPlaying(false);
|
|
|
- }else{
|
|
|
- gainAudio();
|
|
|
- setPlaying(true);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- let audioTransitioning = false;
|
|
|
- function fadeAudio(){
|
|
|
- audioTransitioning = true;
|
|
|
- if(audioElement[0].volume > 0){
|
|
|
- var testval = audioElement[0].volume - 0.1;
|
|
|
- audioElement[0].volume = Math.max(0,testval);
|
|
|
- setTimeout(fadeAudio, 50);
|
|
|
- }else{
|
|
|
- audioElement[0].pause();
|
|
|
- setTimeout(function(){
|
|
|
- audioElement[0].volume = originalVol;
|
|
|
- updateStateReferingURL();
|
|
|
- audioTransitioning = false;
|
|
|
- }, 100);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- var targetVol = 0;
|
|
|
- function gainAudio(){
|
|
|
- targetVol = audioElement[0].volume;
|
|
|
- audioElement[0].volume = 0;
|
|
|
- setTimeout(function(){
|
|
|
- audioElement[0].play();
|
|
|
- recursiveGainAudio();
|
|
|
- }, 100);
|
|
|
- }
|
|
|
-
|
|
|
- function recursiveGainAudio(){
|
|
|
- audioTransitioning = true;
|
|
|
- if(audioElement[0].volume < targetVol){
|
|
|
- if (audioElement[0].volume + 0.1 < 1){
|
|
|
- audioElement[0].volume += 0.1
|
|
|
- }else{
|
|
|
- //Audio volume larger than 1, make it 1
|
|
|
- audioElement[0].volume = 1
|
|
|
- }
|
|
|
- setTimeout(recursiveGainAudio, 50);
|
|
|
- }else{
|
|
|
- setTimeout(function(){
|
|
|
- audioElement[0].volume = targetVol;
|
|
|
- updateStateReferingURL();
|
|
|
- audioTransitioning = false;
|
|
|
- }, 100);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- //Update all icons on play and pause buttons
|
|
|
- function setPlaying(playing){
|
|
|
- if (playing == true){
|
|
|
- $(".PlayButton").attr("class","pause icon whiteFont PlayButton");
|
|
|
- currentPlaying = true;
|
|
|
- }else{
|
|
|
- $(".PlayButton").attr("class","play icon whiteFont PlayButton");
|
|
|
- currentPlaying = false;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function loadAndPlayAudioFile(sourceURL,playAfterLoad = true){
|
|
|
- var audio = audioElement;
|
|
|
-
|
|
|
- //These was added to fix the legacy file listing issue
|
|
|
- var fd = sourceURL.split("=");
|
|
|
- var rootPath = fd.shift();
|
|
|
- var filename = fd.join("=");
|
|
|
- var playbackURL = "../" + rootPath + "=" + encodeURIComponent(filename); //Convert absolute dir to relative
|
|
|
- //console.log(playbackURL);
|
|
|
- $("#mainAudioPlayer").attr("src", playbackURL);
|
|
|
-
|
|
|
- audio[0].pause();
|
|
|
- audio[0].load();
|
|
|
- if(playAfterLoad){
|
|
|
- audio[0].oncanplaythrough = audio[0].play();
|
|
|
- }else{
|
|
|
- var currentTime = audioElement[0].currentTime;
|
|
|
- var duration = audioElement[0].duration;
|
|
|
- $('#audioProgressBar').css('width','0%');
|
|
|
- updatePlaybackDisplayTime(currentTime,duration);
|
|
|
- }
|
|
|
-
|
|
|
- loadThumbnail(sourceURL);
|
|
|
- }
|
|
|
-
|
|
|
- function updateMiniPlayerUI(displayName, fileinfo,id){
|
|
|
- $("#miniPlayerDisplayName").text(displayName);
|
|
|
- $("#miniPlayerInformation").text(fileinfo);
|
|
|
- $("#miniPlayerIDtab").text(id + "/" + totalMusicCount);
|
|
|
- }
|
|
|
-
|
|
|
- function updateMainPlayerUI(songname, fileinfo, id, filepath){
|
|
|
- $("#mainPlayerSongTitle").text(songname);
|
|
|
- $("#mainPlayerSongDesc").text(fileinfo);
|
|
|
- $("#mainPlayerMiniTab").text(id + "/" + totalMusicCount);
|
|
|
- if ('mediaSession' in navigator){
|
|
|
- var infoRewrite = fileinfo.split(" / ")
|
|
|
- updateTitle(songname, infoRewrite[1] + " (" + infoRewrite[0] + ")", id + "/" + totalMusicCount, filepath);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- /*
|
|
|
- Playlist related functions
|
|
|
-
|
|
|
- */
|
|
|
-
|
|
|
- function renderPlaylistByName(listname){
|
|
|
- $("#interfaceTitle").text(listname);
|
|
|
- ao_module_agirun("Music/functions/playlist.js", {
|
|
|
- opr: "list",
|
|
|
- playlistname: listname
|
|
|
- }, function(data){
|
|
|
- $("#mainList").html("");
|
|
|
- if (data.error !== undefined){
|
|
|
- //This playlist no longer exists. Back to main
|
|
|
- loadPlaylistView();
|
|
|
- }else{
|
|
|
- //Updat ethe global values
|
|
|
- displayList = data;
|
|
|
- totalMusicCount = data.length;
|
|
|
- currentPath = listname;
|
|
|
- //Add the back btn for back to playlist view
|
|
|
- $("#mainList").append(`<div class="mainList item extrapadding" onClick="loadPlaylistView();">
|
|
|
- <div class="ui header selectable" style="margin:0px !important;">
|
|
|
- <i class="angle left icon whiteFont" style="overflow:hidden;"></i>
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
|
|
|
- ../
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>`);
|
|
|
-
|
|
|
-
|
|
|
- //List the resulting song list
|
|
|
- for (var i = 0; i < data.length; i++){
|
|
|
- var songInfo = data[i];
|
|
|
-
|
|
|
- $("#mainList").append(`<div class="mainList file item" filepath=${ao_module_utils.objectToAttr(songInfo[0])} id=${i+1} rawname=${ao_module_utils.objectToAttr(songInfo[1])}>
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">
|
|
|
- <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">
|
|
|
- ${songInfo[1]}
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">${songInfo[2]} / ${songInfo[3]}</div>
|
|
|
- </div>
|
|
|
- </div>\
|
|
|
- <div class="topRightCorner" align="center">
|
|
|
- ${i + 1}
|
|
|
- </div>
|
|
|
- <div class="mainList rightFunctionBar" type="file" align="center" onclick="showMore(this);">
|
|
|
- <i class="ellipsis vertical icon" style="margin-top:1.2em;"></i>
|
|
|
- </div>
|
|
|
- </div>`);
|
|
|
-
|
|
|
- //Update the interface Detail
|
|
|
- $("#interfaceDetails").text("[" + data.length + " files]");
|
|
|
- };
|
|
|
-
|
|
|
-
|
|
|
- loadThumbnailToMusicList();
|
|
|
- //Fix some legacy css isseus
|
|
|
- $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
- }
|
|
|
- });
|
|
|
- }
|
|
|
- //Open a given playlist
|
|
|
- function openPlaylist(object){
|
|
|
- var listname = $(object).attr("listname");
|
|
|
- renderPlaylistByName(listname);
|
|
|
- }
|
|
|
-
|
|
|
- function addToNewPlaylist(){
|
|
|
- var playlistname = prompt("Enter new playlist name");
|
|
|
- if (playlistname != null && playlistname != ""){
|
|
|
- //Add to playlist
|
|
|
- addSongToPlayList(playlistname);
|
|
|
- }else{
|
|
|
- $("#succSnackBar").find(".content").html(`<i class="remove icon"></i> Invalid playlist name`);
|
|
|
- $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
|
|
|
-
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function addToPlaylistFromMoreMenu(){
|
|
|
- $(".showMoreMenus").fadeOut('fast');
|
|
|
- $("#showmoreUIcover").fadeOut('fast');
|
|
|
- showPlaylistInterface();
|
|
|
- }
|
|
|
-
|
|
|
- //Remove the curernt file from the current playlist
|
|
|
- function removeFromPlylistFromMoreMenu(){
|
|
|
- if (currentMode == "playlist"){
|
|
|
- //Remvoe the playlistAddPendingFile
|
|
|
- //In playlist mode, the currentPath is used to store the plylist name
|
|
|
- ao_module_agirun("Music/functions/playlist.js", {
|
|
|
- opr: "remove",
|
|
|
- playlistname: currentPath,
|
|
|
- musicpath: playlistAddPendingFile.replace("/media?file=","")
|
|
|
- }, function(data){
|
|
|
- if (data.error !== undefined){
|
|
|
- alert(data.error);
|
|
|
- }else{
|
|
|
- //Removed. Reload playlist
|
|
|
- renderPlaylistByName(currentPath);
|
|
|
-
|
|
|
- //Hide the MoreMenu
|
|
|
- hideShowMoreMenu();
|
|
|
- }
|
|
|
- });
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function addSongToSelectedPlaylist(object){
|
|
|
- var playlistName = $(object).attr("name").trim();
|
|
|
- addSongToPlayList(playlistName);
|
|
|
- }
|
|
|
-
|
|
|
- function addSongToPlayList(listname){
|
|
|
- //Get current music file name
|
|
|
- var currentSongPath = "";
|
|
|
- if (playlistAddPendingFile !== ""){
|
|
|
- currentSongPath = playlistAddPendingFile.replace("/media?file=","");
|
|
|
- }else{
|
|
|
- $("#succSnackBar").find(".content").html(`<i class="remove icon"></i> No song selected`);
|
|
|
- $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- //Add to playlist
|
|
|
- ao_module_agirun("Music/functions/playlist.js", {
|
|
|
- opr: "add",
|
|
|
- playlistname: listname,
|
|
|
- musicpath: currentSongPath
|
|
|
- },function(data){
|
|
|
- //Show success
|
|
|
- $("#succSnackBar").find(".content").html(`<i class="checkmark icon"></i> Added to playlist ${listname}`);
|
|
|
- $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
|
|
|
-
|
|
|
- //Reload the playlist
|
|
|
- initPlaylistInterfaceList();
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- //List the number of playlist stored in this database
|
|
|
- function loadPlaylistView(){
|
|
|
- currentMode = "playlist";
|
|
|
- //Clear the main list
|
|
|
- $("#mainList").html("");
|
|
|
- togglePagingMode(false);
|
|
|
- ao_module_agirun("Music/functions/playlist.js", {
|
|
|
- opr: "root",
|
|
|
- },function(data){
|
|
|
- //Get the list of playlist
|
|
|
- console.log(data);
|
|
|
-
|
|
|
- //Render the elements
|
|
|
- data.forEach(playlist => {
|
|
|
- $("#mainList").append(`<div class="mainList item" listname="${playlist.name}" tag="playlist" onClick="openPlaylist(this);">
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" >
|
|
|
- <img class="ui small image" src="img/list.svg" style="margin-right: 0.2em;"></img>
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
|
|
|
- ${playlist.name}
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">[${playlist.count} files]</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="mainList rightFunctionBar" tag="moreInfo" align="center" onClick="openPlaylist(this);">
|
|
|
- <i class="chevron right icon" style="margin-top:1.2em;"></i>
|
|
|
- </div>
|
|
|
- </div>`);
|
|
|
- });
|
|
|
- hideLeftMenu();
|
|
|
-
|
|
|
- //Update the headers
|
|
|
- $("#interfaceTitle").text("Playlist");
|
|
|
- $("#AMmenuIcon").attr("class","list icon large whiteFont");
|
|
|
- $("#interfaceDetails").text("[" + data.length + " playlist]");
|
|
|
- });
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function loadFolderView(){
|
|
|
- currentMode = "folder";
|
|
|
- currentPath = "";
|
|
|
- var tempalte = '<div class="mainList item" filepath={filepath} id={id} tag="folder" onClick="openFolder(this);">\
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" >\
|
|
|
- <img class="ui small image" src="img/fo.svg" style="margin-right: 0.2em;"></img>\
|
|
|
- <div class="content whiteFont" style="padding-top: 2px; line-height:1em;width : 80%; font-weight: lighter;">\
|
|
|
- {foldername}\
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">{fileinfo}</div>\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- <div class="mainList rightFunctionBar" tag="moreInfo" align="center" onClick="moreFolder(this);">\
|
|
|
- <i class="chevron right icon" style="margin-top:1.2em;"></i>\
|
|
|
- </div>\
|
|
|
- </div>';
|
|
|
- $("#interfaceTitle").text("Storage");
|
|
|
- togglePagingMode(false);
|
|
|
- $("#AMmenuIcon").attr("class","folder open icon large whiteFont");
|
|
|
- ao_module_agirun("Music/functions/listSong.js", {
|
|
|
- listdir: "root",
|
|
|
- },function(data){
|
|
|
- $("#mainList").html("");
|
|
|
- for (var i =0; i < data.length; i++){
|
|
|
- var folderName = data[i][0];
|
|
|
- var folderPath = data[i][1];
|
|
|
- var fileCount = data[i][2];
|
|
|
- var folderCount = data[i][3];
|
|
|
- var box = tempalte;
|
|
|
- box = replaceAll("{filepath}",folderPath,box);
|
|
|
- box = replaceAll("{id}",i + 1,box);
|
|
|
-
|
|
|
- if (folderName == ".cache" || folderName == ".trash"){
|
|
|
- //Hidden folders
|
|
|
- continue;
|
|
|
- }
|
|
|
- if (folderPath.includes("/media")){
|
|
|
- //This is from external storage devices. List its number as well.
|
|
|
- var tmp = folderPath.split("/");
|
|
|
- var extStoragePath = "/" + tmp[1] + "/" + tmp[2];
|
|
|
- box = replaceAll("{foldername}",folderName + " ( " + extStoragePath + " )",box);
|
|
|
- }else{
|
|
|
- //This is directory inside normal folders.
|
|
|
- box = replaceAll("{foldername}",folderName,box);
|
|
|
- }
|
|
|
-
|
|
|
- /*
|
|
|
- var fileinfo = fileinfo = "[No playable files]";
|
|
|
- console.log(folderCount, fileCount);
|
|
|
- if (folderCount > 0 && fileCount > 0){
|
|
|
- fileinfo = "[" + fileCount + " files, " + folderCount +" folders]"
|
|
|
- }else if (fileCount > 0){
|
|
|
- fileinfo = "[" + fileCount + " files]";
|
|
|
- }else if (fileCount == -1 && folderCount == -1){
|
|
|
- fileinfo = "";
|
|
|
- }
|
|
|
- */
|
|
|
- var fileinfo = folderPath;
|
|
|
-
|
|
|
-
|
|
|
- box = replaceAll("{fileinfo}",fileinfo,box);
|
|
|
- rootPaths.push(folderPath);
|
|
|
- $("#mainList").append(box);
|
|
|
- }
|
|
|
- $("#interfaceDetails").text("[" + data.length + " folders]");
|
|
|
- });
|
|
|
- hideLeftMenu();
|
|
|
- }
|
|
|
-
|
|
|
- function isMobile(){
|
|
|
- if( /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) ) {
|
|
|
- return true
|
|
|
- }
|
|
|
- return false
|
|
|
- }
|
|
|
-
|
|
|
- function setWindowHash(hashValue){
|
|
|
- hashValue = JSON.stringify(hashValue);
|
|
|
-
|
|
|
- if (!isAndroid){
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- if(history.pushState) {
|
|
|
- window.history.pushState(null, null, '#' + hashValue);
|
|
|
- }else {
|
|
|
- location.hash = '#' + hashValue;
|
|
|
- }
|
|
|
- if(ao_module_virtualDesktop && !isMobile()){
|
|
|
- //Update the iframe src as well
|
|
|
- var newsrc = window.frameElement.getAttribute("src");
|
|
|
- if (newsrc.includes("#")){
|
|
|
- newsrc = newsrc.split("#")
|
|
|
- newsrc.pop();
|
|
|
- newsrc = newsrc.join("#");
|
|
|
- }
|
|
|
- newsrc = newsrc + "#" + hashValue;
|
|
|
- $(window.frameElement).attr("src",newsrc);
|
|
|
- //console.log(window.frameElement.getAttribute("src"));
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function updateStateReferingURL(){
|
|
|
- var paused = audioElement[0].paused; //Check the current playing state of the player
|
|
|
- setWindowHash({path: currentPath, pfp:playingFileDetail,pause:paused});
|
|
|
- }
|
|
|
-
|
|
|
- function openFolder(object, playParameters=null){
|
|
|
- var playIDAfterOpen = -1;
|
|
|
- var startPaused = false;
|
|
|
- if (playParameters !== null){
|
|
|
- playIDAfterOpen = playParameters.playIDAfterOpen;
|
|
|
- startPaused = playParameters.startPaused;
|
|
|
- }
|
|
|
- var backbtnTemplate = '<div class="mainList item extrapadding" filepath={filepath} id={id} onClick="openFolder(this);">\
|
|
|
- <div class="ui header selectable" style="margin:0px !important;">\
|
|
|
- <i class="angle left icon whiteFont" style="overflow:hidden;"></i>\
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">\
|
|
|
- ../\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- </div>';
|
|
|
-
|
|
|
- var folderTemplate = '<div class="mainList item" filepath={filepath} tag="folder" id={id} onClick="openFolder(this);">\
|
|
|
- <div class="ui header selectable" style="margin:0px !important;">\
|
|
|
- <img class="ui small image" src="img/fo.svg" style="margin-right: 0.2em;"></img>\
|
|
|
- <div class="content whiteFont" style="padding-top:2px;line-height:1em; width:80%; font-weight: lighter;">\
|
|
|
- {foldername}\
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">{fileinfo}</div>\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- <div class="mainList rightFunctionBar" type="folder" align="center">\
|
|
|
- <i class="chevron right icon" style="margin-top:1.2em;"></i>\
|
|
|
- </div>\
|
|
|
- </div>';
|
|
|
- var fileTemplate = `<div class="mainList file item" filepath={filepath} id={id} rawname={rawname}>\
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">\
|
|
|
- <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
- <!-- <i class="music icon whiteFont" style="overflow:hidden;"></i> -->\
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">\
|
|
|
- {songtitle}\
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
|
|
|
- </div>\
|
|
|
- </div>\
|
|
|
- <div class="topRightCorner" align="center">\
|
|
|
- {id}\
|
|
|
- </div>\
|
|
|
- <div class="mainList rightFunctionBar" type="file" align="center">\
|
|
|
- <i class="ellipsis vertical icon" style="margin-top:1.2em;"></i>\
|
|
|
- </div>\
|
|
|
- </div>`;
|
|
|
-
|
|
|
- var targetPath = decodeURIComponent($(object).attr("filepath"));
|
|
|
- if (targetPath == "root" || targetPath == "/"){
|
|
|
- //User request to go back to storage root
|
|
|
- loadFolderView();
|
|
|
- return;
|
|
|
- }
|
|
|
- currentPath = targetPath.split("/./").join("/");
|
|
|
- currentPath = currentPath.split("//").join("/");
|
|
|
- $("#interfaceTitle").text(ao_module_codec.decodeHexFoldername(targetPath.substring(0,targetPath.length -1).split("/").pop()));
|
|
|
- var parentDir = currentPath.split("/");
|
|
|
- parentDir.pop();parentDir.pop();
|
|
|
- parentDir = (parentDir.join("/") + "/");
|
|
|
- //console.log(currentPath);
|
|
|
- var backbtn;
|
|
|
- if (rootPaths.includes(currentPath) == false){
|
|
|
- //We are currently not at root
|
|
|
- backbtn = backbtnTemplate;
|
|
|
- backbtn = replaceAll("{filepath}",encodeURIComponent(parentDir),backbtn);
|
|
|
- backbtn = replaceAll("{id}",0,backbtn);
|
|
|
- }else{
|
|
|
- //Create a back button to go back to root of storage
|
|
|
- backbtn = backbtnTemplate;
|
|
|
- backbtn = replaceAll("{filepath}","root",backbtn);
|
|
|
- backbtn = replaceAll("{id}",0,backbtn);
|
|
|
- }
|
|
|
-
|
|
|
- //Request the server for a list of folders and file in this directory
|
|
|
- ao_module_agirun("Music/functions/listSong.js", {
|
|
|
- listdir: targetPath.split("//").join("/"),
|
|
|
- },function(data){
|
|
|
- $("#mainList").html("");
|
|
|
- $("#mainList").append(backbtn);
|
|
|
- if (Array.isArray(data)){
|
|
|
- var folders = data[0];
|
|
|
- var files = data[1];
|
|
|
- //List all folders
|
|
|
- for (var i =0; i < folders.length; i++){
|
|
|
- var folderName = folders[i][0];
|
|
|
- var folderPath = folders[i][1];
|
|
|
- var fileCount = folders[i][2];
|
|
|
- var folderCount = folders[i][3];
|
|
|
- var box = folderTemplate;
|
|
|
- if (folderName == ".cache" || folderName == ".trash"){
|
|
|
- //Hidden folders
|
|
|
- continue;
|
|
|
- }
|
|
|
- box = replaceAll("{filepath}",encodeURIComponent(folderPath),box);
|
|
|
- box = replaceAll("{id}","folder-" + i + 1,box);
|
|
|
- box = replaceAll("{foldername}",ao_module_codec.decodeHexFoldername(folderName),box);
|
|
|
- var fileinfo = "[" + fileCount + " files]"
|
|
|
- if (folderCount > 0){
|
|
|
- fileinfo = "[" + fileCount + " files, " + folderCount +" folders]"
|
|
|
- }else if (folderCount == -1){
|
|
|
- fileinfo = folderPath;
|
|
|
- }
|
|
|
- box = replaceAll("{fileinfo}",fileinfo,box);
|
|
|
- $("#mainList").append(box);
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- //List all files
|
|
|
- for ( var i =0; i < files.length; i++){
|
|
|
- var songInfo = files[i];
|
|
|
- var path = songInfo[0];
|
|
|
- var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
|
|
|
- var ext = songInfo[2];
|
|
|
- var size = songInfo[3];
|
|
|
- var box = fileTemplate;
|
|
|
- box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
|
|
|
- box = replaceAll("{id}",i + 1,box); //User count from 1
|
|
|
- box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
|
|
|
- box = replaceAll("{songtitle}",displayname,box);
|
|
|
- box = replaceAll("{ext}",ext,box);
|
|
|
- box = replaceAll("{size}",size,box);
|
|
|
- $("#mainList").append(box);
|
|
|
- }
|
|
|
- displayList = files;
|
|
|
- totalMusicCount = files.length;
|
|
|
- $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
-
|
|
|
- }else{
|
|
|
- //Something went wrong
|
|
|
- //console.log('[AirMusic] Something went wrong: ' + data);
|
|
|
- loadFolderView();
|
|
|
- return;
|
|
|
- }
|
|
|
- if (folders.length > 0 && files.length > 0){
|
|
|
- $("#interfaceDetails").text("[" + files.length + " files, " + folders.length+ " folders]");
|
|
|
- }else if (folders.length > 0){
|
|
|
- $("#interfaceDetails").text("[" + folders.length + " folders]");
|
|
|
- }else if (files.length > 0){
|
|
|
- $("#interfaceDetails").text("[" + files.length + " files]");
|
|
|
- }else{
|
|
|
- $("#interfaceDetails").text("[0 files or folders]");
|
|
|
- }
|
|
|
- //Hook all the events for the moreinfo on folders
|
|
|
- hookMoreFolderInfoClickEvent();
|
|
|
-
|
|
|
- //Check if the current playing file is located inside this list.
|
|
|
- highLightPlayingMusic();
|
|
|
-
|
|
|
- //Check if auto playback is required. If yes, play it with the given filepath.
|
|
|
- if (playIDAfterOpen != -1){
|
|
|
- $(".mainList.item").each(function(){
|
|
|
- if ($(this).attr("id") == playIDAfterOpen){
|
|
|
- //This is the file that require to playback after the folder loaded
|
|
|
- playSong($(this).find(".ts.header.selectable"),startPaused);
|
|
|
- setTimeout(resizeQuickAdjust,500);
|
|
|
- }
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- //Load thumbnail for song in list
|
|
|
- loadThumbnailToMusicList();
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function moreInfo(object){
|
|
|
- if ($(object).parent().attr('tag') == "folder"){
|
|
|
- //This is a folder. Ask if play as playlist
|
|
|
- alert("Folder playback work in progress");
|
|
|
- }else{
|
|
|
- showMore(object);
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function hookMoreFolderInfoClickEvent(){
|
|
|
- $(".mainList.rightFunctionBar").on("click", function(e){
|
|
|
- e.stopPropagation();
|
|
|
- moreInfo(this);
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function highLightPlayingMusic(){
|
|
|
- //This function is used for highlighting the current playing music if the music piece is found inside the current list
|
|
|
- $(".mainList.item").removeClass("playingTrack");
|
|
|
- $(".mainList.item").each(function(){
|
|
|
- var id = parseInt($(this).attr("id"));
|
|
|
- try{
|
|
|
- var filepath = ao_module_utils.attrToObject($(this).attr("filepath"));
|
|
|
- }catch{
|
|
|
- //Use back the previous method of filepath storage for compatibility
|
|
|
- var filepath = $(this).attr("filepath");
|
|
|
- }
|
|
|
-
|
|
|
- //Id is ignored in the 9/9/2019 updates and only check if the path matches.
|
|
|
- if (filepath == playingFileDetail[1]){
|
|
|
- $(this).addClass("playingTrack");
|
|
|
- }
|
|
|
- //console.log([id,filepath], playingFileDetail);
|
|
|
- });
|
|
|
-
|
|
|
- //Update dropdownList as well if exists
|
|
|
- $(".dropdownList.item").removeClass("playingTrack");
|
|
|
- $(".dropdownList.item").each(function(){
|
|
|
- try{
|
|
|
- var filepath = ao_module_utils.attrToObject($(this).attr("filepath"));
|
|
|
- }catch{
|
|
|
- //Use back the previous method of filepath storage for compatibility
|
|
|
- var filepath = $(this).attr("filepath");
|
|
|
- }
|
|
|
- //var filepath = $(this).attr("filepath");
|
|
|
- if (filepath == playingFileDetail[1]){
|
|
|
- $(this).addClass("playingTrack");
|
|
|
- }
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function hideShowMoreMenu(){
|
|
|
- $(".showMoreMenus").fadeOut('fast');
|
|
|
- $("#showmoreUIcover").fadeOut('fast');
|
|
|
-
|
|
|
- //Return the playlistAddPendingFile object back to the playing one
|
|
|
- if (playingFileDetail !== undefined){
|
|
|
- playlistAddPendingFile = playingFileDetail[1];
|
|
|
- }else{
|
|
|
- //No song is being play back
|
|
|
- playlistAddPendingFile = "";
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function playFromShowMoreMenu(){
|
|
|
- nextSong(showMoreOprPlayID - 1);
|
|
|
- $(".showMoreMenus").fadeOut('fast');
|
|
|
- }
|
|
|
-
|
|
|
- function showFileInfo(){
|
|
|
- $("#showFileInfo").show();
|
|
|
- $("#showMoreUI").hide();
|
|
|
- }
|
|
|
-
|
|
|
- function startRelatedSearch(){
|
|
|
- enterSearchMode();
|
|
|
- $("#searchInputBar").val(showMoreoprDisplayName);
|
|
|
- $("#searchInputBar").focus();
|
|
|
- hideShowMoreMenu();
|
|
|
- }
|
|
|
-
|
|
|
- function searchYoutubeViaShowMore(){
|
|
|
- if (showMoreoprDisplayName != "" || showMoreoprDisplayName != undefined){
|
|
|
- var filename = showMoreoprDisplayName;
|
|
|
- searchOnYoutube(filename);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- var showMoreOprPlayID,showMoreOprFilepath, showMoreoprDisplayName;
|
|
|
- function showMore(object){
|
|
|
- var filepath = $(object).parent().attr("filepath");
|
|
|
- filepath = JSON.parse(decodeURIComponent(filepath));
|
|
|
- //User want more action on this file. assume this is the file to add
|
|
|
- playlistAddPendingFile = filepath;
|
|
|
- var id = $(object).parent().attr('id');
|
|
|
- var rawname = $(object).parent().attr("rawname");
|
|
|
- var displayName = JSON.parse(decodeURIComponent(ao_module_codec.decodeUmFilename(rawname)));
|
|
|
- //Update global variable for quick operations
|
|
|
- showMoreOprPlayID = id;
|
|
|
- showMoreOprFilepath = filepath;
|
|
|
- showMoreoprDisplayName = displayName;
|
|
|
- $("#showMoreUI").find(".songTitle").text(displayName);
|
|
|
- $("#showMoreUI").find(".songID").text(id);
|
|
|
- $("#showFileInfo").find(".songTitle").text(displayName);
|
|
|
- $("#showMoreUI").fadeIn('fast');
|
|
|
- $("#showmoreUIcover").fadeIn('fast');
|
|
|
- //Update fileinformation as well
|
|
|
- if (filepath.substring(0, 12) == "/media?file="){
|
|
|
- filepath = filepath.substring(12);
|
|
|
- }
|
|
|
-
|
|
|
- //Pre-render the file info
|
|
|
- ao_module_agirun("Music/functions/getFileInfo.js", {
|
|
|
- filepath: filepath,
|
|
|
- }, function(data){
|
|
|
- $("#showFileInfo").find(".filename").text(ao_module_codec.decodeUmFilename(data[0]));
|
|
|
- $("#showFileInfo").find(".rawname").text(data[0]);
|
|
|
- $("#showFileInfo").find(".filepath").text(data[1]);
|
|
|
- $("#showFileInfo").find(".filesize").text(data[2] + " (" + data[3] + " Bytes)");
|
|
|
- $("#showFileInfo").find(".filedate").text(data[4]);
|
|
|
- });
|
|
|
-
|
|
|
- if (currentMode != "playlist"){
|
|
|
- $(".playlistonly").hide();
|
|
|
- }else{
|
|
|
- $(".playlistonly").show();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function playSongFromDropdownList(object){
|
|
|
- //Play song from dropdown list. Hence, no need to update dropdownList
|
|
|
- if ($(object).parent().hasClass("playingTrack")){
|
|
|
- //This song already playing. Ignore play request.
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- if (pagingEnableForPlayingList && $(object).attr("targetPage") != currentPlayingPage){
|
|
|
- let newPage = parseInt($(object).attr("targetPage"));
|
|
|
- if (!isNaN(newPage)){
|
|
|
- currentPlayingPage = newPage;
|
|
|
- }
|
|
|
- }
|
|
|
- playSong(object);
|
|
|
- //Check if the song is also in main list. If yes, highlight it as well
|
|
|
- /*
|
|
|
- $(".mainList.item").each(function(){
|
|
|
- if ($(this).attr("filepath") == $(object).parent().attr("filepath")){
|
|
|
- $(".mainList.item.playingTrack").removeClass("playingTrack");
|
|
|
- $(this).addClass("playingTrack");
|
|
|
- }
|
|
|
- });
|
|
|
- */
|
|
|
- $("#dropdownSonglist").delay(500).slideUp();
|
|
|
- highLightPlayingMusic();
|
|
|
- }
|
|
|
-
|
|
|
- //Move the playing song list into the dropdown song list. This list might not be the same as the one in the main list.
|
|
|
- function parsePlayingSongList(){
|
|
|
- $("#currentPlayingMainList").html("");
|
|
|
- var counter = playingList.length;
|
|
|
- var totalSize = 0.0;
|
|
|
- var renderRange = [0, playingList.length];
|
|
|
- var pageKeepRange = [0, playingList.length]
|
|
|
- if (pagingEnableForPlayingList){
|
|
|
- //Paging Enabled. Only show 1/2 max + current page song + 1/2 max
|
|
|
- var startEndRange = getPageStartAndEndByPageNumber(currentPlayingPage, playingList);
|
|
|
- startEndRange = [startEndRange[0] - pagingCutoffCount /2, startEndRange[1] + pagingCutoffCount / 2]
|
|
|
- //console.log("DEFAULT START END RANGE", startEndRange)
|
|
|
- //page keep range is the middle 1/2 number of songs, [1/4 => prev page][1/2][1/4 => next page]
|
|
|
- pageKeepRange[0] = startEndRange[0] + pagingCutoffCount / 2;
|
|
|
- pageKeepRange[1] = startEndRange[1] - pagingCutoffCount / 2;
|
|
|
-
|
|
|
- if (startEndRange[0] < 0){
|
|
|
- startEndRange[0] = 0;
|
|
|
- }
|
|
|
-
|
|
|
- if (startEndRange[1] > playingList.length){
|
|
|
- startEndRange[1] = playingList.length;
|
|
|
- }
|
|
|
-
|
|
|
- renderRange = startEndRange;
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
- if (pageKeepRange[0] < 0){
|
|
|
- pageKeepRange[0] = 0;
|
|
|
- }
|
|
|
- if (pageKeepRange[1] > playingList.length){
|
|
|
- pageKeepRange[1] = playingList.length;
|
|
|
- }
|
|
|
- //console.log("KEEP RANGE", pageKeepRange);
|
|
|
- }
|
|
|
-
|
|
|
- for (var i = renderRange[0]; i < renderRange[1]; i++){
|
|
|
- var displayname = ao_module_codec.decodeUmFilename(playingList[i][1]);
|
|
|
- if (playingList[i][3].includes("MB")){
|
|
|
- totalSize += parseFloat(playingList[i][3].split(" ")[0]);
|
|
|
- }else if (playingList[i][3].includes("KB")){
|
|
|
- totalSize += parseFloat(playingList[i][3].split(" ")[0]) / 1000;
|
|
|
- }else if (playingList[i][3].includes("GB")){
|
|
|
- totalSize += parseFloat(playingList[i][3].split(" ")[0]) * 1000;
|
|
|
- }
|
|
|
-
|
|
|
- //Get thumbnail from the current list to prevent re-loading form remote
|
|
|
- let targetThumbnail = $("#" + (i + 1)).find("img").attr("src");
|
|
|
- let opacityShowing = "";
|
|
|
- let realVpath = playingList[i][0].split("=");
|
|
|
- realVpath.shift();
|
|
|
- realVpath = realVpath.join("=");
|
|
|
-
|
|
|
- if ($("#" + (i + 1)).length == 0 && displayList.length == playingList.length){
|
|
|
- //Item not exists
|
|
|
- targetThumbnail = "img/eq.svg";
|
|
|
- opacityShowing = "opacity: 0.6;"
|
|
|
- }else if (pagingEnableForPlayingList){
|
|
|
- if (i < pageKeepRange[0] || i > pageKeepRange[1]){
|
|
|
- //Out of paging range. Do hidden display
|
|
|
- targetThumbnail = "img/eq.svg";
|
|
|
- opacityShowing = "opacity: 0.6;"
|
|
|
- }else{
|
|
|
- targetThumbnail = `../system/file_system/loadThumbnail?bytes=true&vpath=${encodeURIComponent(realVpath)}`;
|
|
|
- }
|
|
|
- }else{
|
|
|
- //Non paging list. Always do all thumbnails.
|
|
|
- targetThumbnail = `../system/file_system/loadThumbnail?bytes=true&vpath=${encodeURIComponent(realVpath)}`;
|
|
|
- }
|
|
|
-
|
|
|
- let targetPage = currentPlayingPage;
|
|
|
- if (i < pageKeepRange[0]){
|
|
|
- targetPage -= 1;
|
|
|
- }
|
|
|
- if (i > pageKeepRange[1]){
|
|
|
- targetPage += 1;
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
- $("#currentPlayingMainList").append(`<div class="dropdownList file item" style="${opacityShowing}" filepath="${ao_module_utils.objectToAttr(playingList[i][0])}" listid="${i + 1}" rawname="${ao_module_utils.objectToAttr(playingList[i][1])}">
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" onClick="playSongFromDropdownList(this);" targetPage="${targetPage}">
|
|
|
- <img class="ui small image" src="${targetThumbnail}" onError="replaceImageToDefault(this);" style="margin-right: 0.2em;"></img>
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
|
|
|
- <span style="font-weight: lighter;">${displayname}</span>
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">${playingList[i][2]} / ${playingList[i][3]}</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="topRightCorner" align="center">
|
|
|
- ${i + 1}
|
|
|
- </div>
|
|
|
- </div>`);
|
|
|
- $("#dropdownListSongCount").text(counter);
|
|
|
- $("#dropdownListIDCount").text(parseInt(totalSize) + "");
|
|
|
- }
|
|
|
- $(".dropdownList.item").each(function(){
|
|
|
- if ($(this).attr('filepath') == playingFileDetail[1]){
|
|
|
- $(this).addClass("playingTrack");
|
|
|
- }
|
|
|
- });
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function replaceImageToDefault(target){
|
|
|
- $(target).attr('src', "img/eq.svg");
|
|
|
- }
|
|
|
-
|
|
|
- function loadSongList(type = "all"){
|
|
|
- currentPath = "";
|
|
|
- currentMode = "music";
|
|
|
-
|
|
|
- $("#interfaceTitle").text("Music");
|
|
|
- $("#AMmenuIcon").attr("class","music icon large whiteFont")
|
|
|
-
|
|
|
- ao_module_agirun("Music/functions/listSong.js", {
|
|
|
- listSong: type,
|
|
|
- },function(data){
|
|
|
- //Initialize the song list
|
|
|
- displayList = data;
|
|
|
- if (type == "all"){
|
|
|
- //Caching is used in here
|
|
|
- displayList = data.list;
|
|
|
- if (data.cached == true){
|
|
|
- //This is a cached version of the music list.
|
|
|
- console.log("Updating cached song list");
|
|
|
- ao_module_agirun("Music/functions/buildCache.js", {}, function(data){
|
|
|
- console.log("Cache updated: ", data);
|
|
|
- });
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- //Updates 2022-07-12: Check if the list is too long. If yes, use paging
|
|
|
- let renderRange = [0,displayList.length];
|
|
|
-
|
|
|
- if (displayList.length > pagingCutoffCount){
|
|
|
- togglePagingMode(true);
|
|
|
- renderRange = [0, pagingCutoffCount];
|
|
|
- }else{
|
|
|
- togglePagingMode(false);
|
|
|
- }
|
|
|
-
|
|
|
- if (playingList == []){
|
|
|
- playingList = Array.from(displayList);
|
|
|
- pagingEnableForPlayingList = pagingEnabled;
|
|
|
- currentPlayingPage = currentPage;
|
|
|
- }
|
|
|
-
|
|
|
- renderDisplayList(displayList,renderRange[0], renderRange[1] );
|
|
|
-
|
|
|
- }, function(){
|
|
|
- alert("Failed to access listSong API with type: " + type)
|
|
|
- });
|
|
|
- hideLeftMenu();
|
|
|
- }
|
|
|
-
|
|
|
- function renderDisplayList(displayList, start, end, callback=undefined){
|
|
|
- $("#mainList").html("");
|
|
|
- for ( var i =start; i < end; i++){
|
|
|
- var songInfo = displayList[i];
|
|
|
- var path = songInfo[0];
|
|
|
- var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
|
|
|
- var ext = songInfo[2];
|
|
|
- var size = songInfo[3];
|
|
|
- $("#mainList").append(`<div class="mainList file item" filepath="${ao_module_utils.objectToAttr(path)}" id="${i + 1}" rawname="${ao_module_utils.objectToAttr(songInfo[1])}">
|
|
|
- <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">
|
|
|
- <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
- <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">
|
|
|
- ${displayname}
|
|
|
- <div class="sub header fileinfo" style="color: #c7c7c7;">${ext} / ${size}</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="topRightCorner" align="center">
|
|
|
- ${i + 1}
|
|
|
- </div>
|
|
|
- <div class="mainList rightFunctionBar" align="center" onClick="showMore(this);">
|
|
|
- <i class="ellipsis vertical icon" style="margin-top:1.2em;"></i>\
|
|
|
- </div>
|
|
|
- </div>`);
|
|
|
- }
|
|
|
-
|
|
|
- if (pagingEnabled){
|
|
|
- //Append the page switch buttons
|
|
|
- let numberOfPages = Math.ceil(parseFloat(displayList.length) / parseFloat(pagingCutoffCount));
|
|
|
- let pageSelector = ``;
|
|
|
- for (var i = 0; i < numberOfPages; i++){
|
|
|
- let disabled = "";
|
|
|
- if (i == currentPage){
|
|
|
- disabled = "disabled";
|
|
|
- }
|
|
|
- pageSelector = pageSelector + `<button id="pagebtn_${i}" onclick="switchToPage(${i});" class="pagebtn noborderbtn inverted basic ui icon button ${disabled}" style="box-shadow: none !important; -webkit-box-shadow: none !important;">
|
|
|
- ${i + 1}
|
|
|
- </button>`;
|
|
|
- }
|
|
|
- $("#mainList").append(`<div class="mainList item" style="cursor: unset;">
|
|
|
- <div>${pageSelector}</div>
|
|
|
- </div>`);
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- totalMusicCount = displayList.length;
|
|
|
- $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
- $("#interfaceDetails").text("[" + totalMusicCount + " songs]");
|
|
|
- highLightPlayingMusic();
|
|
|
-
|
|
|
-
|
|
|
- //Load thumbnail for song in list
|
|
|
- loadThumbnailToMusicList();
|
|
|
-
|
|
|
- if (callback != undefined){
|
|
|
- callback();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function switchToPage(pageNumber, callback=undefined){
|
|
|
- if (!pagingEnabled){
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- let thisCallback = callback;
|
|
|
- if (pageNumber == currentPage){
|
|
|
- //Already in that page
|
|
|
- if (thisCallback != undefined){
|
|
|
- callback();
|
|
|
- }
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- currentPage = pageNumber;
|
|
|
-
|
|
|
- let startPage = pageNumber * pagingCutoffCount;
|
|
|
- let endPage = startPage + pagingCutoffCount;
|
|
|
- if (endPage > displayList.length){
|
|
|
- endPage = displayList.length;
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- if (thisCallback == undefined){
|
|
|
- renderDisplayList(displayList, startPage, endPage, function(){
|
|
|
- window.scrollTo(0, document.body.scrollHeight);
|
|
|
- });
|
|
|
- }else{
|
|
|
- renderDisplayList(displayList, startPage, endPage,thisCallback);
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function getPageNumberByPlaybackId(id){
|
|
|
- let numberOfPages = Math.ceil(parseFloat(displayList.length) / parseFloat(pagingCutoffCount));
|
|
|
- let targetPageNo = Math.ceil(id / pagingCutoffCount) - 1;
|
|
|
- return targetPageNo;
|
|
|
- }
|
|
|
-
|
|
|
- function getPageStartAndEndByPageNumber(pageNumber, targetList){
|
|
|
- let startPage = pageNumber * pagingCutoffCount;
|
|
|
- let endPage = startPage + pagingCutoffCount;
|
|
|
- if (endPage > targetList.length){
|
|
|
- endPage = targetList.length;
|
|
|
- }
|
|
|
-
|
|
|
- return [startPage, endPage];
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- function toggleLeftMenu(){
|
|
|
- if (leftMenuShown){
|
|
|
- $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 120,function(){
|
|
|
- $("#leftSideBar").hide();
|
|
|
- });
|
|
|
- $("#sideBarCover").fadeOut();
|
|
|
- leftMenuShown = false;
|
|
|
- }else{
|
|
|
- $("#leftSideBar").show();
|
|
|
- $("#leftSideBar").animate({left:0}, 300);
|
|
|
- $("#sideBarCover").fadeIn();
|
|
|
- leftMenuShown = true;
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
- function hideLeftMenu(){
|
|
|
- if (leftMenuShown){
|
|
|
- $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 120,function(){
|
|
|
- $("#leftSideBar").hide();
|
|
|
- });
|
|
|
- $("#sideBarCover").fadeOut();
|
|
|
- leftMenuShown = false;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- function showFileInformation(){
|
|
|
- ao_module_agirun("Music/functions/getFileInfo.js", {
|
|
|
- filepath: playingFileDetail[1],
|
|
|
- },function(data){
|
|
|
- //console.log(data);
|
|
|
- $("#settingInterface").hide();
|
|
|
- $("#filepropInterface").show();
|
|
|
- $("#filepropInterface").find(".filename").text(ao_module_codec.decodeUmFilename(data[0]));
|
|
|
- $("#filepropInterface").find(".rawname").text(data[0]);
|
|
|
- $("#filepropInterface").find(".filepath").text(data[1]);
|
|
|
- $("#filepropInterface").find(".filesize").text(data[2] + " (" + data[3] + " Bytes)");
|
|
|
- $("#filepropInterface").find(".filedate").text(data[4]);
|
|
|
- });
|
|
|
- }
|
|
|
-
|
|
|
- function resizeQuickAdjust(){
|
|
|
- //Resize the position of the leftMenu
|
|
|
- if (!leftMenuShown){
|
|
|
- $("#leftSideBar").css("left", $("#leftSideBar").width() * -1);
|
|
|
- }
|
|
|
- if ($("#playerInterface").offset().left != 0){
|
|
|
- $("#playerInterface").css("left",window.innerWidth);
|
|
|
- }
|
|
|
-
|
|
|
- $("#albumnArtImage").css("max-height",window.innerHeight - 255);
|
|
|
-
|
|
|
- setTimeout(function(){
|
|
|
- $("#albumnArt").css({
|
|
|
- "height": window.innerHeight - 255,
|
|
|
- "top": (window.innerHeight / 2 - $("#albumnArtImage").height()/2)
|
|
|
- });
|
|
|
- },50);
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
- //var imageTop = (window.innerHeight - 255 - $("#albumnArtImage").height()) / 2;
|
|
|
- //$("#albumnArtImage").css("top",imageTop + "px");
|
|
|
- }
|
|
|
-
|
|
|
- function setStorage(configName,configValue){
|
|
|
- ao_module_storage.setStorage("AirMusic",configName,configValue);
|
|
|
- /*
|
|
|
- $.ajax({
|
|
|
- type: 'GET',
|
|
|
- url: "../system/file_system/preference",
|
|
|
- data: {key: "AirMusic/" + configName,value:configValue},
|
|
|
- success: function(data){},
|
|
|
- async:true
|
|
|
- });
|
|
|
- */
|
|
|
- return true;
|
|
|
- }
|
|
|
-
|
|
|
- function loadStorage(configName){
|
|
|
- /*
|
|
|
- var result = "";
|
|
|
- $.ajax({
|
|
|
- type: 'GET',
|
|
|
- url: "../system/file_system/preference",
|
|
|
- data: {key: "AirMusic/" + configName},
|
|
|
- success: function(data){
|
|
|
- if (data.error !== undefined){
|
|
|
- result = "";
|
|
|
- }else{
|
|
|
- result = data;
|
|
|
- }
|
|
|
- },
|
|
|
- error: function(data){result = "";},
|
|
|
- async:false,
|
|
|
- timeout: 3000
|
|
|
- });
|
|
|
- return result;
|
|
|
- */
|
|
|
-
|
|
|
- return ao_module_storage.loadStorage("AirMusic",configName);
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- //Wipe controller for mobile users
|
|
|
- $("#albumnArt")[0].addEventListener('touchstart', handleTouchStart, false);
|
|
|
- $("#albumnArt")[0].addEventListener('touchmove', handleTouchMove, false);
|
|
|
-
|
|
|
- var xDown = null;
|
|
|
- var yDown = null;
|
|
|
-
|
|
|
- function getTouches(evt) {
|
|
|
- return evt.touches || // browser API
|
|
|
- evt.originalEvent.touches; // jQuery
|
|
|
- }
|
|
|
-
|
|
|
- function handleTouchStart(evt) {
|
|
|
- const firstTouch = getTouches(evt)[0];
|
|
|
- xDown = firstTouch.clientX;
|
|
|
- yDown = firstTouch.clientY;
|
|
|
- };
|
|
|
-
|
|
|
- function handleTouchMove(evt) {
|
|
|
- if ( ! xDown || ! yDown ) {
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
- var xUp = evt.touches[0].clientX;
|
|
|
- var yUp = evt.touches[0].clientY;
|
|
|
-
|
|
|
- var xDiff = xDown - xUp;
|
|
|
- var yDiff = yDown - yUp;
|
|
|
-
|
|
|
- if ( Math.abs( xDiff ) > Math.abs( yDiff ) ) {/*most significant*/
|
|
|
- if ( xDiff > 0 ) {
|
|
|
- /* left swipe */
|
|
|
- //Going to the next song
|
|
|
- nextSong();
|
|
|
- } else {
|
|
|
- /* right swipe */
|
|
|
- //Going back one song
|
|
|
- previousSong();
|
|
|
- }
|
|
|
- } else {
|
|
|
- if ( yDiff > 0 ) {
|
|
|
- /* up swipe */
|
|
|
- } else {
|
|
|
- /* down swipe */
|
|
|
- }
|
|
|
- }
|
|
|
- /* reset values */
|
|
|
- xDown = null;
|
|
|
- yDown = null;
|
|
|
- };
|
|
|
-
|
|
|
- //Handle audio progress to time conversion
|
|
|
- function secondsToHms(d) {
|
|
|
- d = Number(d);
|
|
|
- var h = Math.floor(d / 3600);
|
|
|
- var m = Math.floor(d % 3600 / 60);
|
|
|
- var s = Math.floor(d % 3600 % 60);
|
|
|
-
|
|
|
- if (h > 0 && h < 10){
|
|
|
- dh = "0" + h + ":";
|
|
|
- }else if (h == 0){
|
|
|
- dh = "";
|
|
|
- }else{
|
|
|
- dh = h + ":";
|
|
|
- }
|
|
|
-
|
|
|
- if (m > 0 && m < 10){
|
|
|
- dm = "0" + m + ":";
|
|
|
- }else if (m == 0){
|
|
|
- if (h > 0){
|
|
|
- dm = "00:";
|
|
|
- }else{
|
|
|
- dm = "0:";
|
|
|
- }
|
|
|
- }else{
|
|
|
- dm = m + ":";
|
|
|
- }
|
|
|
-
|
|
|
- if (s > 0 && s < 10){
|
|
|
- ds = "0" + s;
|
|
|
- }else if (s == 0){
|
|
|
- if (m > 0 || h > 0){
|
|
|
- ds = "00";
|
|
|
- }else{
|
|
|
- ds = "00";
|
|
|
- }
|
|
|
- }else{
|
|
|
- ds = s;
|
|
|
- }
|
|
|
-
|
|
|
- return dh + dm + ds;
|
|
|
- }
|
|
|
-
|
|
|
- //Handle window resize events
|
|
|
- $(window).on("resize", function(){
|
|
|
- resizeQuickAdjust();
|
|
|
- });
|
|
|
-
|
|
|
- function AllQuickMenuHidden(){
|
|
|
- var result = true;
|
|
|
- $(".quickMenu").each(function(){
|
|
|
- if ( $(this).is(':visible') ){
|
|
|
- result = false;
|
|
|
- }
|
|
|
- });
|
|
|
- return result;
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- function AllSubMenuHidden(){
|
|
|
- var result = true;
|
|
|
- $(".showMoreMenus").each(function(){
|
|
|
- if ( $(this).is(':visible') ){
|
|
|
- result = false;
|
|
|
- }
|
|
|
- });
|
|
|
- return result;
|
|
|
- }
|
|
|
-
|
|
|
- //Handle back button press on mobile devices
|
|
|
- function handleBackButton() {
|
|
|
- if (!AllQuickMenuHidden()){
|
|
|
- hideAllquickMenu();
|
|
|
- window.history.pushState({}, '');
|
|
|
- }else if (mainPlayerShown()){
|
|
|
- //Close the main Menu
|
|
|
- hideMainPlayerInterface();
|
|
|
- window.history.pushState({}, '');
|
|
|
- }else if (!AllSubMenuHidden()){
|
|
|
- $(".showMoreMenus").fadeOut('fast');
|
|
|
- $("#showmoreUIcover").fadeOut('fast');
|
|
|
- window.history.pushState({}, '');
|
|
|
- }else{
|
|
|
- window.history.back();
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
- window.addEventListener('popstate', handleBackButton);
|
|
|
- window.history.pushState({}, '');
|
|
|
-</script>
|
|
|
-</body>
|
|
|
+<!DOCTYPE html>
|
|
|
+<html>
|
|
|
+<head>
|
|
|
+ <meta charset="UTF-8">
|
|
|
+ <meta name="apple-mobile-web-app-capable" content="yes" />
|
|
|
+ <meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1, maximum-scale=1"/>
|
|
|
+ <meta name="theme-color" content="#4b75ff">
|
|
|
+ <link rel="stylesheet" href="../script/semantic/semantic.min.css">
|
|
|
+ <link rel="stylesheet" href="./main.css">
|
|
|
+ <script src="../script/jquery.min.js"></script>
|
|
|
+ <script src="../script/ao_module.js"></script>
|
|
|
+ <script src="../script/semantic/semantic.min.js"></script>
|
|
|
+ <link rel="icon" type="image/png" href="img/favicon.png" />
|
|
|
+ <!-- Handle native playback on Chrome Andoird-->
|
|
|
+ <script src="native.js"></script>
|
|
|
+
|
|
|
+ <link rel="manifest" crossorigin="use-credentials" href="manifest.json">
|
|
|
+ <title>AirMusic</title>
|
|
|
+</head>
|
|
|
+<body>
|
|
|
+<div id="mainMenu" class="ui basic fluid AMmenu menu bottomBlue" style="z-index:80;position:fixed;position:top:0px;left:0px;width:100%;">
|
|
|
+ <button class="ui item icon noBorder AMmenu button" onClick="toggleLeftMenu();"><i class="content icon"></i></button>
|
|
|
+ <div class="item noBorder AMmenu" style="padding-right:0px;padding-top:12px;"><i id="AMmenuIcon" class="music icon large whiteFont"></i></div>
|
|
|
+ <div class="item noBorder AMmenu" style="padding-left:3px;padding-top:3px;padding-bottom:5px;min-width:50%;">
|
|
|
+ <div class="ui header whiteFont">
|
|
|
+ <div id="interfaceTitle" style="display:inline; font-weight: lighter;">Music</div>
|
|
|
+ <div id="interfaceDetails" class="sub header" style="font-size:80%; line-height:1em; color: white;">[0 songs]</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <button class="ui right item icon AMmenu noBorder button" onClick="enterSearchMode();"><i class="search icon"></i></button>
|
|
|
+ <!-- <button class="ui right item icon AMmenu noBorder button"><i class="ellipsis vertical icon"></i></button> -->
|
|
|
+</div>
|
|
|
+<div id="searchModeMenu" class="ui basic fluid AMmenu menu bottomBlue" style="z-index:85;position:fixed;top:0px;left:0px;width:100%;height:51px;display:none;margin-top: 0px;">
|
|
|
+ <button class="ui item icon noBorder AMmenu button" onClick="exitSearchMode();"><i class="arrow left icon"></i></button>
|
|
|
+ <div class="ui input" style="flex-grow: 1;">
|
|
|
+ <input id="searchInputBar" onkeypress="//handleSearchInputEnter(event);" 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">
|
|
|
+ </div>
|
|
|
+ <button class="ui right item icon AMmenu noBorder button" onClick="searchSong();"><i class="search icon"></i></button>
|
|
|
+</div>
|
|
|
+<!-- The main list that display the current information-->
|
|
|
+<div id="mainList">
|
|
|
+
|
|
|
+</div>
|
|
|
+
|
|
|
+<!-- Advance functions config menu-->
|
|
|
+<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>
|
|
|
+<div id="showMoreUI" class="showMoreMenus" style="display:none;">
|
|
|
+ <div class="ui large header whiteFont songTitle">N/A</div>
|
|
|
+ <div class="showMoreMenuItem" onClick="playFromShowMoreMenu();">Play</div>
|
|
|
+ <div class="showMoreMenuItem" onclick='addToPlaylistFromMoreMenu();'>Add to Playlist</div>
|
|
|
+ <div class="showMoreMenuItem playlistonly" onclick="removeFromPlylistFromMoreMenu();">Remove from current Playlist</div>
|
|
|
+ <div class="showMoreMenuItem" onClick="searchYoutubeViaShowMore();">Search on Youtube</div>
|
|
|
+ <div class="showMoreMenuItem">Edit Tag</div>
|
|
|
+ <div class="showMoreMenuItem">Share</div>
|
|
|
+ <div class="showMoreMenuItem" onClick="startRelatedSearch();">Related Search</div>
|
|
|
+ <div class="showMoreMenuItem" onClick="showFileInfo();">File Information</div>
|
|
|
+
|
|
|
+ <div class="topRightCorner songID" align="center" style="margin-top:10px !important;padding-top:3px !important;">-1</div>
|
|
|
+</div>
|
|
|
+<div id="showFileInfo" class="showMoreMenus" style="display:none;">
|
|
|
+ <div class="ui large header">
|
|
|
+ <span class="filename whiteFont">Title</span>
|
|
|
+ <div class="sub header rawname whiteFont" style="word-break: break-all !important; font-size:80%;">Storage Name</div>
|
|
|
+ </div>
|
|
|
+ <div class="ui list whiteFont">
|
|
|
+ <div class="item whiteFont">Storage Location: <span class="filepath" style="word-break:break-all;">N/A</span></div>
|
|
|
+ <div class="item whiteFont">File Size: <span class="filesize">N/A</span></div>
|
|
|
+ <div class="item whiteFont">Last Modification Date: <span class="filedate">N/A</span></div>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+
|
|
|
+<!-- Side bar -->
|
|
|
+<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>
|
|
|
+<div id="leftSideBar" class="leftsb" style="max-width:60%;height:100%;z-index:90;display:none;">
|
|
|
+ <div class="leftsbObject sidebarBanner">
|
|
|
+ <img class="ui tiny middle aligned image" src="img/main_icon.png" style="max-width: 50px;"><h5 class="whiteFont" style="display:inline;padding-left:8px;">AirMusic</h5>
|
|
|
+ </div>
|
|
|
+ <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadSongList();setStorage('viewingTab','music');">
|
|
|
+ <div class="ui header">
|
|
|
+ <i class="music tiny icon whiteFont"></i>
|
|
|
+ <div class="content whiteFont">
|
|
|
+ Music
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadFolderView();setStorage('viewingTab','folder');">
|
|
|
+ <div class="ui header">
|
|
|
+ <i class="folder tiny icon whiteFont"></i>
|
|
|
+ <div class="content whiteFont">
|
|
|
+ Folder
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadPlaylistView(); setStorage('viewingTab','playlist');">
|
|
|
+ <div class="ui header">
|
|
|
+ <i class="align justify tiny icon whiteFont"></i>
|
|
|
+ <div class="content whiteFont">
|
|
|
+ Playlists
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="loadNetworkView(); setStorage('viewingTab','network');">
|
|
|
+ <div class="ui header">
|
|
|
+ <i class="world tiny icon whiteFont"></i>
|
|
|
+ <div class="content whiteFont">
|
|
|
+ Network
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <!--
|
|
|
+ <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;" onClick="">
|
|
|
+ <div class="ui header">
|
|
|
+ <i class="upload icon whiteFont"></i>
|
|
|
+ <div class="content whiteFont">
|
|
|
+ Upload
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ -->
|
|
|
+ <div class="leftsbObject sbSelectable sbPaddingIn" style="font-size:80%;border-top: 2px solid #4a4a4a;">
|
|
|
+ <div class="ui header">
|
|
|
+ <i class="setting tiny icon whiteFont"></i>
|
|
|
+ <div class="content whiteFont">
|
|
|
+ Settings
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+
|
|
|
+<!-- Mini player controller put on the bottom side of the list interface-->
|
|
|
+<div id="miniPlayer" class="hidden">
|
|
|
+ <div style="padding-left:20px;padding-top:5px;" >
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;width:300px;cursor:pointer;" onClick="showMainPlayerInterface();">
|
|
|
+ <img id="smallPlayerThumb" class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;">
|
|
|
+ <div id="miniPlayerDisplayName" style="text-overflow: ellipsis;overflow: hidden;max-width:200px; white-space: nowrap;">N/A</div>
|
|
|
+ <div id="miniPlayerInformation" class="sub header" style="color: #c7c7c7;size:95%;">No information</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div id="miniPlayerIDtab" class="miniPlayer minitab">
|
|
|
+ 0/0
|
|
|
+ </div>
|
|
|
+ <div class="miniPlayerControls">
|
|
|
+ <button class="miniPlayer selectable" style="font-size:90%; font-weight: lighter;" onClick="previousSong();"><i class="step backward icon whiteFont"></i></button>
|
|
|
+ <button class="miniPlayer selectable" style="font-size:120%; font-weight: lighter;" onClick="togglePlayMode();"><i class="play icon whiteFont PlayButton"></i></button>
|
|
|
+ <button class="miniPlayer selectable" style="font-size:90%; font-weight: lighter;" onClick="nextSong();"><i class="step forward icon whiteFont "></i></button>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+<!-- Main interface for player and audio controller-->
|
|
|
+<div id="playerInterface" style="display:none;">
|
|
|
+ <div id="infoBar" class="whiteFont" style="padding:20px;" align="center">
|
|
|
+ <!-- Top Information Bar-->
|
|
|
+
|
|
|
+ <div class="whiteFont songTitleWrapper" style="margin:0px;display:inline-block;">
|
|
|
+ <div id="mainPlayerSongTitle" class="whiteFont" style="font-weight: bold;font-size:120%;">Song Title</div>
|
|
|
+ <small id="mainPlayerSongDesc">This is some small text for display</small>
|
|
|
+ </div>
|
|
|
+ <div style="position:absolute;left:20px;top:30px;cursor:pointer;" onClick="hideMainPlayerInterface();">
|
|
|
+ <i class="large chevron left icon"></i>
|
|
|
+ </div>
|
|
|
+ <div style="position:absolute;right:20px;top:30px;cursor:pointer;" onClick='$("#dropdownSonglist").slideDown();'>
|
|
|
+ <i class="large content icon"></i>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div style="padding-left:20px; padding-right:20px;padding-bottom:5px; position: relative;" align="center">
|
|
|
+ <!-- Function Menu Bar-->
|
|
|
+ <div class="ui grid">
|
|
|
+ <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onclick="showPlaylistInterface();"><i class="large plus icon whiteFont"></i></button></div>
|
|
|
+ <div class="two wide column"><button class="topPanelButtons" id="downloadBtn" style="cursor: pointer;" onClick="downloadPlayingSong();"><i class="large download icon whiteFont"></i></button></div>
|
|
|
+ <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onClick="openInFileExplorer();"><i class="large folder open icon whiteFont"></i></button></div>
|
|
|
+ <div class="two wide column desktopOnly" style="margin-top: -4px;"><button class="topPanelButtons" style="cursor: pointer; opacity: 0.6;" onClick="mute(this);" ><i class="icons"><i class="big red dont icon"></i><i style="margin-left: -3px;" class="large volume off icon whiteFont"></i></i></button></div>
|
|
|
+ <div class="two wide column desktopOnly"><button class="topPanelButtons" id="voldownBtn" style="cursor: pointer;" onClick="addAudioVolume(-0.05);"><i class="large volume down icon whiteFont"></i></button></div>
|
|
|
+ <div class="two wide column desktopOnly"><button class="topPanelButtons" id="volupBtn" style="cursor: pointer;" onClick="addAudioVolume(0.05);"><i class="large volume up icon whiteFont"></i></button></div>
|
|
|
+ <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onClick="showTimerInterface();"><i class="large clock icon whiteFont"></i></button></div>
|
|
|
+ <div class="two wide column"><button class="topPanelButtons" style="cursor: pointer;" onClick="showSettingInterface();"><i class="large setting icon whiteFont"></i></button></div>
|
|
|
+ </div>
|
|
|
+ <div id="volumeGUI" align="center">
|
|
|
+ <i class="volume off large icon whiteFont" style="position:absolute;left:2em;top:1em;"></i>
|
|
|
+ <div class="ui tiny progress" style=" margin-top: calc(2em - 5px); margin-bottom: 0; background-color: rgba(255,255,255,0.2);" align="left">
|
|
|
+ <div id="volBar" class="bar" style="min-width: 0px; width: 40%; height: 10px; background-color: #4b75ff;"></div>
|
|
|
+ </div>
|
|
|
+ <i class="volume up large icon whiteFont" style="position:absolute;right:2em;top:1em;"></i>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <!-- Download progress bar-->
|
|
|
+ <div id="downloadProgressBar" class="ui attached fluid tiny progress" style="background-color:rgba(20, 20, 20, 0.9);display:none;">
|
|
|
+ <div class="bar" style="min-width: 0px; width: 0%;background-color:rgb(75, 117, 255) !important;"></div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ <div id="albumnArt" align="center">
|
|
|
+ <img id="albumnArtImage" style="max-width: 80%; pointer-events: none; user-select: none;" src="img/default.png">
|
|
|
+ </div>
|
|
|
+ <div id="mainPlayerControlInterface">
|
|
|
+ <!-- main control interface-->
|
|
|
+ <div style="width:100%;" align="center">
|
|
|
+ <div id="progressTime">
|
|
|
+ 0:00
|
|
|
+ </div>
|
|
|
+ <div id="mainPlayerMiniTab" class="mainPlayer minitab">
|
|
|
+ 0/0
|
|
|
+ </div>
|
|
|
+ <div id="remainTime">
|
|
|
+ -0:00
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ <div style="position:absolute;width:100%;left:0px;bottom:0px;">
|
|
|
+ <div style="padding-left:10px;padding-right:10px;">
|
|
|
+ <div class="ui small primary progress" style="background-color: rgba(255,255,255,0.4);">
|
|
|
+ <div id="audioProgressBar" class="bar" style="min-width: 0px; width: 100%; background: #4b75ff;"></div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="mainControlButtons">
|
|
|
+ <button class="panelButtons" style="font-size:90%;padding:15px;" onClick="previousSong();"><i class="step backward icon whiteFont"></i></button>
|
|
|
+ <button class="panelButtons" style="font-size:130%;" onClick="togglePlayMode();"><i class="play icon whiteFont PlayButton"></i></button>
|
|
|
+ <button class="panelButtons" style="font-size:90%;padding:15px;" onClick="nextSong();"><i class="step forward icon whiteFont "></i></button>
|
|
|
+ </div>
|
|
|
+ <div id="randomModeButton" style="position: absolute; left:30px;bottom:20px;" onClick="toggleRandomMode(this);">
|
|
|
+ <i class="large random icon disabled"></i>
|
|
|
+ </div>
|
|
|
+ <div id="repeatModeButton" style="position: absolute; right:30px;bottom:20px;" onClick="toggleRepeatMode(this);">
|
|
|
+ <i class="large retweet icon disabled"></i>
|
|
|
+ <p class="whiteFont singleLoop" style="position:absolute;right:0px;bottom:0px;margin-bottom:-13px;margin-right:-2px;display:none;">1</p>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <audio id="mainAudioPlayer" style="display:none;" src=""></audio>
|
|
|
+
|
|
|
+ <div id="dropdownSonglist" class="dropdownMusicList" style="display:none;">
|
|
|
+ <div class="dropdownMusicListMiniMenu">
|
|
|
+ <i class="icons" style="margin-right:8px;">
|
|
|
+ <i class="content icon"></i>
|
|
|
+ <i class="corner music icon" style='color:#333333'></i>
|
|
|
+ </i>
|
|
|
+ Now Playing
|
|
|
+ <div style="position:absolute;top:3px;right:5px;">
|
|
|
+ [ <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>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div id="currentPlayingMainList" style="overflow-x:hidden;">
|
|
|
+
|
|
|
+ </div>
|
|
|
+ <div class="dropdownMusicListBottom" align="center" onClick='$("#dropdownSonglist").slideUp();'><i class="caret up icon"></i></div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+</div>
|
|
|
+
|
|
|
+<!-- Quick Menus and other tool windows-->
|
|
|
+
|
|
|
+ <!-- Timer control interface-->
|
|
|
+ <div id="timerInterface" class="quickMenu">
|
|
|
+ <h4 class="whiteFont">Countdown Timer</h4>
|
|
|
+ <br>
|
|
|
+ <div id="timerSettingInterface" style="width:100%;" align="center">
|
|
|
+ <div>
|
|
|
+ <div class="ui statistic">
|
|
|
+ <div class="label whiteFont">hours</div>
|
|
|
+ <div id="timerHour" class="value whiteFont">0</div>
|
|
|
+ </div>
|
|
|
+ <div class="ui statistic">
|
|
|
+ <div class="label whiteFont">minutes</div>
|
|
|
+ <div id="timerMinute" class="value whiteFont">0</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div>
|
|
|
+ <div class="ui icon mini buttons">
|
|
|
+ <button class="ui secondary button" onClick="addTimer('h',1);"><i class="add icon"></i></button>
|
|
|
+ <button class="ui secondary button" onClick="addTimer('h','r');"><i class="refresh icon"></i></button>
|
|
|
+ <button class="ui secondary button" onClick="addTimer('h',-1);"><i class="minus icon"></i></button>
|
|
|
+ </div>
|
|
|
+ <div class="ui icon mini buttons">
|
|
|
+ <button class="ui secondary button" onClick="addTimer('m',10);"><i class="add icon"></i></button>
|
|
|
+ <button class="ui secondary button" onClick="addTimer('m','r');"><i class="refresh icon"></i></button>
|
|
|
+ <button class="ui secondary button" onClick="addTimer('m',-1);"><i class="minus icon"></i></button>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div id="timerCountingInterface" style="width:100%; display:none;" align="center">
|
|
|
+ <div class="ui statistic">
|
|
|
+ <div id="remainingUI" class="value whiteFont">0:00:00</div>
|
|
|
+ <div class="label whiteFont">Remaining</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="ui divider"></div>
|
|
|
+ <div>
|
|
|
+ <div class="ui grid">
|
|
|
+ <div class="eight wide column">
|
|
|
+ Times up action
|
|
|
+ </div>
|
|
|
+ <div class="eight wide column">
|
|
|
+ <select id="stopMode" class="ui selection mini fluid dropdown">
|
|
|
+ <option>Fade Out & Stop</option>
|
|
|
+ <option>Stop Playing</option>
|
|
|
+ </select>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div style="width:100%;position:absolute;left:0px;bottom:20px;" align="right">
|
|
|
+ <div class="ui grid" >
|
|
|
+ <div class="eight wide column" style="padding-right: 0px;" align="center"><button class="greenBtn fluid" onClick='hideAllquickMenu();'>Close</button></div>
|
|
|
+ <div class="eight wide column" style="padding-left: 0px;" align="center"><button class="greenBtn fluid" onClick="toggleCountDown(this);">Start</button></div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+
|
|
|
+<!-- File properties interface-->
|
|
|
+<div id="filepropInterface" class="quickMenu">
|
|
|
+ <div class="ui header">
|
|
|
+ <span class="filename whiteFont">Title</span>
|
|
|
+ <div class="sub header rawname whiteFont" style="word-break:break-all;">Storage Name</div>
|
|
|
+ </div>
|
|
|
+ <div class="ui list whiteFont">
|
|
|
+ <div class="item whiteFont">Storage Location: <span class="filepath" style="word-break:break-all;">N/A</span></div>
|
|
|
+ <div class="item whiteFont">File Size: <span class="filesize">N/A</span></div>
|
|
|
+ <div class="item whiteFont">Last Modification Date: <span class="filedate">N/A</span></div>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+<!-- Setting interface-->
|
|
|
+<div id="settingInterface" class="quickMenu">
|
|
|
+ <div style="">
|
|
|
+ <div class="ui relaxed list" style="padding:0px !important;">
|
|
|
+ <div id="settingMenuTitle" class="item whiteFont" style="">N/A</div>
|
|
|
+ <div class="item selectable whiteFont" onclick="showPlaylistInterface(); $('#settingInterface').fadeOut('fast');"><i class="add icon" style="margin-right:5px;"></i> Add to playlist</div>
|
|
|
+ <div class="item selectable whiteFont" onClick="openInFileExplorer(); hideAllquickMenu();"><i class="folder open icon" style="margin-right:5px;"></i> Open in File Explorer</div>
|
|
|
+ <div class="item selectable whiteFont" onClick="searchOnYoutube(); hideAllquickMenu();"><i class="youtube play icon" style="margin-right:5px;"></i>Search on Youtube</div>
|
|
|
+ <div class="item selectable whiteFont" onClick="showFileInformation();"><i class="file icon" style="margin-right:5px;"></i> File Information</div>
|
|
|
+ <div class="ui divider"></div>
|
|
|
+ <div id="closeWebAppButton" class="item selectable whiteFont" onClick="ao_module_close();"><i class="remove icon" style="margin-right:5px;"></i> Exit Application</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+
|
|
|
+
|
|
|
+<!-- Playlist interface-->
|
|
|
+<div id="playlistInterface" class="quickMenu">
|
|
|
+ <div class="ui header">
|
|
|
+ <span class="whiteFont">Add to Playlist</span>
|
|
|
+ </div>
|
|
|
+ <div class="playlist item selectable whiteFont" onclick="addToNewPlaylist();">
|
|
|
+ <i class="add icon whiteFont"></i> New Playlist
|
|
|
+ </div>
|
|
|
+ <!-- playlist list-->
|
|
|
+ <div id="existsingPlaylist" style="overflow-y: scroll; height: 50%; border: 1px solid white;">
|
|
|
+ <div class="playlist item selectable whiteFont">
|
|
|
+ <i class="list icon whiteFont"></i> Test Playlist
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <br>
|
|
|
+ <div style="width:100%;position:absolute;left:0px;bottom:0px; margin-bottom: 10px;" align="center">
|
|
|
+ <button class="greenBtn" onClick='hideAllquickMenu();'>Close</button>
|
|
|
+ </div>
|
|
|
+ <div class="ui snackbar" id="succSnackBar">
|
|
|
+ <div class="content">
|
|
|
+ Action Completed
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+</div>
|
|
|
+
|
|
|
+<!-- Utils Interfaces, usually hidden-->
|
|
|
+<div id="fadeReturnScreen" class="fadeScreen whiteFont" onClick="hideAllquickMenu();"></div>
|
|
|
+
|
|
|
+<script>
|
|
|
+ var leftMenuShown = false;
|
|
|
+ var currentPath = "";
|
|
|
+ var currentMode = "music";
|
|
|
+ var rootPaths = [];
|
|
|
+ var totalMusicCount = 0;
|
|
|
+ var currentPlaying = false;
|
|
|
+ var audioElement = $("#mainAudioPlayer");
|
|
|
+ var audioElementObject = document.getElementById("mainAudioPlayer"); //Directly expose the audio object
|
|
|
+ var playingFileDetail = []; //[id, filepath]
|
|
|
+ var displayList = []; //This is the list where the current UI is displaying.
|
|
|
+ var playingList = []; //This is the list where the player last clicked on an item to play
|
|
|
+ var pagingEnabled = false; //Enable this when there are too many songs in list
|
|
|
+ var pagingEnableForPlayingList = false;
|
|
|
+ var pagingCutoffCount = 100; //No. of songs to start paging
|
|
|
+ var currentPage = 0;
|
|
|
+ var currentPlayingPage = 0;
|
|
|
+ var randomMode = false;
|
|
|
+ var repeatMode = 0; //Repeat mode 0 -> No repeat, 1 -> Repeat all 2 -> Repeat one
|
|
|
+ var updateSystemVolume = true;
|
|
|
+ var timerMode = false;
|
|
|
+ var timerRemaining = 0;
|
|
|
+ var playlistAddPendingFile = ""; //The filepath of the file that is pending to be added to playlist
|
|
|
+ var timerEndMode = "fade"; //Fade or End, Provide volume fadeout or end immediately while times up
|
|
|
+ var ua = navigator.userAgent.toLowerCase();
|
|
|
+ var isAndroid = ua.indexOf("android") > -1; //&& ua.indexOf("mobile");
|
|
|
+ var isChrome = /Chrome/.test(navigator.userAgent) && /Google Inc/.test(navigator.vendor);
|
|
|
+ var cacheAvailable = 'caches' in self;
|
|
|
+ let cacheDb, cacheStore, cacheDbTx;
|
|
|
+ var dbExists = !(!window.indexedDB);
|
|
|
+
|
|
|
+ //Init database connection if indexedDB Exists
|
|
|
+ if (dbExists){
|
|
|
+ let cacheDbConn = window.indexedDB.open('airmusic', 4);
|
|
|
+ cacheDbConn.onupgradeneeded = function(event) {
|
|
|
+ cacheDb = event.target.result;
|
|
|
+ cacheDbTx = event.target.transaction;
|
|
|
+ cacheStore = cacheDb.createObjectStore("files",{keyPath:"filename"})
|
|
|
+ };
|
|
|
+ cacheDbConn.onerror = function(event) {
|
|
|
+ console.log("ERROR: " + event.target.errorCode)
|
|
|
+ };
|
|
|
+ cacheDbConn.onsuccess = function(event) {
|
|
|
+ cacheDb = event.target.result
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //Embed native mediaSession for any device that supports it
|
|
|
+ initNativeMediaPlayer();
|
|
|
+
|
|
|
+
|
|
|
+ //Update implementatio nof user agent detection
|
|
|
+ if (typeof InstallTrigger !== 'undefined'){
|
|
|
+ ua = "firefox";
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!!window.chrome && (!!window.chrome.webstore || !!window.chrome.runtime)){
|
|
|
+ ua = "chrome";
|
|
|
+ }
|
|
|
+
|
|
|
+ var isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
|
|
|
+ if (isSafari == true){
|
|
|
+ ua = "safari";
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ $(document).ready(function(){
|
|
|
+ $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 300);
|
|
|
+ //Update 7-8-2019, default loading tab is handled over to the returnToPreviousState() function.
|
|
|
+ //loadSongList();
|
|
|
+ //Initiate the module's volume to the system's
|
|
|
+ syncSystemVol();
|
|
|
+ initAudioListeners();
|
|
|
+ resizeQuickAdjust(); //Adjust all the DOM items position according to window size
|
|
|
+ restoreAllSettings(); //Restore all user prefered settings from storage
|
|
|
+ hideMainPlayerInterface();//Hide the main interface
|
|
|
+ setTimeout(function(){
|
|
|
+ $("#playerInterface").show(); //Only show this interface after it has been out of the working area.
|
|
|
+ },500);
|
|
|
+
|
|
|
+ //Hide the exit application button if it is not in Virtual Desktop Mode
|
|
|
+ if (!ao_module_virtualDesktop){
|
|
|
+ $("#closeWebAppButton").hide();
|
|
|
+ }
|
|
|
+ returnToPreviousState(); //Try to return to previous state using the window hash value
|
|
|
+ });
|
|
|
+
|
|
|
+ function returnToPreviousState(){
|
|
|
+ var hash = window.location.hash;
|
|
|
+ if (hash != ""){
|
|
|
+ //There are state to restore. Restore using the given hash value
|
|
|
+ hash = decodeURIComponent(hash.substring(1));
|
|
|
+ var previousState = JSON.parse(hash);
|
|
|
+ console.log(previousState);
|
|
|
+ var currentPath = previousState.path;
|
|
|
+ var playingFileDetail = previousState.pfp;
|
|
|
+ var initPauseState = previousState.pause;
|
|
|
+ if (currentPath != ""){
|
|
|
+ //Create a dummy folder and trick the system to load to the previous states
|
|
|
+ var a = '<div></div>';
|
|
|
+ a = $(a).attr("filepath",currentPath);
|
|
|
+ openFolder(a[0],{playIDAfterOpen: playingFileDetail[0],startPaused: initPauseState});
|
|
|
+ }else{
|
|
|
+ //currentpath is empty, aka Music Mode
|
|
|
+ loadSongList();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ }else{
|
|
|
+ //No state to restore. Restore Viewing Tab instead.
|
|
|
+ var vt = loadStorage("viewingTab");
|
|
|
+ if (vt != ""){
|
|
|
+ if (vt == "music"){
|
|
|
+ loadSongList();
|
|
|
+
|
|
|
+ }else if(vt == "folder"){
|
|
|
+ loadFolderView();
|
|
|
+
|
|
|
+ }else if(vt == "playlist"){
|
|
|
+ loadPlaylistView();
|
|
|
+
|
|
|
+ }else if(vt == "network"){
|
|
|
+ loadNetworkView();
|
|
|
+
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ //Default action
|
|
|
+ loadSongList();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadNetworkView(){
|
|
|
+ currentMode = "network";
|
|
|
+ togglePagingMode(false);
|
|
|
+ }
|
|
|
+
|
|
|
+ function togglePagingMode(enabled=false){
|
|
|
+ if (pagingEnabled != enabled){
|
|
|
+ currentPage = 0;
|
|
|
+ }
|
|
|
+ pagingEnabled = enabled;
|
|
|
+ }
|
|
|
+
|
|
|
+ function initAudioListeners(){
|
|
|
+ //Initiate audio progress bar and link it to the progress bar on the main player interface
|
|
|
+ audioElement[0].addEventListener("timeupdate", function() {
|
|
|
+ var currentTime = audioElement[0].currentTime;
|
|
|
+ var duration = audioElement[0].duration;
|
|
|
+ $('#audioProgressBar').css('width',((currentTime +.25)/duration)*100 + '%');
|
|
|
+ updatePlaybackDisplayTime(currentTime,duration);
|
|
|
+ });
|
|
|
+ audioElement[0].onended = function() {
|
|
|
+ nextSongHandler();
|
|
|
+ };
|
|
|
+
|
|
|
+ //Handle clicks on progress bar to skip through the playback
|
|
|
+ $("#audioProgressBar").parent().on("click",function(e){
|
|
|
+ var duration = audioElement[0].duration;
|
|
|
+ var jumpTo = (e.offsetX / $(this).width()) * duration;
|
|
|
+ if (!audioElement[0].paused){
|
|
|
+ audioElement[0].pause();
|
|
|
+ audioElement[0].currentTime = jumpTo;
|
|
|
+ audioElement[0].play();
|
|
|
+ }else{
|
|
|
+ audioElement[0].currentTime = jumpTo;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function restoreAllSettings(){
|
|
|
+ //Repeat mode setting
|
|
|
+ var strRmode = loadStorage("repeatMode");
|
|
|
+ if (strRmode != ""){
|
|
|
+ repeatMode = parseInt(strRmode);
|
|
|
+ setRepeatMode(repeatMode,$("#repeatModeButton")[0]);
|
|
|
+ }
|
|
|
+ //Random Mode setting
|
|
|
+ var random = loadStorage("randomMode");
|
|
|
+ if (random != ""){
|
|
|
+ if (random == "true"){
|
|
|
+ //random Mode is true when loaded from configuration. Toggle once to match the setting
|
|
|
+ toggleRandomMode($("#randomModeButton")[0]);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ //Timer mode setting
|
|
|
+ var TEM = loadStorage("timerEndMode");
|
|
|
+ if (TEM != ""){
|
|
|
+ if (TEM == "fade"){
|
|
|
+ $("#stopMode").val("Fade Out & Stop");
|
|
|
+ }else{
|
|
|
+ $("#stopMode").val("Stop Playing");
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ //Setting menu function
|
|
|
+ function searchOnYoutube(filename = ""){
|
|
|
+ if (filename == ""){
|
|
|
+ var filename = $("#mainPlayerSongTitle").text().trim();
|
|
|
+ }
|
|
|
+
|
|
|
+ var url = "https://www.youtube.com/results?search_query=" + filename ;
|
|
|
+ window.open(url);
|
|
|
+ }
|
|
|
+
|
|
|
+ //Searching related functions
|
|
|
+ var preSearchItemBuffer = "";
|
|
|
+ var searchModeEnabled = false;
|
|
|
+ var previousPagingEnabled = false;
|
|
|
+ function enterSearchMode(){
|
|
|
+ if (searchModeEnabled){
|
|
|
+ //Search mode already enabled
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ previousPagingEnabled = pagingEnabled;
|
|
|
+ togglePagingMode(false);
|
|
|
+ $("#searchModeMenu").show();
|
|
|
+ $("#mainMenu").hide();
|
|
|
+ let previousDisplayList = Array.from(displayList);
|
|
|
+ preSearchItemBuffer = [$("#mainList").html(),$("#interfaceTitle").text(),$("#AMmenuIcon").attr("class"), $("#interfaceDetails").html(), previousDisplayList];
|
|
|
+ searchModeEnabled = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ function exitSearchMode(){
|
|
|
+ $("#searchModeMenu").hide();
|
|
|
+ $("#mainMenu").show();
|
|
|
+ $("#searchInputBar").val("");
|
|
|
+ if (preSearchItemBuffer != ""){
|
|
|
+ //If previous record exists, recover the previous view from storage
|
|
|
+ $("#mainList").html(preSearchItemBuffer[0]);
|
|
|
+ $("#interfaceTitle").text(preSearchItemBuffer[1]);
|
|
|
+ $("#AMmenuIcon").attr("class",preSearchItemBuffer[2]);
|
|
|
+ $("#interfaceDetails").html(preSearchItemBuffer[3]);
|
|
|
+ displayList = preSearchItemBuffer[4];
|
|
|
+ preSearchItemBuffer = "";
|
|
|
+ }
|
|
|
+
|
|
|
+ if (previousPagingEnabled){
|
|
|
+ togglePagingMode(true);
|
|
|
+ }
|
|
|
+ searchModeEnabled = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ document.getElementById("searchInputBar").addEventListener("keypress", function(event){
|
|
|
+ console.log(event, event.keyCode);
|
|
|
+ if(event.keyCode == 13){
|
|
|
+ //Enter pressed. start searching
|
|
|
+ searchSong();
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ /*
|
|
|
+ function handleSearchInputEnter(e){
|
|
|
+ if(e.keyCode == 13){
|
|
|
+ //Enter pressed. start searching
|
|
|
+ searchSong();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ */
|
|
|
+
|
|
|
+ function searchSong(){
|
|
|
+ var keyword = $("#searchInputBar").val();
|
|
|
+ if (keyword.includes("?") || keyword.includes("&")){
|
|
|
+ alert("Search keyword cannot contains & or ? symbol.");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (keyword == ""){
|
|
|
+ //Empty keyword. Return nothing
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ //Clear the mainList, replace with a dummy loading screen
|
|
|
+ $("#mainList").html('<div class="mainList item whiteFont">\
|
|
|
+ <div class="ui header" style="margin:0px !important;padding-bottom:15px;padding-top:5px;">\
|
|
|
+ <i class="loading spinner icon whiteFont" style="overflow:hidden;margin-top:5px;"></i>\
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
|
|
|
+ Searching...\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ <div class="topRightCorner" align="center">\
|
|
|
+ -1\
|
|
|
+ </div>\
|
|
|
+ </div>');
|
|
|
+
|
|
|
+ ao_module_agirun("Music/functions/listSong.js", {
|
|
|
+ listSong: "search:" + keyword
|
|
|
+ }, function(data){
|
|
|
+ currentPath = "";
|
|
|
+ var template = '<div class="mainList file item" filepath={filepath} id={id} rawname={rawname}>\
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">\
|
|
|
+ <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>\
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
|
|
|
+ {songtitle}\
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ <div class="topRightCorner" align="center">\
|
|
|
+ {id}\
|
|
|
+ </div>\
|
|
|
+ <div class="mainList rightFunctionBar" align="center" onClick="showMore(this);">\
|
|
|
+ <i class="ellipsis vertical icon" style="top: 2em;"></i>\
|
|
|
+ </div>\
|
|
|
+ </div>';
|
|
|
+
|
|
|
+ if ($("#searchModeMenu").is(":hidden")){
|
|
|
+ //The user exited search mode before the search results come back
|
|
|
+ //Dispose the search result
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (data.length == 0){
|
|
|
+ //No search results.
|
|
|
+ $("#mainList").html("");
|
|
|
+ $("#mainList").append('<div class="mainList item whiteFont">\
|
|
|
+ <div class="ui header" style="margin:0px !important;padding-bottom:15px;padding-top:5px;">\
|
|
|
+ <i class="remove icon whiteFont" style="overflow:hidden;margin-top:5px;"></i>\
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">\
|
|
|
+ No result for keyword: ' + keyword + '.\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ <div class="topRightCorner" align="center">\
|
|
|
+ -1\
|
|
|
+ </div>\
|
|
|
+ </div>');
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ $("#interfaceTitle").text("Search");
|
|
|
+ $("#AMmenuIcon").attr("class","search icon large whiteFont")
|
|
|
+ //Initialize the song list
|
|
|
+ displayList = data;
|
|
|
+ if (playingList == []){
|
|
|
+ playingList = Array.from(displayList);
|
|
|
+ pagingEnableForPlayingList = pagingEnabled;
|
|
|
+ currentPlayingPage = currentPage;
|
|
|
+ }
|
|
|
+ $("#mainList").html("");
|
|
|
+ for ( var i =0; i < data.length; i++){
|
|
|
+ var songInfo = data[i];
|
|
|
+ var path = songInfo[0];
|
|
|
+ var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
|
|
|
+ var ext = songInfo[2];
|
|
|
+ var size = songInfo[3];
|
|
|
+ var box = template;
|
|
|
+ box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
|
|
|
+ box = replaceAll("{id}",i + 1,box); //User count from 1
|
|
|
+ box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
|
|
|
+ box = replaceAll("{songtitle}",displayname,box);
|
|
|
+ box = replaceAll("{ext}",ext,box);
|
|
|
+ box = replaceAll("{size}",size,box);
|
|
|
+ $("#mainList").append(box);
|
|
|
+ }
|
|
|
+ totalMusicCount = data.length;
|
|
|
+ $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
+ $("#interfaceDetails").text("[" + i + " songs]");
|
|
|
+ highLightPlayingMusic();
|
|
|
+ loadThumbnailToMusicList(".mainList.file.item");
|
|
|
+ });
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function enterRemotePlayID(object){
|
|
|
+ var id = $(object).text().trim();
|
|
|
+ $("#remotePlayID").val($("#remotePlayID").val() + id);
|
|
|
+ }
|
|
|
+
|
|
|
+ function ridOpr(opr){
|
|
|
+ if (opr == 'backspace'){
|
|
|
+ var currentText = $("#remotePlayID").val();
|
|
|
+ var newText = currentText.substring(0,currentText.length - 1);
|
|
|
+ $("#remotePlayID").val(newText);
|
|
|
+ }else if (opr == 'reset'){
|
|
|
+ $("#remotePlayID").val("");
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function openInFileExplorer(){
|
|
|
+ var dirname = playingFileDetail[1].split("/");
|
|
|
+ var fname = dirname.pop(); //Pop away the filename
|
|
|
+ dirname = dirname.join("/") + "/";
|
|
|
+ var path = dirname;
|
|
|
+ if (dirname.substring(0,1) != "/"){
|
|
|
+ //This is not absolute path. Start the path from airMusic
|
|
|
+ path = "AirMusic/" + dirname;
|
|
|
+ }
|
|
|
+
|
|
|
+ path = path.replace("media?file=","");
|
|
|
+ if (path.substring(0,1) == "/"){
|
|
|
+ path = path.substring(1)
|
|
|
+ }
|
|
|
+ ao_module_openPath(path, fname);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ //quickmenu Functions
|
|
|
+ function hideAllquickMenu(){
|
|
|
+ $(".quickMenu").fadeOut('fast');
|
|
|
+ $("#fadeReturnScreen").fadeOut('fast');
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ //Setting related function
|
|
|
+ function showSettingInterface(){
|
|
|
+ $("#settingInterface").fadeIn('fast');
|
|
|
+ $("#fadeReturnScreen").fadeIn('fast');
|
|
|
+ //Update contents in the setting menu
|
|
|
+ $("#settingMenuTitle").text($("#mainPlayerSongTitle").text());
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ //Timer related functions
|
|
|
+ function showTimerInterface(){
|
|
|
+ $("#timerInterface").fadeIn('fast');
|
|
|
+ $("#fadeReturnScreen").fadeIn('fast');
|
|
|
+ }
|
|
|
+
|
|
|
+ function initPlaylistInterfaceList(){
|
|
|
+ $("#existsingPlaylist").html("");
|
|
|
+ //Get the playlist list from server side
|
|
|
+ ao_module_agirun("Music/functions/playlist.js", {
|
|
|
+ opr: "root"
|
|
|
+ }, function(data){
|
|
|
+ data.forEach(playlist => {
|
|
|
+ $("#existsingPlaylist").append(`<div class="playlist item selectable whiteFont" name="${playlist.name}" onclick="addSongToSelectedPlaylist(this);">
|
|
|
+ <i class="list icon whiteFont"></i> ${playlist.name} [${playlist.count} files]
|
|
|
+ </div>`);
|
|
|
+ })
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function showPlaylistInterface(){
|
|
|
+ initPlaylistInterfaceList();
|
|
|
+ $("#playlistInterface").fadeIn('fast');
|
|
|
+ $("#fadeReturnScreen").fadeIn('fast');
|
|
|
+ }
|
|
|
+
|
|
|
+ function addToThisPlaylist(){
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ $("#stopMode").change(function () {
|
|
|
+ var end = this.value;
|
|
|
+ if (end.includes("Fade Out & Stop")){
|
|
|
+ //Start fading out the volume
|
|
|
+ timerEndMode = "fade";
|
|
|
+ setStorage("timerEndMode","fade");
|
|
|
+ }else{
|
|
|
+ //End immediately
|
|
|
+ timerEndMode = "end";
|
|
|
+ setStorage("timerEndMode","end");
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ var fadeOutStepping = 0.1; //The steps size for fading out during timer countdown
|
|
|
+ var defaultVolumeBeforeFadeout = 0;
|
|
|
+ function toggleCountDown(button){
|
|
|
+ if (!timerMode){
|
|
|
+ //Start count down
|
|
|
+ timerMode = true;
|
|
|
+ $(button).text("Stop");
|
|
|
+ timerRemaining = ((parseInt($("#timerHour").text()) * 60) + parseInt($("#timerMinute").text())) * 60; //in seconds
|
|
|
+ if (timerRemaining <= 0){
|
|
|
+ //Start with 0
|
|
|
+ timerMode = false;
|
|
|
+ $(button).text("Start");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ //Update the value on the remaining counter
|
|
|
+ $("#remainingUI").text(parseTimer(timerRemaining));
|
|
|
+ $("#timerSettingInterface").slideUp('fast');
|
|
|
+ $("#timerCountingInterface").slideDown('fast');
|
|
|
+ $("#stopMode").attr("disabled","");
|
|
|
+ defaultVolumeBeforeFadeout = audioElement[0].volume;
|
|
|
+ fadeOutStepping = (audioElement[0].volume) / timerRemaining;
|
|
|
+ progressCounter();
|
|
|
+ }else{
|
|
|
+ //Stop count down
|
|
|
+ timerMode = false;
|
|
|
+ $(button).text("Start");
|
|
|
+ timerRemaining = 0;
|
|
|
+ $("#timerSettingInterface").slideDown('fast');
|
|
|
+ $("#timerCountingInterface").slideUp('fast');
|
|
|
+ $("#stopMode").removeAttr("disabled");
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function progressCounter(){
|
|
|
+ //Update the timer
|
|
|
+ timerRemaining = timerRemaining - 1;
|
|
|
+ $("#remainingUI").text(parseTimer(timerRemaining));
|
|
|
+ if (timerRemaining > 0){
|
|
|
+ //Continues
|
|
|
+ if (timerEndMode == "fade"){
|
|
|
+ var newvol = audioElement[0].volume - fadeOutStepping;
|
|
|
+ audioElement[0].volume = Math.max(0,newvol);
|
|
|
+ }
|
|
|
+ setTimeout(progressCounter,1000);
|
|
|
+ }else{
|
|
|
+ //Times up
|
|
|
+ handleTimerEnd();
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function handleTimerEnd(){
|
|
|
+ //Stop the playback
|
|
|
+ originalVol = audioElement[0].volume;
|
|
|
+ fadeAudio();
|
|
|
+ setPlaying(false);
|
|
|
+ audioElement[0].pause();
|
|
|
+ setTimeout(function(){
|
|
|
+ audioElement[0].volume = defaultVolumeBeforeFadeout;
|
|
|
+ },500);
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function parseTimer(remainingTime){
|
|
|
+ //remainingTime in seconds, prase to HH:mm:ss
|
|
|
+ var hours = Math.floor(remainingTime / 3600);
|
|
|
+ var remainingTime = remainingTime % 3600;
|
|
|
+ var minutes = Math.floor(remainingTime / 60);
|
|
|
+ var seconds = remainingTime % 60;
|
|
|
+ if (minutes < 10){
|
|
|
+ minutes = "0" + minutes;
|
|
|
+ }
|
|
|
+ if (seconds < 10){
|
|
|
+ seconds = "0" + seconds;
|
|
|
+ }
|
|
|
+ var formatted = hours + ":" + minutes + ":" + seconds;
|
|
|
+ return formatted;
|
|
|
+ }
|
|
|
+
|
|
|
+ function addTimer(unit,value){
|
|
|
+ if (unit == "h"){
|
|
|
+ if (value == "r"){
|
|
|
+ $("#timerHour").text("0");
|
|
|
+ }else{
|
|
|
+ $("#timerHour").text($("#timerHour").text() - -1 * value);
|
|
|
+ }
|
|
|
+
|
|
|
+ }else if (unit == "m"){
|
|
|
+ if (value == "r"){
|
|
|
+ $("#timerMinute").text("0");
|
|
|
+ }else{
|
|
|
+ $("#timerMinute").text($("#timerMinute").text() - -1 * value);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (parseInt($("#timerHour").text()) < 0){
|
|
|
+ $("#timerHour").text("0");
|
|
|
+ }
|
|
|
+ if (parseInt($("#timerMinute").text()) < 0){
|
|
|
+ if (parseInt($("#timerHour").text()) > 0){
|
|
|
+ //subtract 1 from the hour
|
|
|
+ $("#timerHour").text($("#timerHour").text() - 1);
|
|
|
+ $("#timerMinute").text(parseInt($("#timerMinute").text()) + 60);
|
|
|
+ }else{
|
|
|
+ //There is no hours to subtract
|
|
|
+ $("#timerMinute").text("0");
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+ if (parseInt($("#timerMinute").text()) > 59){
|
|
|
+ $("#timerMinute").text(parseInt($("#timerMinute").text()) - 60);
|
|
|
+ $("#timerHour").text($("#timerHour").text() - -1);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //Download function
|
|
|
+ function downloadPlayingSong(){
|
|
|
+ var url = audioElement.attr("src");
|
|
|
+ var ext = url.split(".").pop();
|
|
|
+ var filename = $("#mainPlayerSongTitle").text().trim();
|
|
|
+ generateDownloadElement(url,filename);
|
|
|
+ /*
|
|
|
+ if (url.includes("/media")){
|
|
|
+ blobDownloadElement(url,filename);
|
|
|
+ }else{
|
|
|
+ generateDownloadElement(url,filename);
|
|
|
+ }
|
|
|
+ */
|
|
|
+ }
|
|
|
+
|
|
|
+ var downloadInProgress = false;
|
|
|
+ function blobDownloadElement(filepath,filename){
|
|
|
+ if (downloadInProgress){
|
|
|
+ alert("Please wait until another download finished.");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ downloadInProgress = true;
|
|
|
+ $("#downloadBtn").find("i").addClass("disabled");
|
|
|
+ $("#downloadProgressBar").find(".bar").css("width","0%");
|
|
|
+ $("#downloadProgressBar").show();
|
|
|
+ var xhr = new XMLHttpRequest();
|
|
|
+ xhr.onreadystatechange = function(){
|
|
|
+ if (this.readyState == 4 && this.status == 200){
|
|
|
+ //handler(this.response);
|
|
|
+ //console.log(this.response, typeof this.response);
|
|
|
+ var url = window.URL || window.webkitURL;
|
|
|
+ //Open the downloaded data in new window for testing
|
|
|
+ //window.open(url.createObjectURL(this.response));
|
|
|
+ generateDownloadElement(url.createObjectURL(this.response),filename);
|
|
|
+ $("#downloadBtn").find("i").removeClass("disabled");
|
|
|
+ $("#downloadProgressBar").hide();
|
|
|
+ downloadInProgress = false;
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+ xhr.onprogress = function (event) {
|
|
|
+ console.log("[AirMusic] Download Progress: " + event.loaded + " / " + event.total + "Bytes");
|
|
|
+ $("#downloadProgressBar").find(".bar").css("width",(event.loaded / event.total) * 100 + "%");
|
|
|
+ };
|
|
|
+ xhr.open('GET', filepath);
|
|
|
+ xhr.responseType = 'blob';
|
|
|
+ xhr.send();
|
|
|
+ }
|
|
|
+
|
|
|
+ function generateDownloadElement(filepath, filename){
|
|
|
+ var link = document.createElement('a');
|
|
|
+
|
|
|
+ //Clean the filepath
|
|
|
+ filepath = filepath.split("//").join("/");
|
|
|
+ //Convert the filepath to download path
|
|
|
+ link.href = filepath.replace("/media?file=", "/media/download/?file=");
|
|
|
+
|
|
|
+ //Generate the download element
|
|
|
+ link.setAttribute('download', filename);
|
|
|
+ document.getElementsByTagName("body")[0].appendChild(link);
|
|
|
+ // Firefox
|
|
|
+ if (document.createEvent) {
|
|
|
+ var event = document.createEvent("MouseEvents");
|
|
|
+ event.initEvent("click", true, true);
|
|
|
+ link.dispatchEvent(event);
|
|
|
+ }
|
|
|
+ // IE
|
|
|
+ else if (link.click) {
|
|
|
+ link.click();
|
|
|
+ }
|
|
|
+ link.parentNode.removeChild(link);
|
|
|
+ }
|
|
|
+
|
|
|
+ //Volume control related features
|
|
|
+ var previousAudioVolume = 0;
|
|
|
+ function mute(button){
|
|
|
+ if (audioElement[0].volume != 0){
|
|
|
+ //Set to mute
|
|
|
+ previousAudioVolume = audioElement[0].volume;
|
|
|
+ audioElement[0].volume = 0;
|
|
|
+ $(button).css("opacity", "1");
|
|
|
+ $("#voldownBtn").addClass("disabled");
|
|
|
+ $("#volupBtn").addClass("disabled");
|
|
|
+ }else{
|
|
|
+ //Restore from mute
|
|
|
+ audioElement[0].volume = previousAudioVolume;
|
|
|
+ $(button).css("opacity", "0.6");
|
|
|
+ $("#voldownBtn").removeClass("disabled");
|
|
|
+ $("#volupBtn").removeClass("disabled");
|
|
|
+ }
|
|
|
+ $("#volBar").css("width",audioElement[0].volume * 100 + "%");
|
|
|
+ }
|
|
|
+ var volbarTimeoutEvt;
|
|
|
+ function addAudioVolume(value){
|
|
|
+ $("#volumeGUI").slideDown('fast');
|
|
|
+ if (volbarTimeoutEvt !== undefined){
|
|
|
+ clearTimeout(volbarTimeoutEvt);
|
|
|
+ }
|
|
|
+ volbarTimeoutEvt = setTimeout(function(){
|
|
|
+ $("#volumeGUI").slideUp('fast');
|
|
|
+ },3000);
|
|
|
+ $("#volumeGUI")
|
|
|
+ if (audioElement[0].volume + value >= 1){
|
|
|
+ audioElement[0].volume = 1;
|
|
|
+ }else if (audioElement[0].volume + value <= 0){
|
|
|
+ audioElement[0].volume = 0;
|
|
|
+ }else{
|
|
|
+ audioElement[0].volume += value;
|
|
|
+ }
|
|
|
+
|
|
|
+ $("#volBar").css("width",Math.min(100,audioElement[0].volume * 100) + "%");
|
|
|
+ }
|
|
|
+
|
|
|
+ function updatePlaybackDisplayTime(currentTime,duration){
|
|
|
+ let progress = Math.round(currentTime);
|
|
|
+ let remainTime = Math.round(duration - currentTime);
|
|
|
+ $("#progressTime").text(secondsToHms(progress));
|
|
|
+ if (!isNaN(remainTime)){
|
|
|
+ $("#remainTime").text("-" + secondsToHms(remainTime));
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function nextSongHandler(){
|
|
|
+ console.log("Next song");
|
|
|
+ if (repeatMode == 1){
|
|
|
+ //Next song in the playlist
|
|
|
+ if (randomMode){
|
|
|
+ var randomTrackID = Math.floor(Math.random() * (playingList.length - 1));
|
|
|
+ //console.log(randomTrackID, playingList[randomTrackID]);
|
|
|
+ nextSong(randomTrackID,true);
|
|
|
+ }else{
|
|
|
+ var playingSongIndex = playingFileDetail[0];
|
|
|
+ var nextSongIndex = playingSongIndex;
|
|
|
+ if (nextSongIndex >= playingList.length){
|
|
|
+ nextSongIndex = 0;
|
|
|
+ }
|
|
|
+ //console.log(playingSongIndex, nextSongIndex);
|
|
|
+ nextSong(nextSongIndex,true);
|
|
|
+ }
|
|
|
+ }else if (repeatMode == 2){
|
|
|
+ //Loop this song only
|
|
|
+ audioElement[0].pause();
|
|
|
+ audioElement[0].currentTime = 0;
|
|
|
+ audioElement[0].play();
|
|
|
+ }else if (repeatMode == 0){
|
|
|
+ //Pause after finish this song
|
|
|
+ setPlaying(false);
|
|
|
+ audioElement[0].pause();
|
|
|
+ audioElement[0].currentTime = 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function mainPlayerShown(){
|
|
|
+ if ($("#playerInterface").offset().left == 0){
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ return false;
|
|
|
+ }
|
|
|
+
|
|
|
+ function toggleRandomMode(button){
|
|
|
+ var btn = $(button).find("i");
|
|
|
+ if (randomMode){
|
|
|
+ //Set the random mode to false
|
|
|
+ btn.removeClass("enabled").addClass("disabled");
|
|
|
+ randomMode = false;
|
|
|
+ }else{
|
|
|
+ btn.removeClass("disabled").addClass("enabled");
|
|
|
+ randomMode = true;
|
|
|
+ }
|
|
|
+ setStorage("randomMode",randomMode);
|
|
|
+ }
|
|
|
+
|
|
|
+ function toggleRepeatMode(button){
|
|
|
+ if (repeatMode == 0){
|
|
|
+ setRepeatMode(1,button);
|
|
|
+ setStorage("repeatMode",1);
|
|
|
+ }else if (repeatMode == 1){
|
|
|
+ setRepeatMode(2,button);
|
|
|
+ setStorage("repeatMode",2);
|
|
|
+ }else if (repeatMode == 2){
|
|
|
+ setRepeatMode(0,button);
|
|
|
+ setStorage("repeatMode",0);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function setRepeatMode(modeCode,button){
|
|
|
+ if (modeCode == 1){
|
|
|
+ //Set repeat mode to "Repeat all"
|
|
|
+ $(button).find(".singleLoop").hide();
|
|
|
+ $(button).find("i").removeClass("disabled").addClass("enabled");
|
|
|
+ repeatMode = 1;
|
|
|
+ }else if (modeCode == 2){
|
|
|
+ //Set repeat mode to "Repeat one"
|
|
|
+ $(button).find(".singleLoop").show();
|
|
|
+ $(button).find("i").removeClass("disabled").addClass("enabled");
|
|
|
+ repeatMode = 2;
|
|
|
+ }else if (modeCode == 0){
|
|
|
+ //Set repeat mode to "None"
|
|
|
+ $(button).find(".singleLoop").hide();
|
|
|
+ $(button).find("i").removeClass("enabled").addClass("disabled");
|
|
|
+ repeatMode = 0;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ function hideMainPlayerInterface(){
|
|
|
+ $("#playerInterface").animate({left: window.innerWidth},300);
|
|
|
+ $("body").css("overflow-y","auto");
|
|
|
+ }
|
|
|
+
|
|
|
+ function showMainPlayerInterface(){
|
|
|
+ $("#playerInterface").animate({left:0},300);
|
|
|
+ $("body").css("overflow-y","hidden");
|
|
|
+ }
|
|
|
+
|
|
|
+ function syncSystemVol(){
|
|
|
+ //Initiate the module's volume to the system's
|
|
|
+ var globalvol = localStorage.getItem("global_volume");
|
|
|
+ //console.log("Global Volume" + globalvol.toString());
|
|
|
+ if (!globalvol){
|
|
|
+ globalvol = 0.1;
|
|
|
+ }
|
|
|
+
|
|
|
+ //Check if it is mobile. If yes, always 100% and leave volume to system
|
|
|
+ if (isMobile()){
|
|
|
+ globalvol = 1;
|
|
|
+ //$(".desktopOnly").addClass("disabled");
|
|
|
+ }
|
|
|
+
|
|
|
+ audioElement[0].volume = parseFloat(globalvol);
|
|
|
+ $("#volBar").css("width",audioElement[0].volume * 100 + "%");
|
|
|
+ }
|
|
|
+
|
|
|
+ function replaceAll(target, replace, original){
|
|
|
+ return original.split(target).join(replace);
|
|
|
+ };
|
|
|
+
|
|
|
+ function playSong(object, startPaused = false){
|
|
|
+ if ($(object).parent().hasClass("playingTrack")){
|
|
|
+ //This is already the song that is currently playing. Show the main player interface instead.
|
|
|
+ showMainPlayerInterface();
|
|
|
+ resizeQuickAdjust();
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ //$(".playingTrack").removeClass("playingTrack");
|
|
|
+ //$(object).parent().addClass("playingTrack");
|
|
|
+ var filepath = ao_module_utils.attrToObject($(object).parent().attr('filepath'));
|
|
|
+ var rawname = ao_module_utils.attrToObject($(object).parent().attr('rawname'));
|
|
|
+ var displayName = ao_module_codec.decodeUmFilename(rawname);
|
|
|
+ var info = $(object).parent().find(".fileinfo").text();
|
|
|
+ var id = $(object).parent().attr('id');
|
|
|
+ if (id === undefined){
|
|
|
+ //This might be a file from dropdown list. use listid instead
|
|
|
+ id = $(object).parent().attr('listid');
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ updateMiniPlayerUI(displayName,info,id);
|
|
|
+
|
|
|
+ //Check if page mode is enabled. If yes, do auto tab switch
|
|
|
+ if (pagingEnabled){
|
|
|
+ var targetPageNumber = getPageNumberByPlaybackId(parseInt(id));
|
|
|
+ switchToPage(targetPageNumber);
|
|
|
+ }
|
|
|
+
|
|
|
+ //Seperate out the filepath without media server path
|
|
|
+ var fileVpath = filepath.split("=")[1];
|
|
|
+ updateMainPlayerUI(displayName,info,id, fileVpath);
|
|
|
+ loadAndPlayAudioFile(filepath, !startPaused);
|
|
|
+ if (!currentPlaying && !startPaused){
|
|
|
+ setPlaying(true);
|
|
|
+ }
|
|
|
+ playingFileDetail = [parseInt(id),filepath];
|
|
|
+
|
|
|
+ //If the user play this file, assume the next file to be added to playlist is this file
|
|
|
+ playlistAddPendingFile = filepath;
|
|
|
+ //Have a backup of the current list just incase the user open another directory while a list is playing
|
|
|
+ //Only backup when it is not call from dropdown list
|
|
|
+ if (!$(object).parent().hasClass("dropdownList")){
|
|
|
+ playingList = Array.from(displayList);
|
|
|
+ pagingEnableForPlayingList = pagingEnabled;
|
|
|
+ currentPlayingPage = currentPage;
|
|
|
+ }
|
|
|
+ parsePlayingSongList();
|
|
|
+ if ($("#miniPlayer").hasClass("hidden")){
|
|
|
+ //This is the first song that has played after the UI init
|
|
|
+ $("#miniPlayer").css("display","none").removeClass("hidden").slideDown();
|
|
|
+ }
|
|
|
+ showMainPlayerInterface();
|
|
|
+
|
|
|
+ //Just in case the albumn image changed, albumn art css is also updated.
|
|
|
+ resizeQuickAdjust();
|
|
|
+
|
|
|
+ updateStateReferingURL();
|
|
|
+ highLightPlayingMusic();
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadThumbnailToMusicList(selector = ".mainList.file.item"){
|
|
|
+ $(selector).each(function(){
|
|
|
+ let filepath = JSON.parse(decodeURIComponent($(this).attr("filepath")));
|
|
|
+ let thisFileItemID = $(this).attr("id");
|
|
|
+ let rawname = JSON.parse(decodeURIComponent($(this).attr("rawname")));
|
|
|
+
|
|
|
+ let realVpath = filepath.split("=");
|
|
|
+ realVpath.shift();
|
|
|
+ realVpath = realVpath.join("=");
|
|
|
+ //console.log(thisFileItemID, realVpath, rawname);
|
|
|
+ //Load the thumbnail for this item in list
|
|
|
+ fetch("../system/file_system/loadThumbnail?vpath=" + encodeURIComponent(realVpath))
|
|
|
+ .then((response) => response.json())
|
|
|
+ .then((data) => {
|
|
|
+ if (data.error !== undefined || data.trim() == ""){
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ $("#" + thisFileItemID).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
+ $(".dropdownList.file.item").each(function(){
|
|
|
+ var thisFilepath = JSON.parse(decodeURIComponent($(this).attr("filepath")));
|
|
|
+ if ($(this).attr("listid") == thisFileItemID && thisFilepath == filepath){
|
|
|
+ $(this).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ });
|
|
|
+ /*
|
|
|
+ ao_module_agirun("Music/functions/getThumbnail.js", {
|
|
|
+ file: realVpath,
|
|
|
+ }, function(data){
|
|
|
+ if (data.error !== undefined){
|
|
|
+
|
|
|
+ }else{
|
|
|
+ $("#" + thisFileItemID).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
+ $(".dropdownList.file.item").each(function(){
|
|
|
+ var thisFilepath = JSON.parse(decodeURIComponent($(this).attr("filepath")));
|
|
|
+ if ($(this).attr("listid") == thisFileItemID && thisFilepath == filepath){
|
|
|
+ $(this).find("img").attr("src","data:image/jpg;base64," + data);
|
|
|
+ }
|
|
|
+ })
|
|
|
+ }
|
|
|
+ });
|
|
|
+ */
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadThumbnail(filepath){
|
|
|
+ //Load the thumbnail
|
|
|
+ var realVpath = filepath.split("=");
|
|
|
+ realVpath.shift();
|
|
|
+ realVpath = realVpath.join("=");
|
|
|
+ ao_module_agirun("Music/functions/getThumbnail.js", {
|
|
|
+ file: realVpath,
|
|
|
+ }, function(data){
|
|
|
+ if (data.error !== undefined){
|
|
|
+ console.log(data.error)
|
|
|
+ $("#albumnArtImage").attr("src","img/default.png");
|
|
|
+ $("#smallPlayerThumb").attr("src","img/default.png");
|
|
|
+ if (navigator.mediaSession.metadata){
|
|
|
+ navigator.mediaSession.metadata.artwork = [
|
|
|
+ { src: "img/default.png", sizes: '512x512', type: 'image/png' }
|
|
|
+ ]
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ $("#albumnArtImage").attr("src","data:image/jpg;base64," + data);
|
|
|
+ $("#smallPlayerThumb").attr("src","data:image/jpg;base64," + data);
|
|
|
+ if (isAndroid && navigator.mediaSession.metadata){
|
|
|
+ //Android
|
|
|
+ navigator.mediaSession.metadata.artwork = [
|
|
|
+ { src: "data:image/jpg;base64," + data, sizes: '480x480', type: 'image/jpg' }
|
|
|
+ ]
|
|
|
+ }
|
|
|
+ }
|
|
|
+ resizeQuickAdjust();
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function nextSong(id = false,forcePlayEvenStopped = false){
|
|
|
+ //Check if the current playlist has more than one songs.
|
|
|
+ var currentPaused = audioElement[0].paused;
|
|
|
+ if (playingList.length > 1){
|
|
|
+ var nextSongIndex = playingFileDetail[0];
|
|
|
+ if (id != false){
|
|
|
+ //Allow manual overwrite for this function
|
|
|
+ nextSongIndex = id;
|
|
|
+ var nextSong = playingList[nextSongIndex];
|
|
|
+ }else{
|
|
|
+ if (playingFileDetail[0] >= playingList.length){
|
|
|
+ var nextSong = playingList[0]; //Back to the first song in list
|
|
|
+ nextSongIndex = 0;
|
|
|
+ }else{
|
|
|
+ var nextSong = playingList[nextSongIndex]; //Index of the playList + 1 as the index for the files is alrady i+1. (User count from one)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ //The songs has to be play via background services instead of relying on DOM
|
|
|
+ var filepath = nextSong[0];
|
|
|
+ var rawname = nextSong[1];
|
|
|
+ var displayName = ao_module_codec.decodeUmFilename(rawname);
|
|
|
+ var info = nextSong[2] + " / " + nextSong[3];
|
|
|
+ var id = nextSongIndex + 1;
|
|
|
+ updateMiniPlayerUI(displayName,info,id);
|
|
|
+
|
|
|
+ //Seperate out the filepath without media server path
|
|
|
+ var fileVpath = filepath.split("=")[1];
|
|
|
+ updateMainPlayerUI(displayName,info,id, fileVpath);
|
|
|
+ if (forcePlayEvenStopped){
|
|
|
+ loadAndPlayAudioFile(filepath,true);
|
|
|
+ }else{
|
|
|
+ loadAndPlayAudioFile(filepath,!currentPaused);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (currentPlaying){
|
|
|
+ setPlaying(true);
|
|
|
+ }
|
|
|
+ playingFileDetail = [id,filepath];
|
|
|
+ //Need not to update the playlist because it is the same
|
|
|
+ if (pagingEnabled){
|
|
|
+ var targetPageNumber = getPageNumberByPlaybackId(parseInt(id));
|
|
|
+ switchToPage(targetPageNumber, function(){
|
|
|
+ parsePlayingSongList();
|
|
|
+ highLightPlayingMusic();
|
|
|
+ });
|
|
|
+ }else{
|
|
|
+ highLightPlayingMusic();
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ }else{
|
|
|
+ //This is the only song in playlist. Play again from start
|
|
|
+ audioElement[0].pause();
|
|
|
+ audioElement[0].currentTime = 0;
|
|
|
+ if (!currentPaused){
|
|
|
+ audioElement[0].play();
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+ updateStateReferingURL();
|
|
|
+ }
|
|
|
+
|
|
|
+ function previousSong(){
|
|
|
+ var currentPaused = audioElement[0].paused;
|
|
|
+ if (playingList.length > 1){
|
|
|
+ var nextSongIndex = playingFileDetail[0] - 1;
|
|
|
+ if (nextSongIndex == 0){
|
|
|
+ nextSongIndex = playingList.length;
|
|
|
+ var nextSong = playingList[nextSongIndex - 1]; //Back to the last song in list
|
|
|
+ }else{
|
|
|
+ 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)
|
|
|
+ }
|
|
|
+ //The songs has to be play via background services instead of relying on DOM
|
|
|
+ var filepath = nextSong[0];
|
|
|
+ var rawname = nextSong[1];
|
|
|
+ var displayName = ao_module_codec.decodeUmFilename(rawname);
|
|
|
+ var info = nextSong[2] + " / " + nextSong[3];
|
|
|
+ var id = nextSongIndex;
|
|
|
+ updateMiniPlayerUI(displayName,info,id);
|
|
|
+
|
|
|
+ //Seperate out the filepath without media server path
|
|
|
+ var fileVpath = filepath.split("=")[1];
|
|
|
+ updateMainPlayerUI(displayName,info,id, fileVpath);
|
|
|
+ loadAndPlayAudioFile(filepath,!currentPaused);
|
|
|
+ if (currentPlaying){
|
|
|
+ setPlaying(true);
|
|
|
+ }
|
|
|
+ playingFileDetail = [id,filepath];
|
|
|
+ //Need not to update the playlist because it is the same
|
|
|
+ if (pagingEnabled){
|
|
|
+ var targetPageNumber = getPageNumberByPlaybackId(parseInt(id));
|
|
|
+ switchToPage(targetPageNumber, function(){
|
|
|
+ parsePlayingSongList();
|
|
|
+ highLightPlayingMusic();
|
|
|
+
|
|
|
+ });
|
|
|
+ }else{
|
|
|
+ highLightPlayingMusic();
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ //This is the only song in playlist. Play again from start
|
|
|
+ audioElement[0].pause();
|
|
|
+ audioElement[0].currentTime = 0;
|
|
|
+ audioElement[0].play();
|
|
|
+ }
|
|
|
+ updateStateReferingURL();
|
|
|
+ }
|
|
|
+
|
|
|
+ var originalVol = 0;
|
|
|
+ function togglePlayMode(){
|
|
|
+ if (currentPlaying){
|
|
|
+ originalVol = audioElement[0].volume;
|
|
|
+ fadeAudio();
|
|
|
+ setPlaying(false);
|
|
|
+ }else{
|
|
|
+ gainAudio();
|
|
|
+ setPlaying(true);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ let audioTransitioning = false;
|
|
|
+ function fadeAudio(){
|
|
|
+ audioTransitioning = true;
|
|
|
+ if(audioElement[0].volume > 0){
|
|
|
+ var testval = audioElement[0].volume - 0.1;
|
|
|
+ audioElement[0].volume = Math.max(0,testval);
|
|
|
+ setTimeout(fadeAudio, 50);
|
|
|
+ }else{
|
|
|
+ audioElement[0].pause();
|
|
|
+ setTimeout(function(){
|
|
|
+ audioElement[0].volume = originalVol;
|
|
|
+ updateStateReferingURL();
|
|
|
+ audioTransitioning = false;
|
|
|
+ }, 100);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ var targetVol = 0;
|
|
|
+ function gainAudio(){
|
|
|
+ targetVol = audioElement[0].volume;
|
|
|
+ audioElement[0].volume = 0;
|
|
|
+ setTimeout(function(){
|
|
|
+ audioElement[0].play();
|
|
|
+ recursiveGainAudio();
|
|
|
+ }, 100);
|
|
|
+ }
|
|
|
+
|
|
|
+ function recursiveGainAudio(){
|
|
|
+ audioTransitioning = true;
|
|
|
+ if(audioElement[0].volume < targetVol){
|
|
|
+ if (audioElement[0].volume + 0.1 < 1){
|
|
|
+ audioElement[0].volume += 0.1
|
|
|
+ }else{
|
|
|
+ //Audio volume larger than 1, make it 1
|
|
|
+ audioElement[0].volume = 1
|
|
|
+ }
|
|
|
+ setTimeout(recursiveGainAudio, 50);
|
|
|
+ }else{
|
|
|
+ setTimeout(function(){
|
|
|
+ audioElement[0].volume = targetVol;
|
|
|
+ updateStateReferingURL();
|
|
|
+ audioTransitioning = false;
|
|
|
+ }, 100);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //Update all icons on play and pause buttons
|
|
|
+ function setPlaying(playing){
|
|
|
+ if (playing == true){
|
|
|
+ $(".PlayButton").attr("class","pause icon whiteFont PlayButton");
|
|
|
+ currentPlaying = true;
|
|
|
+ }else{
|
|
|
+ $(".PlayButton").attr("class","play icon whiteFont PlayButton");
|
|
|
+ currentPlaying = false;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadAndPlayAudioFile(sourceURL,playAfterLoad = true){
|
|
|
+ var audio = audioElement;
|
|
|
+
|
|
|
+ //These was added to fix the legacy file listing issue
|
|
|
+ var fd = sourceURL.split("=");
|
|
|
+ var rootPath = fd.shift();
|
|
|
+ var filename = fd.join("=");
|
|
|
+ var playbackURL = "../" + rootPath + "=" + encodeURIComponent(filename); //Convert absolute dir to relative
|
|
|
+ var ext = filename.split(".").pop();
|
|
|
+ if (dbExists){
|
|
|
+ cacheDbTx = cacheDb.transaction("files","readwrite");
|
|
|
+ cacheStore = cacheDbTx.objectStore("files");
|
|
|
+
|
|
|
+ let retrvReq = cacheStore.get(filename);
|
|
|
+ retrvReq.onsuccess = (event) => {
|
|
|
+ let cachefile = retrvReq.result;
|
|
|
+ if (cachefile == undefined || cachefile.content == undefined){
|
|
|
+ //No cache. Play from streaming and cache this
|
|
|
+ $("#mainAudioPlayer").attr("src", playbackURL);
|
|
|
+ startPlaybackAfterAudioLoaded(audio, playAfterLoad);
|
|
|
+
|
|
|
+ if (ext == "mp3"){
|
|
|
+ loadAudioFileURLAsBlob(playbackURL, function(fileBlob){
|
|
|
+ //Store the blob into indexDB
|
|
|
+ let cacheDbTx2 = cacheDb.transaction("files","readwrite");
|
|
|
+ let cacheStore2 = cacheDbTx2.objectStore("files");
|
|
|
+ cacheStore2.put({
|
|
|
+ "filename": filename,
|
|
|
+ "cachetime": parseInt(Date.now() / 1000),
|
|
|
+ "content": fileBlob
|
|
|
+ });
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ }else{
|
|
|
+ //Cache exists. Load this instead
|
|
|
+ console.log("Loaded from cache ", filename)
|
|
|
+ let reader = new FileReader();
|
|
|
+ reader.onload = function(e) {
|
|
|
+ let srcUrl = e.target.result;
|
|
|
+ $("#mainAudioPlayer").attr("src", srcUrl);
|
|
|
+ startPlaybackAfterAudioLoaded(audio, playAfterLoad);
|
|
|
+ };
|
|
|
+ reader.readAsDataURL(cachefile.content);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ //IndexedDB not found
|
|
|
+ $("#mainAudioPlayer").attr("src", playbackURL);
|
|
|
+ startPlaybackAfterAudioLoaded(audio, playAfterLoad);
|
|
|
+ }
|
|
|
+
|
|
|
+ loadThumbnail(sourceURL);
|
|
|
+ }
|
|
|
+
|
|
|
+ function startPlaybackAfterAudioLoaded(audio, playAfterLoad=true){
|
|
|
+ audio[0].pause();
|
|
|
+ audio[0].load();
|
|
|
+ if(playAfterLoad){
|
|
|
+ audio[0].oncanplaythrough = audio[0].play();
|
|
|
+ }else{
|
|
|
+ var currentTime = audioElement[0].currentTime;
|
|
|
+ var duration = audioElement[0].duration;
|
|
|
+ $('#audioProgressBar').css('width','0%');
|
|
|
+ updatePlaybackDisplayTime(currentTime,duration);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function updateMiniPlayerUI(displayName, fileinfo,id){
|
|
|
+ $("#miniPlayerDisplayName").text(displayName);
|
|
|
+ $("#miniPlayerInformation").text(fileinfo);
|
|
|
+ $("#miniPlayerIDtab").text(id + "/" + totalMusicCount);
|
|
|
+ }
|
|
|
+
|
|
|
+ function updateMainPlayerUI(songname, fileinfo, id, filepath){
|
|
|
+ $("#mainPlayerSongTitle").text(songname);
|
|
|
+ $("#mainPlayerSongDesc").text(fileinfo);
|
|
|
+ $("#mainPlayerMiniTab").text(id + "/" + totalMusicCount);
|
|
|
+ if ('mediaSession' in navigator){
|
|
|
+ var infoRewrite = fileinfo.split(" / ")
|
|
|
+ updateTitle(songname, infoRewrite[1] + " (" + infoRewrite[0] + ")", id + "/" + totalMusicCount, filepath);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ /*
|
|
|
+ Playlist related functions
|
|
|
+
|
|
|
+ */
|
|
|
+
|
|
|
+ function renderPlaylistByName(listname){
|
|
|
+ $("#interfaceTitle").text(listname);
|
|
|
+ ao_module_agirun("Music/functions/playlist.js", {
|
|
|
+ opr: "list",
|
|
|
+ playlistname: listname
|
|
|
+ }, function(data){
|
|
|
+ $("#mainList").html("");
|
|
|
+ if (data.error !== undefined){
|
|
|
+ //This playlist no longer exists. Back to main
|
|
|
+ loadPlaylistView();
|
|
|
+ }else{
|
|
|
+ //Updat ethe global values
|
|
|
+ displayList = data;
|
|
|
+ totalMusicCount = data.length;
|
|
|
+ currentPath = listname;
|
|
|
+ //Add the back btn for back to playlist view
|
|
|
+ $("#mainList").append(`<div class="mainList item extrapadding" onClick="loadPlaylistView();">
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;">
|
|
|
+ <i class="angle left icon whiteFont" style="overflow:hidden;"></i>
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
|
|
|
+ ../
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </div>`);
|
|
|
+
|
|
|
+
|
|
|
+ //List the resulting song list
|
|
|
+ for (var i = 0; i < data.length; i++){
|
|
|
+ var songInfo = data[i];
|
|
|
+
|
|
|
+ $("#mainList").append(`<div class="mainList file item" filepath=${ao_module_utils.objectToAttr(songInfo[0])} id=${i+1} rawname=${ao_module_utils.objectToAttr(songInfo[1])}>
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">
|
|
|
+ <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">
|
|
|
+ ${songInfo[1]}
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">${songInfo[2]} / ${songInfo[3]}</div>
|
|
|
+ </div>
|
|
|
+ </div>\
|
|
|
+ <div class="topRightCorner" align="center">
|
|
|
+ ${i + 1}
|
|
|
+ </div>
|
|
|
+ <div class="mainList rightFunctionBar" type="file" align="center" onclick="showMore(this);">
|
|
|
+ <i class="ellipsis vertical icon" style="margin-top:1.2em;"></i>
|
|
|
+ </div>
|
|
|
+ </div>`);
|
|
|
+
|
|
|
+ //Update the interface Detail
|
|
|
+ $("#interfaceDetails").text("[" + data.length + " files]");
|
|
|
+ };
|
|
|
+
|
|
|
+
|
|
|
+ loadThumbnailToMusicList();
|
|
|
+ //Fix some legacy css isseus
|
|
|
+ $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ //Open a given playlist
|
|
|
+ function openPlaylist(object){
|
|
|
+ var listname = $(object).attr("listname");
|
|
|
+ renderPlaylistByName(listname);
|
|
|
+ }
|
|
|
+
|
|
|
+ function addToNewPlaylist(){
|
|
|
+ var playlistname = prompt("Enter new playlist name");
|
|
|
+ if (playlistname != null && playlistname != ""){
|
|
|
+ //Add to playlist
|
|
|
+ addSongToPlayList(playlistname);
|
|
|
+ }else{
|
|
|
+ $("#succSnackBar").find(".content").html(`<i class="remove icon"></i> Invalid playlist name`);
|
|
|
+ $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
|
|
|
+
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function addToPlaylistFromMoreMenu(){
|
|
|
+ $(".showMoreMenus").fadeOut('fast');
|
|
|
+ $("#showmoreUIcover").fadeOut('fast');
|
|
|
+ showPlaylistInterface();
|
|
|
+ }
|
|
|
+
|
|
|
+ //Remove the curernt file from the current playlist
|
|
|
+ function removeFromPlylistFromMoreMenu(){
|
|
|
+ if (currentMode == "playlist"){
|
|
|
+ //Remvoe the playlistAddPendingFile
|
|
|
+ //In playlist mode, the currentPath is used to store the plylist name
|
|
|
+ ao_module_agirun("Music/functions/playlist.js", {
|
|
|
+ opr: "remove",
|
|
|
+ playlistname: currentPath,
|
|
|
+ musicpath: playlistAddPendingFile.replace("/media?file=","")
|
|
|
+ }, function(data){
|
|
|
+ if (data.error !== undefined){
|
|
|
+ alert(data.error);
|
|
|
+ }else{
|
|
|
+ //Removed. Reload playlist
|
|
|
+ renderPlaylistByName(currentPath);
|
|
|
+
|
|
|
+ //Hide the MoreMenu
|
|
|
+ hideShowMoreMenu();
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function addSongToSelectedPlaylist(object){
|
|
|
+ var playlistName = $(object).attr("name").trim();
|
|
|
+ addSongToPlayList(playlistName);
|
|
|
+ }
|
|
|
+
|
|
|
+ function addSongToPlayList(listname){
|
|
|
+ //Get current music file name
|
|
|
+ var currentSongPath = "";
|
|
|
+ if (playlistAddPendingFile !== ""){
|
|
|
+ currentSongPath = playlistAddPendingFile.replace("/media?file=","");
|
|
|
+ }else{
|
|
|
+ $("#succSnackBar").find(".content").html(`<i class="remove icon"></i> No song selected`);
|
|
|
+ $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ //Add to playlist
|
|
|
+ ao_module_agirun("Music/functions/playlist.js", {
|
|
|
+ opr: "add",
|
|
|
+ playlistname: listname,
|
|
|
+ musicpath: currentSongPath
|
|
|
+ },function(data){
|
|
|
+ //Show success
|
|
|
+ $("#succSnackBar").find(".content").html(`<i class="checkmark icon"></i> Added to playlist ${listname}`);
|
|
|
+ $("#succSnackBar").slideDown("fast").delay(3000).slideUp("fast");
|
|
|
+
|
|
|
+ //Reload the playlist
|
|
|
+ initPlaylistInterfaceList();
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ //List the number of playlist stored in this database
|
|
|
+ function loadPlaylistView(){
|
|
|
+ currentMode = "playlist";
|
|
|
+ //Clear the main list
|
|
|
+ $("#mainList").html("");
|
|
|
+ togglePagingMode(false);
|
|
|
+ ao_module_agirun("Music/functions/playlist.js", {
|
|
|
+ opr: "root",
|
|
|
+ },function(data){
|
|
|
+ //Get the list of playlist
|
|
|
+ console.log(data);
|
|
|
+
|
|
|
+ //Render the elements
|
|
|
+ data.forEach(playlist => {
|
|
|
+ $("#mainList").append(`<div class="mainList item" listname="${playlist.name}" tag="playlist" onClick="openPlaylist(this);">
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" >
|
|
|
+ <img class="ui small image" src="img/list.svg" style="margin-right: 0.2em;"></img>
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
|
|
|
+ ${playlist.name}
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">[${playlist.count} files]</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="mainList rightFunctionBar" tag="moreInfo" align="center" onClick="openPlaylist(this);">
|
|
|
+ <i class="chevron right icon" style="margin-top:1.2em;"></i>
|
|
|
+ </div>
|
|
|
+ </div>`);
|
|
|
+ });
|
|
|
+ hideLeftMenu();
|
|
|
+
|
|
|
+ //Update the headers
|
|
|
+ $("#interfaceTitle").text("Playlist");
|
|
|
+ $("#AMmenuIcon").attr("class","list icon large whiteFont");
|
|
|
+ $("#interfaceDetails").text("[" + data.length + " playlist]");
|
|
|
+ });
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadFolderView(){
|
|
|
+ currentMode = "folder";
|
|
|
+ currentPath = "";
|
|
|
+ var tempalte = '<div class="mainList item" filepath={filepath} id={id} tag="folder" onClick="openFolder(this);">\
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" >\
|
|
|
+ <img class="ui small image" src="img/fo.svg" style="margin-right: 0.2em;"></img>\
|
|
|
+ <div class="content whiteFont" style="padding-top: 2px; line-height:1em;width : 80%; font-weight: lighter;">\
|
|
|
+ {foldername}\
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">{fileinfo}</div>\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ <div class="mainList rightFunctionBar" tag="moreInfo" align="center" onClick="moreFolder(this);">\
|
|
|
+ <i class="chevron right icon" style="margin-top:1.2em;"></i>\
|
|
|
+ </div>\
|
|
|
+ </div>';
|
|
|
+ $("#interfaceTitle").text("Storage");
|
|
|
+ togglePagingMode(false);
|
|
|
+ $("#AMmenuIcon").attr("class","folder open icon large whiteFont");
|
|
|
+ ao_module_agirun("Music/functions/listSong.js", {
|
|
|
+ listdir: "root",
|
|
|
+ },function(data){
|
|
|
+ $("#mainList").html("");
|
|
|
+ for (var i =0; i < data.length; i++){
|
|
|
+ var folderName = data[i][0];
|
|
|
+ var folderPath = data[i][1];
|
|
|
+ var fileCount = data[i][2];
|
|
|
+ var folderCount = data[i][3];
|
|
|
+ var box = tempalte;
|
|
|
+ box = replaceAll("{filepath}",folderPath,box);
|
|
|
+ box = replaceAll("{id}",i + 1,box);
|
|
|
+
|
|
|
+ if (folderName == ".cache" || folderName == ".trash"){
|
|
|
+ //Hidden folders
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ if (folderPath.includes("/media")){
|
|
|
+ //This is from external storage devices. List its number as well.
|
|
|
+ var tmp = folderPath.split("/");
|
|
|
+ var extStoragePath = "/" + tmp[1] + "/" + tmp[2];
|
|
|
+ box = replaceAll("{foldername}",folderName + " ( " + extStoragePath + " )",box);
|
|
|
+ }else{
|
|
|
+ //This is directory inside normal folders.
|
|
|
+ box = replaceAll("{foldername}",folderName,box);
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ var fileinfo = fileinfo = "[No playable files]";
|
|
|
+ console.log(folderCount, fileCount);
|
|
|
+ if (folderCount > 0 && fileCount > 0){
|
|
|
+ fileinfo = "[" + fileCount + " files, " + folderCount +" folders]"
|
|
|
+ }else if (fileCount > 0){
|
|
|
+ fileinfo = "[" + fileCount + " files]";
|
|
|
+ }else if (fileCount == -1 && folderCount == -1){
|
|
|
+ fileinfo = "";
|
|
|
+ }
|
|
|
+ */
|
|
|
+ var fileinfo = folderPath;
|
|
|
+
|
|
|
+
|
|
|
+ box = replaceAll("{fileinfo}",fileinfo,box);
|
|
|
+ rootPaths.push(folderPath);
|
|
|
+ $("#mainList").append(box);
|
|
|
+ }
|
|
|
+ $("#interfaceDetails").text("[" + data.length + " folders]");
|
|
|
+ });
|
|
|
+ hideLeftMenu();
|
|
|
+ }
|
|
|
+
|
|
|
+ function isMobile(){
|
|
|
+ if( /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) ) {
|
|
|
+ return true
|
|
|
+ }
|
|
|
+ return false
|
|
|
+ }
|
|
|
+
|
|
|
+ function setWindowHash(hashValue){
|
|
|
+ hashValue = JSON.stringify(hashValue);
|
|
|
+
|
|
|
+ if (!isAndroid){
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if(history.pushState) {
|
|
|
+ window.history.pushState(null, null, '#' + hashValue);
|
|
|
+ }else {
|
|
|
+ location.hash = '#' + hashValue;
|
|
|
+ }
|
|
|
+ if(ao_module_virtualDesktop && !isMobile()){
|
|
|
+ //Update the iframe src as well
|
|
|
+ var newsrc = window.frameElement.getAttribute("src");
|
|
|
+ if (newsrc.includes("#")){
|
|
|
+ newsrc = newsrc.split("#")
|
|
|
+ newsrc.pop();
|
|
|
+ newsrc = newsrc.join("#");
|
|
|
+ }
|
|
|
+ newsrc = newsrc + "#" + hashValue;
|
|
|
+ $(window.frameElement).attr("src",newsrc);
|
|
|
+ //console.log(window.frameElement.getAttribute("src"));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function updateStateReferingURL(){
|
|
|
+ var paused = audioElement[0].paused; //Check the current playing state of the player
|
|
|
+ setWindowHash({path: currentPath, pfp:playingFileDetail,pause:paused});
|
|
|
+ }
|
|
|
+
|
|
|
+ function openFolder(object, playParameters=null){
|
|
|
+ var playIDAfterOpen = -1;
|
|
|
+ var startPaused = false;
|
|
|
+ if (playParameters !== null){
|
|
|
+ playIDAfterOpen = playParameters.playIDAfterOpen;
|
|
|
+ startPaused = playParameters.startPaused;
|
|
|
+ }
|
|
|
+ var backbtnTemplate = '<div class="mainList item extrapadding" filepath={filepath} id={id} onClick="openFolder(this);">\
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;">\
|
|
|
+ <i class="angle left icon whiteFont" style="overflow:hidden;"></i>\
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">\
|
|
|
+ ../\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ </div>';
|
|
|
+
|
|
|
+ var folderTemplate = '<div class="mainList item" filepath={filepath} tag="folder" id={id} onClick="openFolder(this);">\
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;">\
|
|
|
+ <img class="ui small image" src="img/fo.svg" style="margin-right: 0.2em;"></img>\
|
|
|
+ <div class="content whiteFont" style="padding-top:2px;line-height:1em; width:80%; font-weight: lighter;">\
|
|
|
+ {foldername}\
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">{fileinfo}</div>\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ <div class="mainList rightFunctionBar" type="folder" align="center">\
|
|
|
+ <i class="chevron right icon" style="margin-top:1.2em;"></i>\
|
|
|
+ </div>\
|
|
|
+ </div>';
|
|
|
+ var fileTemplate = `<div class="mainList file item" filepath={filepath} id={id} rawname={rawname}>\
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">\
|
|
|
+ <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
+ <!-- <i class="music icon whiteFont" style="overflow:hidden;"></i> -->\
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">\
|
|
|
+ {songtitle}\
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">{ext} / {size}</div>\
|
|
|
+ </div>\
|
|
|
+ </div>\
|
|
|
+ <div class="topRightCorner" align="center">\
|
|
|
+ {id}\
|
|
|
+ </div>\
|
|
|
+ <div class="mainList rightFunctionBar" type="file" align="center">\
|
|
|
+ <i class="ellipsis vertical icon" style="margin-top:1.2em;"></i>\
|
|
|
+ </div>\
|
|
|
+ </div>`;
|
|
|
+
|
|
|
+ var targetPath = decodeURIComponent($(object).attr("filepath"));
|
|
|
+ if (targetPath == "root" || targetPath == "/"){
|
|
|
+ //User request to go back to storage root
|
|
|
+ loadFolderView();
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ currentPath = targetPath.split("/./").join("/");
|
|
|
+ currentPath = currentPath.split("//").join("/");
|
|
|
+ $("#interfaceTitle").text(ao_module_codec.decodeHexFoldername(targetPath.substring(0,targetPath.length -1).split("/").pop()));
|
|
|
+ var parentDir = currentPath.split("/");
|
|
|
+ parentDir.pop();parentDir.pop();
|
|
|
+ parentDir = (parentDir.join("/") + "/");
|
|
|
+ //console.log(currentPath);
|
|
|
+ var backbtn;
|
|
|
+ if (rootPaths.includes(currentPath) == false){
|
|
|
+ //We are currently not at root
|
|
|
+ backbtn = backbtnTemplate;
|
|
|
+ backbtn = replaceAll("{filepath}",encodeURIComponent(parentDir),backbtn);
|
|
|
+ backbtn = replaceAll("{id}",0,backbtn);
|
|
|
+ }else{
|
|
|
+ //Create a back button to go back to root of storage
|
|
|
+ backbtn = backbtnTemplate;
|
|
|
+ backbtn = replaceAll("{filepath}","root",backbtn);
|
|
|
+ backbtn = replaceAll("{id}",0,backbtn);
|
|
|
+ }
|
|
|
+
|
|
|
+ //Request the server for a list of folders and file in this directory
|
|
|
+ ao_module_agirun("Music/functions/listSong.js", {
|
|
|
+ listdir: targetPath.split("//").join("/"),
|
|
|
+ },function(data){
|
|
|
+ $("#mainList").html("");
|
|
|
+ $("#mainList").append(backbtn);
|
|
|
+ if (Array.isArray(data)){
|
|
|
+ var folders = data[0];
|
|
|
+ var files = data[1];
|
|
|
+ //List all folders
|
|
|
+ for (var i =0; i < folders.length; i++){
|
|
|
+ var folderName = folders[i][0];
|
|
|
+ var folderPath = folders[i][1];
|
|
|
+ var fileCount = folders[i][2];
|
|
|
+ var folderCount = folders[i][3];
|
|
|
+ var box = folderTemplate;
|
|
|
+ if (folderName == ".cache" || folderName == ".trash"){
|
|
|
+ //Hidden folders
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ box = replaceAll("{filepath}",encodeURIComponent(folderPath),box);
|
|
|
+ box = replaceAll("{id}","folder-" + i + 1,box);
|
|
|
+ box = replaceAll("{foldername}",ao_module_codec.decodeHexFoldername(folderName),box);
|
|
|
+ var fileinfo = "[" + fileCount + " files]"
|
|
|
+ if (folderCount > 0){
|
|
|
+ fileinfo = "[" + fileCount + " files, " + folderCount +" folders]"
|
|
|
+ }else if (folderCount == -1){
|
|
|
+ fileinfo = folderPath;
|
|
|
+ }
|
|
|
+ box = replaceAll("{fileinfo}",fileinfo,box);
|
|
|
+ $("#mainList").append(box);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ //List all files
|
|
|
+ for ( var i =0; i < files.length; i++){
|
|
|
+ var songInfo = files[i];
|
|
|
+ var path = songInfo[0];
|
|
|
+ var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
|
|
|
+ var ext = songInfo[2];
|
|
|
+ var size = songInfo[3];
|
|
|
+ var box = fileTemplate;
|
|
|
+ box = replaceAll("{filepath}",ao_module_utils.objectToAttr(path),box);
|
|
|
+ box = replaceAll("{id}",i + 1,box); //User count from 1
|
|
|
+ box = replaceAll("{rawname}",ao_module_utils.objectToAttr(songInfo[1]),box);
|
|
|
+ box = replaceAll("{songtitle}",displayname,box);
|
|
|
+ box = replaceAll("{ext}",ext,box);
|
|
|
+ box = replaceAll("{size}",size,box);
|
|
|
+ $("#mainList").append(box);
|
|
|
+ }
|
|
|
+ displayList = files;
|
|
|
+ totalMusicCount = files.length;
|
|
|
+ $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
+
|
|
|
+ }else{
|
|
|
+ //Something went wrong
|
|
|
+ //console.log('[AirMusic] Something went wrong: ' + data);
|
|
|
+ loadFolderView();
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ if (folders.length > 0 && files.length > 0){
|
|
|
+ $("#interfaceDetails").text("[" + files.length + " files, " + folders.length+ " folders]");
|
|
|
+ }else if (folders.length > 0){
|
|
|
+ $("#interfaceDetails").text("[" + folders.length + " folders]");
|
|
|
+ }else if (files.length > 0){
|
|
|
+ $("#interfaceDetails").text("[" + files.length + " files]");
|
|
|
+ }else{
|
|
|
+ $("#interfaceDetails").text("[0 files or folders]");
|
|
|
+ }
|
|
|
+ //Hook all the events for the moreinfo on folders
|
|
|
+ hookMoreFolderInfoClickEvent();
|
|
|
+
|
|
|
+ //Check if the current playing file is located inside this list.
|
|
|
+ highLightPlayingMusic();
|
|
|
+
|
|
|
+ //Check if auto playback is required. If yes, play it with the given filepath.
|
|
|
+ if (playIDAfterOpen != -1){
|
|
|
+ $(".mainList.item").each(function(){
|
|
|
+ if ($(this).attr("id") == playIDAfterOpen){
|
|
|
+ //This is the file that require to playback after the folder loaded
|
|
|
+ playSong($(this).find(".ts.header.selectable"),startPaused);
|
|
|
+ setTimeout(resizeQuickAdjust,500);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ //Load thumbnail for song in list
|
|
|
+ loadThumbnailToMusicList();
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function moreInfo(object){
|
|
|
+ if ($(object).parent().attr('tag') == "folder"){
|
|
|
+ //This is a folder. Ask if play as playlist
|
|
|
+ alert("Folder playback work in progress");
|
|
|
+ }else{
|
|
|
+ showMore(object);
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function hookMoreFolderInfoClickEvent(){
|
|
|
+ $(".mainList.rightFunctionBar").on("click", function(e){
|
|
|
+ e.stopPropagation();
|
|
|
+ moreInfo(this);
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function highLightPlayingMusic(){
|
|
|
+ //This function is used for highlighting the current playing music if the music piece is found inside the current list
|
|
|
+ $(".mainList.item").removeClass("playingTrack");
|
|
|
+ $(".mainList.item").each(function(){
|
|
|
+ var id = parseInt($(this).attr("id"));
|
|
|
+ try{
|
|
|
+ var filepath = ao_module_utils.attrToObject($(this).attr("filepath"));
|
|
|
+ }catch{
|
|
|
+ //Use back the previous method of filepath storage for compatibility
|
|
|
+ var filepath = $(this).attr("filepath");
|
|
|
+ }
|
|
|
+
|
|
|
+ //Id is ignored in the 9/9/2019 updates and only check if the path matches.
|
|
|
+ if (filepath == playingFileDetail[1]){
|
|
|
+ $(this).addClass("playingTrack");
|
|
|
+ }
|
|
|
+ //console.log([id,filepath], playingFileDetail);
|
|
|
+ });
|
|
|
+
|
|
|
+ //Update dropdownList as well if exists
|
|
|
+ $(".dropdownList.item").removeClass("playingTrack");
|
|
|
+ $(".dropdownList.item").each(function(){
|
|
|
+ try{
|
|
|
+ var filepath = ao_module_utils.attrToObject($(this).attr("filepath"));
|
|
|
+ }catch{
|
|
|
+ //Use back the previous method of filepath storage for compatibility
|
|
|
+ var filepath = $(this).attr("filepath");
|
|
|
+ }
|
|
|
+ //var filepath = $(this).attr("filepath");
|
|
|
+ if (filepath == playingFileDetail[1]){
|
|
|
+ $(this).addClass("playingTrack");
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function hideShowMoreMenu(){
|
|
|
+ $(".showMoreMenus").fadeOut('fast');
|
|
|
+ $("#showmoreUIcover").fadeOut('fast');
|
|
|
+
|
|
|
+ //Return the playlistAddPendingFile object back to the playing one
|
|
|
+ if (playingFileDetail !== undefined){
|
|
|
+ playlistAddPendingFile = playingFileDetail[1];
|
|
|
+ }else{
|
|
|
+ //No song is being play back
|
|
|
+ playlistAddPendingFile = "";
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function playFromShowMoreMenu(){
|
|
|
+ nextSong(showMoreOprPlayID - 1);
|
|
|
+ $(".showMoreMenus").fadeOut('fast');
|
|
|
+ }
|
|
|
+
|
|
|
+ function showFileInfo(){
|
|
|
+ $("#showFileInfo").show();
|
|
|
+ $("#showMoreUI").hide();
|
|
|
+ }
|
|
|
+
|
|
|
+ function startRelatedSearch(){
|
|
|
+ enterSearchMode();
|
|
|
+ $("#searchInputBar").val(showMoreoprDisplayName);
|
|
|
+ $("#searchInputBar").focus();
|
|
|
+ hideShowMoreMenu();
|
|
|
+ }
|
|
|
+
|
|
|
+ function searchYoutubeViaShowMore(){
|
|
|
+ if (showMoreoprDisplayName != "" || showMoreoprDisplayName != undefined){
|
|
|
+ var filename = showMoreoprDisplayName;
|
|
|
+ searchOnYoutube(filename);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ var showMoreOprPlayID,showMoreOprFilepath, showMoreoprDisplayName;
|
|
|
+ function showMore(object){
|
|
|
+ var filepath = $(object).parent().attr("filepath");
|
|
|
+ filepath = JSON.parse(decodeURIComponent(filepath));
|
|
|
+ //User want more action on this file. assume this is the file to add
|
|
|
+ playlistAddPendingFile = filepath;
|
|
|
+ var id = $(object).parent().attr('id');
|
|
|
+ var rawname = $(object).parent().attr("rawname");
|
|
|
+ var displayName = JSON.parse(decodeURIComponent(ao_module_codec.decodeUmFilename(rawname)));
|
|
|
+ //Update global variable for quick operations
|
|
|
+ showMoreOprPlayID = id;
|
|
|
+ showMoreOprFilepath = filepath;
|
|
|
+ showMoreoprDisplayName = displayName;
|
|
|
+ $("#showMoreUI").find(".songTitle").text(displayName);
|
|
|
+ $("#showMoreUI").find(".songID").text(id);
|
|
|
+ $("#showFileInfo").find(".songTitle").text(displayName);
|
|
|
+ $("#showMoreUI").fadeIn('fast');
|
|
|
+ $("#showmoreUIcover").fadeIn('fast');
|
|
|
+ //Update fileinformation as well
|
|
|
+ if (filepath.substring(0, 12) == "/media?file="){
|
|
|
+ filepath = filepath.substring(12);
|
|
|
+ }
|
|
|
+
|
|
|
+ //Pre-render the file info
|
|
|
+ ao_module_agirun("Music/functions/getFileInfo.js", {
|
|
|
+ filepath: filepath,
|
|
|
+ }, function(data){
|
|
|
+ $("#showFileInfo").find(".filename").text(ao_module_codec.decodeUmFilename(data[0]));
|
|
|
+ $("#showFileInfo").find(".rawname").text(data[0]);
|
|
|
+ $("#showFileInfo").find(".filepath").text(data[1]);
|
|
|
+ $("#showFileInfo").find(".filesize").text(data[2] + " (" + data[3] + " Bytes)");
|
|
|
+ $("#showFileInfo").find(".filedate").text(data[4]);
|
|
|
+ });
|
|
|
+
|
|
|
+ if (currentMode != "playlist"){
|
|
|
+ $(".playlistonly").hide();
|
|
|
+ }else{
|
|
|
+ $(".playlistonly").show();
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function playSongFromDropdownList(object){
|
|
|
+ //Play song from dropdown list. Hence, no need to update dropdownList
|
|
|
+ if ($(object).parent().hasClass("playingTrack")){
|
|
|
+ //This song already playing. Ignore play request.
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (pagingEnableForPlayingList && $(object).attr("targetPage") != currentPlayingPage){
|
|
|
+ let newPage = parseInt($(object).attr("targetPage"));
|
|
|
+ if (!isNaN(newPage)){
|
|
|
+ currentPlayingPage = newPage;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ playSong(object);
|
|
|
+ //Check if the song is also in main list. If yes, highlight it as well
|
|
|
+ /*
|
|
|
+ $(".mainList.item").each(function(){
|
|
|
+ if ($(this).attr("filepath") == $(object).parent().attr("filepath")){
|
|
|
+ $(".mainList.item.playingTrack").removeClass("playingTrack");
|
|
|
+ $(this).addClass("playingTrack");
|
|
|
+ }
|
|
|
+ });
|
|
|
+ */
|
|
|
+ $("#dropdownSonglist").delay(500).slideUp();
|
|
|
+ highLightPlayingMusic();
|
|
|
+ }
|
|
|
+
|
|
|
+ //Move the playing song list into the dropdown song list. This list might not be the same as the one in the main list.
|
|
|
+ function parsePlayingSongList(){
|
|
|
+ $("#currentPlayingMainList").html("");
|
|
|
+ var counter = playingList.length;
|
|
|
+ var totalSize = 0.0;
|
|
|
+ var renderRange = [0, playingList.length];
|
|
|
+ var pageKeepRange = [0, playingList.length]
|
|
|
+ if (pagingEnableForPlayingList){
|
|
|
+ //Paging Enabled. Only show 1/2 max + current page song + 1/2 max
|
|
|
+ var startEndRange = getPageStartAndEndByPageNumber(currentPlayingPage, playingList);
|
|
|
+ startEndRange = [startEndRange[0] - pagingCutoffCount /2, startEndRange[1] + pagingCutoffCount / 2]
|
|
|
+ //console.log("DEFAULT START END RANGE", startEndRange)
|
|
|
+ //page keep range is the middle 1/2 number of songs, [1/4 => prev page][1/2][1/4 => next page]
|
|
|
+ pageKeepRange[0] = startEndRange[0] + pagingCutoffCount / 2;
|
|
|
+ pageKeepRange[1] = startEndRange[1] - pagingCutoffCount / 2;
|
|
|
+
|
|
|
+ if (startEndRange[0] < 0){
|
|
|
+ startEndRange[0] = 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (startEndRange[1] > playingList.length){
|
|
|
+ startEndRange[1] = playingList.length;
|
|
|
+ }
|
|
|
+
|
|
|
+ renderRange = startEndRange;
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ if (pageKeepRange[0] < 0){
|
|
|
+ pageKeepRange[0] = 0;
|
|
|
+ }
|
|
|
+ if (pageKeepRange[1] > playingList.length){
|
|
|
+ pageKeepRange[1] = playingList.length;
|
|
|
+ }
|
|
|
+ //console.log("KEEP RANGE", pageKeepRange);
|
|
|
+ }
|
|
|
+
|
|
|
+ for (var i = renderRange[0]; i < renderRange[1]; i++){
|
|
|
+ var displayname = ao_module_codec.decodeUmFilename(playingList[i][1]);
|
|
|
+ if (playingList[i][3].includes("MB")){
|
|
|
+ totalSize += parseFloat(playingList[i][3].split(" ")[0]);
|
|
|
+ }else if (playingList[i][3].includes("KB")){
|
|
|
+ totalSize += parseFloat(playingList[i][3].split(" ")[0]) / 1000;
|
|
|
+ }else if (playingList[i][3].includes("GB")){
|
|
|
+ totalSize += parseFloat(playingList[i][3].split(" ")[0]) * 1000;
|
|
|
+ }
|
|
|
+
|
|
|
+ //Get thumbnail from the current list to prevent re-loading form remote
|
|
|
+ let targetThumbnail = $("#" + (i + 1)).find("img").attr("src");
|
|
|
+ let opacityShowing = "";
|
|
|
+ let realVpath = playingList[i][0].split("=");
|
|
|
+ realVpath.shift();
|
|
|
+ realVpath = realVpath.join("=");
|
|
|
+
|
|
|
+ if ($("#" + (i + 1)).length == 0 && displayList.length == playingList.length){
|
|
|
+ //Item not exists
|
|
|
+ targetThumbnail = "img/eq.svg";
|
|
|
+ opacityShowing = "opacity: 0.6;"
|
|
|
+ }else if (pagingEnableForPlayingList){
|
|
|
+ if (i < pageKeepRange[0] || i > pageKeepRange[1]){
|
|
|
+ //Out of paging range. Do hidden display
|
|
|
+ targetThumbnail = "img/eq.svg";
|
|
|
+ opacityShowing = "opacity: 0.6;"
|
|
|
+ }else{
|
|
|
+ targetThumbnail = `../system/file_system/loadThumbnail?bytes=true&vpath=${encodeURIComponent(realVpath)}`;
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ //Non paging list. Always do all thumbnails.
|
|
|
+ targetThumbnail = `../system/file_system/loadThumbnail?bytes=true&vpath=${encodeURIComponent(realVpath)}`;
|
|
|
+ }
|
|
|
+
|
|
|
+ let targetPage = currentPlayingPage;
|
|
|
+ if (i < pageKeepRange[0]){
|
|
|
+ targetPage -= 1;
|
|
|
+ }
|
|
|
+ if (i > pageKeepRange[1]){
|
|
|
+ targetPage += 1;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ $("#currentPlayingMainList").append(`<div class="dropdownList file item" style="${opacityShowing}" filepath="${ao_module_utils.objectToAttr(playingList[i][0])}" listid="${i + 1}" rawname="${ao_module_utils.objectToAttr(playingList[i][1])}">
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" onClick="playSongFromDropdownList(this);" targetPage="${targetPage}">
|
|
|
+ <img class="ui small image" src="${targetThumbnail}" onError="replaceImageToDefault(this);" style="margin-right: 0.2em;"></img>
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%;">
|
|
|
+ <span style="font-weight: lighter;">${displayname}</span>
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">${playingList[i][2]} / ${playingList[i][3]}</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="topRightCorner" align="center">
|
|
|
+ ${i + 1}
|
|
|
+ </div>
|
|
|
+ </div>`);
|
|
|
+ $("#dropdownListSongCount").text(counter);
|
|
|
+ $("#dropdownListIDCount").text(parseInt(totalSize) + "");
|
|
|
+ }
|
|
|
+ $(".dropdownList.item").each(function(){
|
|
|
+ if ($(this).attr('filepath') == playingFileDetail[1]){
|
|
|
+ $(this).addClass("playingTrack");
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadAudioFileURLAsBlob(url, callback){
|
|
|
+ var xhr = new XMLHttpRequest();
|
|
|
+ xhr.open('GET', url);
|
|
|
+ xhr.responseType = 'blob';
|
|
|
+ xhr.onload = function(e){
|
|
|
+ console.log("Buffer completed: ", url);
|
|
|
+ callback(xhr.response);
|
|
|
+ }
|
|
|
+ xhr.send();
|
|
|
+ }
|
|
|
+
|
|
|
+ //Clear all cached audio files
|
|
|
+ function clearAllCache(){
|
|
|
+ let dbtx = cacheDb.transaction("files","readwrite");
|
|
|
+ let fstore = dbtx.objectStore("files");
|
|
|
+ var clearReq = fstore.clear();
|
|
|
+ clearReq.onsuccess = function(e){
|
|
|
+ console.log("All cache file cleared");
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function getAllItems(callback) {
|
|
|
+ var trans = cacheDb.transaction("files", IDBTransaction.READ_ONLY);
|
|
|
+ var store = trans.objectStore("files");
|
|
|
+ var items = [];
|
|
|
+ trans.oncomplete = function(evt) {
|
|
|
+ callback(items);
|
|
|
+ };
|
|
|
+ var cursorRequest = store.openCursor();
|
|
|
+ cursorRequest.onerror = function(error) {
|
|
|
+ console.log(error);
|
|
|
+ };
|
|
|
+ cursorRequest.onsuccess = function(evt) {
|
|
|
+ var cursor = evt.target.result;
|
|
|
+ if (cursor) {
|
|
|
+ items.push(cursor.value);
|
|
|
+ cursor.continue();
|
|
|
+ }
|
|
|
+ };
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ function replaceImageToDefault(target){
|
|
|
+ $(target).attr('src', "img/eq.svg");
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadSongList(type = "all"){
|
|
|
+ currentPath = "";
|
|
|
+ currentMode = "music";
|
|
|
+
|
|
|
+ $("#interfaceTitle").text("Music");
|
|
|
+ $("#AMmenuIcon").attr("class","music icon large whiteFont")
|
|
|
+
|
|
|
+ ao_module_agirun("Music/functions/listSong.js", {
|
|
|
+ listSong: type,
|
|
|
+ },function(data){
|
|
|
+ //Initialize the song list
|
|
|
+ displayList = data;
|
|
|
+ if (type == "all"){
|
|
|
+ //Caching is used in here
|
|
|
+ displayList = data.list;
|
|
|
+ if (data.cached == true){
|
|
|
+ //This is a cached version of the music list.
|
|
|
+ console.log("Updating cached song list");
|
|
|
+ ao_module_agirun("Music/functions/buildCache.js", {}, function(data){
|
|
|
+ console.log("Cache updated: ", data);
|
|
|
+ });
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //Updates 2022-07-12: Check if the list is too long. If yes, use paging
|
|
|
+ let renderRange = [0,displayList.length];
|
|
|
+
|
|
|
+ if (displayList.length > pagingCutoffCount){
|
|
|
+ togglePagingMode(true);
|
|
|
+ renderRange = [0, pagingCutoffCount];
|
|
|
+ }else{
|
|
|
+ togglePagingMode(false);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (playingList == []){
|
|
|
+ playingList = Array.from(displayList);
|
|
|
+ pagingEnableForPlayingList = pagingEnabled;
|
|
|
+ currentPlayingPage = currentPage;
|
|
|
+ }
|
|
|
+
|
|
|
+ renderDisplayList(displayList,renderRange[0], renderRange[1] );
|
|
|
+
|
|
|
+ }, function(){
|
|
|
+ alert("Failed to access listSong API with type: " + type)
|
|
|
+ });
|
|
|
+ hideLeftMenu();
|
|
|
+ }
|
|
|
+
|
|
|
+ function renderDisplayList(displayList, start, end, callback=undefined){
|
|
|
+ $("#mainList").html("");
|
|
|
+ for ( var i =start; i < end; i++){
|
|
|
+ var songInfo = displayList[i];
|
|
|
+ var path = songInfo[0];
|
|
|
+ var displayname = ao_module_codec.decodeUmFilename(songInfo[1]);
|
|
|
+ var ext = songInfo[2];
|
|
|
+ var size = songInfo[3];
|
|
|
+ $("#mainList").append(`<div class="mainList file item" filepath="${ao_module_utils.objectToAttr(path)}" id="${i + 1}" rawname="${ao_module_utils.objectToAttr(songInfo[1])}">
|
|
|
+ <div class="ui header selectable" style="margin:0px !important;" onClick="playSong(this);">
|
|
|
+ <img class="ui small image" src="img/eq.svg" style="margin-right: 0.2em;"></img>
|
|
|
+ <div class="content whiteFont" style="padding-top:5px;line-height:1em;width : 80%; font-weight: lighter;">
|
|
|
+ ${displayname}
|
|
|
+ <div class="sub header fileinfo" style="color: #c7c7c7;">${ext} / ${size}</div>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <div class="topRightCorner" align="center">
|
|
|
+ ${i + 1}
|
|
|
+ </div>
|
|
|
+ <div class="mainList rightFunctionBar" align="center" onClick="showMore(this);">
|
|
|
+ <i class="ellipsis vertical icon" style="margin-top:1.2em;"></i>\
|
|
|
+ </div>
|
|
|
+ </div>`);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (pagingEnabled){
|
|
|
+ //Append the page switch buttons
|
|
|
+ let numberOfPages = Math.ceil(parseFloat(displayList.length) / parseFloat(pagingCutoffCount));
|
|
|
+ let pageSelector = ``;
|
|
|
+ for (var i = 0; i < numberOfPages; i++){
|
|
|
+ let disabled = "";
|
|
|
+ if (i == currentPage){
|
|
|
+ disabled = "disabled";
|
|
|
+ }
|
|
|
+ pageSelector = pageSelector + `<button id="pagebtn_${i}" onclick="switchToPage(${i});" class="pagebtn noborderbtn inverted basic ui icon button ${disabled}" style="box-shadow: none !important; -webkit-box-shadow: none !important;">
|
|
|
+ ${i + 1}
|
|
|
+ </button>`;
|
|
|
+ }
|
|
|
+ $("#mainList").append(`<div class="mainList item" style="cursor: unset;">
|
|
|
+ <div>${pageSelector}</div>
|
|
|
+ </div>`);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ totalMusicCount = displayList.length;
|
|
|
+ $("#mainList").append("<br><br><br><br><br><br><br>");
|
|
|
+ $("#interfaceDetails").text("[" + totalMusicCount + " songs]");
|
|
|
+ highLightPlayingMusic();
|
|
|
+
|
|
|
+
|
|
|
+ //Load thumbnail for song in list
|
|
|
+ loadThumbnailToMusicList();
|
|
|
+
|
|
|
+ if (callback != undefined){
|
|
|
+ callback();
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function switchToPage(pageNumber, callback=undefined){
|
|
|
+ if (!pagingEnabled){
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ let thisCallback = callback;
|
|
|
+ if (pageNumber == currentPage){
|
|
|
+ //Already in that page
|
|
|
+ if (thisCallback != undefined){
|
|
|
+ callback();
|
|
|
+ }
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ currentPage = pageNumber;
|
|
|
+
|
|
|
+ let startPage = pageNumber * pagingCutoffCount;
|
|
|
+ let endPage = startPage + pagingCutoffCount;
|
|
|
+ if (endPage > displayList.length){
|
|
|
+ endPage = displayList.length;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ if (thisCallback == undefined){
|
|
|
+ renderDisplayList(displayList, startPage, endPage, function(){
|
|
|
+ window.scrollTo(0, document.body.scrollHeight);
|
|
|
+ });
|
|
|
+ }else{
|
|
|
+ renderDisplayList(displayList, startPage, endPage,thisCallback);
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function getPageNumberByPlaybackId(id){
|
|
|
+ let numberOfPages = Math.ceil(parseFloat(displayList.length) / parseFloat(pagingCutoffCount));
|
|
|
+ let targetPageNo = Math.ceil(id / pagingCutoffCount) - 1;
|
|
|
+ return targetPageNo;
|
|
|
+ }
|
|
|
+
|
|
|
+ function getPageStartAndEndByPageNumber(pageNumber, targetList){
|
|
|
+ let startPage = pageNumber * pagingCutoffCount;
|
|
|
+ let endPage = startPage + pagingCutoffCount;
|
|
|
+ if (endPage > targetList.length){
|
|
|
+ endPage = targetList.length;
|
|
|
+ }
|
|
|
+
|
|
|
+ return [startPage, endPage];
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ function toggleLeftMenu(){
|
|
|
+ if (leftMenuShown){
|
|
|
+ $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 120,function(){
|
|
|
+ $("#leftSideBar").hide();
|
|
|
+ });
|
|
|
+ $("#sideBarCover").fadeOut();
|
|
|
+ leftMenuShown = false;
|
|
|
+ }else{
|
|
|
+ $("#leftSideBar").show();
|
|
|
+ $("#leftSideBar").animate({left:0}, 300);
|
|
|
+ $("#sideBarCover").fadeIn();
|
|
|
+ leftMenuShown = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function hideLeftMenu(){
|
|
|
+ if (leftMenuShown){
|
|
|
+ $("#leftSideBar").animate({left: $("#leftSideBar").width() * -1}, 120,function(){
|
|
|
+ $("#leftSideBar").hide();
|
|
|
+ });
|
|
|
+ $("#sideBarCover").fadeOut();
|
|
|
+ leftMenuShown = false;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function showFileInformation(){
|
|
|
+ ao_module_agirun("Music/functions/getFileInfo.js", {
|
|
|
+ filepath: playingFileDetail[1],
|
|
|
+ },function(data){
|
|
|
+ //console.log(data);
|
|
|
+ $("#settingInterface").hide();
|
|
|
+ $("#filepropInterface").show();
|
|
|
+ $("#filepropInterface").find(".filename").text(ao_module_codec.decodeUmFilename(data[0]));
|
|
|
+ $("#filepropInterface").find(".rawname").text(data[0]);
|
|
|
+ $("#filepropInterface").find(".filepath").text(data[1]);
|
|
|
+ $("#filepropInterface").find(".filesize").text(data[2] + " (" + data[3] + " Bytes)");
|
|
|
+ $("#filepropInterface").find(".filedate").text(data[4]);
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function resizeQuickAdjust(){
|
|
|
+ //Resize the position of the leftMenu
|
|
|
+ if (!leftMenuShown){
|
|
|
+ $("#leftSideBar").css("left", $("#leftSideBar").width() * -1);
|
|
|
+ }
|
|
|
+ if ($("#playerInterface").offset().left != 0){
|
|
|
+ $("#playerInterface").css("left",window.innerWidth);
|
|
|
+ }
|
|
|
+
|
|
|
+ $("#albumnArtImage").css("max-height",window.innerHeight - 255);
|
|
|
+
|
|
|
+ setTimeout(function(){
|
|
|
+ $("#albumnArt").css({
|
|
|
+ "height": window.innerHeight - 255,
|
|
|
+ "top": (window.innerHeight / 2 - $("#albumnArtImage").height()/2)
|
|
|
+ });
|
|
|
+ },50);
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ //var imageTop = (window.innerHeight - 255 - $("#albumnArtImage").height()) / 2;
|
|
|
+ //$("#albumnArtImage").css("top",imageTop + "px");
|
|
|
+ }
|
|
|
+
|
|
|
+ function setStorage(configName,configValue){
|
|
|
+ ao_module_storage.setStorage("AirMusic",configName,configValue);
|
|
|
+ /*
|
|
|
+ $.ajax({
|
|
|
+ type: 'GET',
|
|
|
+ url: "../system/file_system/preference",
|
|
|
+ data: {key: "AirMusic/" + configName,value:configValue},
|
|
|
+ success: function(data){},
|
|
|
+ async:true
|
|
|
+ });
|
|
|
+ */
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadStorage(configName){
|
|
|
+ /*
|
|
|
+ var result = "";
|
|
|
+ $.ajax({
|
|
|
+ type: 'GET',
|
|
|
+ url: "../system/file_system/preference",
|
|
|
+ data: {key: "AirMusic/" + configName},
|
|
|
+ success: function(data){
|
|
|
+ if (data.error !== undefined){
|
|
|
+ result = "";
|
|
|
+ }else{
|
|
|
+ result = data;
|
|
|
+ }
|
|
|
+ },
|
|
|
+ error: function(data){result = "";},
|
|
|
+ async:false,
|
|
|
+ timeout: 3000
|
|
|
+ });
|
|
|
+ return result;
|
|
|
+ */
|
|
|
+
|
|
|
+ return ao_module_storage.loadStorage("AirMusic",configName);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ //Wipe controller for mobile users
|
|
|
+ $("#albumnArt")[0].addEventListener('touchstart', handleTouchStart, false);
|
|
|
+ $("#albumnArt")[0].addEventListener('touchmove', handleTouchMove, false);
|
|
|
+
|
|
|
+ var xDown = null;
|
|
|
+ var yDown = null;
|
|
|
+
|
|
|
+ function getTouches(evt) {
|
|
|
+ return evt.touches || // browser API
|
|
|
+ evt.originalEvent.touches; // jQuery
|
|
|
+ }
|
|
|
+
|
|
|
+ function handleTouchStart(evt) {
|
|
|
+ const firstTouch = getTouches(evt)[0];
|
|
|
+ xDown = firstTouch.clientX;
|
|
|
+ yDown = firstTouch.clientY;
|
|
|
+ };
|
|
|
+
|
|
|
+ function handleTouchMove(evt) {
|
|
|
+ if ( ! xDown || ! yDown ) {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ var xUp = evt.touches[0].clientX;
|
|
|
+ var yUp = evt.touches[0].clientY;
|
|
|
+
|
|
|
+ var xDiff = xDown - xUp;
|
|
|
+ var yDiff = yDown - yUp;
|
|
|
+
|
|
|
+ if ( Math.abs( xDiff ) > Math.abs( yDiff ) ) {/*most significant*/
|
|
|
+ if ( xDiff > 0 ) {
|
|
|
+ /* left swipe */
|
|
|
+ //Going to the next song
|
|
|
+ nextSong();
|
|
|
+ } else {
|
|
|
+ /* right swipe */
|
|
|
+ //Going back one song
|
|
|
+ previousSong();
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ if ( yDiff > 0 ) {
|
|
|
+ /* up swipe */
|
|
|
+ } else {
|
|
|
+ /* down swipe */
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* reset values */
|
|
|
+ xDown = null;
|
|
|
+ yDown = null;
|
|
|
+ };
|
|
|
+
|
|
|
+ //Handle audio progress to time conversion
|
|
|
+ function secondsToHms(d) {
|
|
|
+ d = Number(d);
|
|
|
+ var h = Math.floor(d / 3600);
|
|
|
+ var m = Math.floor(d % 3600 / 60);
|
|
|
+ var s = Math.floor(d % 3600 % 60);
|
|
|
+
|
|
|
+ if (h > 0 && h < 10){
|
|
|
+ dh = "0" + h + ":";
|
|
|
+ }else if (h == 0){
|
|
|
+ dh = "";
|
|
|
+ }else{
|
|
|
+ dh = h + ":";
|
|
|
+ }
|
|
|
+
|
|
|
+ if (m > 0 && m < 10){
|
|
|
+ dm = "0" + m + ":";
|
|
|
+ }else if (m == 0){
|
|
|
+ if (h > 0){
|
|
|
+ dm = "00:";
|
|
|
+ }else{
|
|
|
+ dm = "0:";
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ dm = m + ":";
|
|
|
+ }
|
|
|
+
|
|
|
+ if (s > 0 && s < 10){
|
|
|
+ ds = "0" + s;
|
|
|
+ }else if (s == 0){
|
|
|
+ if (m > 0 || h > 0){
|
|
|
+ ds = "00";
|
|
|
+ }else{
|
|
|
+ ds = "00";
|
|
|
+ }
|
|
|
+ }else{
|
|
|
+ ds = s;
|
|
|
+ }
|
|
|
+
|
|
|
+ return dh + dm + ds;
|
|
|
+ }
|
|
|
+
|
|
|
+ //Handle window resize events
|
|
|
+ $(window).on("resize", function(){
|
|
|
+ resizeQuickAdjust();
|
|
|
+ });
|
|
|
+
|
|
|
+ function AllQuickMenuHidden(){
|
|
|
+ var result = true;
|
|
|
+ $(".quickMenu").each(function(){
|
|
|
+ if ( $(this).is(':visible') ){
|
|
|
+ result = false;
|
|
|
+ }
|
|
|
+ });
|
|
|
+ return result;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ function AllSubMenuHidden(){
|
|
|
+ var result = true;
|
|
|
+ $(".showMoreMenus").each(function(){
|
|
|
+ if ( $(this).is(':visible') ){
|
|
|
+ result = false;
|
|
|
+ }
|
|
|
+ });
|
|
|
+ return result;
|
|
|
+ }
|
|
|
+
|
|
|
+ //Handle back button press on mobile devices
|
|
|
+ function handleBackButton() {
|
|
|
+ if (!AllQuickMenuHidden()){
|
|
|
+ hideAllquickMenu();
|
|
|
+ window.history.pushState({}, '');
|
|
|
+ }else if (mainPlayerShown()){
|
|
|
+ //Close the main Menu
|
|
|
+ hideMainPlayerInterface();
|
|
|
+ window.history.pushState({}, '');
|
|
|
+ }else if (!AllSubMenuHidden()){
|
|
|
+ $(".showMoreMenus").fadeOut('fast');
|
|
|
+ $("#showmoreUIcover").fadeOut('fast');
|
|
|
+ window.history.pushState({}, '');
|
|
|
+ }else{
|
|
|
+ window.history.back();
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+ window.addEventListener('popstate', handleBackButton);
|
|
|
+ window.history.pushState({}, '');
|
|
|
+</script>
|
|
|
+</body>
|
|
|
</html>
|