/* myMPD (c) 2018 Juergen Mang This project's homepage is: https://github.com/jcorporation/ympd myMPD ist fork of: ympd (c) 2013-2014 Andrew Karpow This project's homepage is: https://www.ympd.org This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ var socket; var last_state; var last_outputs; var pagination = 0; var filterLetter = '!' var browsepath = ''; var artist = ''; var lastSongTitle = ''; var current_song = new Object(); var MAX_ELEMENTS_PER_PAGE = 100; var isTouch = Modernizr.touch ? 1 : 0; var playstate = ''; var progressBar; var volumeBar; var coverImageFile = ''; var app = {}; app.apps = {"playback": { "state": "" }, "queue": { "state": "0/Any Tag/" }, "browse": { "active": "browseDatabase", "tabs": { "Filesystem": {"state": "0/!/"}, "Playlists": {"state": "0/!/" }, "Database": { "active":"Artist", "views": {"Artist": {"state": "0/!/" }, "Album": {"state": "0/!/" } } } } }, "search": { "state": "0/Any Tag/" } }; app.currentApp=''; app.currentTab=''; app.currentView=''; app.prepare=function() { $('#navbar-bottom > div').removeClass('active'); $('#cardPlayback').addClass('hide'); $('#cardQueue').addClass('hide'); $('#cardBrowse').addClass('hide'); $('#cardSearch').addClass('hide'); $('#searchqueue > input').val(''); $('#cardBrowsePlaylists').addClass('hide'); $('#cardBrowseDatabase').addClass('hide'); $('#cardBrowseFilesystem').addClass('hide'); $('#cardBrowseNavPlaylists').removeClass('active'); $('#cardBrowseNavDatabase').removeClass('active'); $('#cardBrowseNavFilesystem').removeClass('active'); } app.goto=function(a,t,v,s) { if (app.apps[a].tabs) { if (t == undefined) t = app.apps[a].active; if (app.apps[a].tabs[t].views) { if (v == undefined) v = app.apps[a].tabs[t].active; location.hash = '/'+a+t+v+'/'+ (s == undefined ? app.apps[a].tabs[t].views[v].state : s); } else { location.hash = '/'+a+t+'/'+ (s == undefined ? app.apps[a].tabs[t].state : s); } } else { location.hash = '/'+a+'/'+ (s == undefined ? app.apps[a].state : s); } } app.route=function() { var hash=decodeURI(location.hash); if (params=hash.match(/^\#\/(playback)\//)) { app.prepare(); app.currentApp=params[1]; app.currentTab=''; app.currentView=''; $('#cardPlayback').removeClass('hide'); $('#navPlayback').addClass('active'); } else if (params=hash.match(/^\#\/(queue)\/((\d+)\/([^\/]+)\/(.*))$/)) { app.currentApp=params[1]; app.currentTab=''; app.currentView=''; app.apps[app.currentApp].state=params[2]; pagination = parseInt(params[3]); var mpdtag = params[4]; var searchstr = params[5]; if ($('#cardQueue').hasClass('hide')) { app.prepare(); if (searchstr == '') { setPagination(pagination); } $('#searchqueuetag > button').each(function() { if ($(this).text == mpdtag) { $(this).removeClass('btn-secondary').addClass('btn-success'); $('#searchqueuetagdesc').text($(this).text()); } }); $('#cardQueue').removeClass('hide'); $('#navQueue').addClass('active'); } if (searchstr.length >= 2) { socket.send('MPD_API_SEARCH_QUEUE,' + mpdtag + ','+pagination+',' + searchstr); } else { socket.send('MPD_API_GET_QUEUE,'+pagination); } } else if (params=hash.match(/^\#\/(browse)(Playlists)\/((\d+)\/(\w|\!)\/)$/)) { app.currentApp=params[1]; app.currentTab=params[2]; app.currentView=''; app.apps[app.currentApp].active=params[2]; app.apps[app.currentApp].tabs[params[2]].state=params[3]; pagination = parseInt(params[4]); filterLetter = params[5]; app.prepare(); $('#navBrowse').addClass('active'); $('#cardBrowse').removeClass('hide'); $('#cardBrowsePlaylists').removeClass('hide'); $('#cardBrowseNavPlaylists').addClass('active'); socket.send('MPD_API_GET_PLAYLISTS,'+pagination+','+filterLetter); doSetFilterLetter('#browsePlaylistsFilter'); } else if (params=hash.match(/^\#\/(browse)(Database)(Artist)\/((\d+)\/(\w|\!)\/)$/)) { app.currentApp=params[1]; app.currentTab=params[2]; app.currentView=params[3]; app.apps[app.currentApp].active=params[2]; app.apps[app.currentApp].tabs[params[2]].active=params[3]; app.apps[app.currentApp].tabs[params[2]].views[params[3]].state=params[4]; app.prepare(); pagination = parseInt(params[5]); filterLetter = params[6]; $('#navBrowse').addClass('active'); $('#cardBrowse').removeClass('hide'); $('#cardBrowseDatabase').removeClass('hide'); $('#cardBrowseNavDatabase').addClass('active'); socket.send('MPD_API_GET_ARTISTS,' + pagination + ',' + filterLetter); doSetFilterLetter('#browseDatabaseFilter'); } else if (params=hash.match(/^\#\/(browse)(Database)(Album)\/((\d+)\/(\w|\!)\/(.*))$/)) { app.currentApp=params[1]; app.currentTab=params[2]; app.currentView=params[3]; app.apps[app.currentApp].active=params[2]; app.apps[app.currentApp].tabs[params[2]].active=params[3]; app.apps[app.currentApp].tabs[params[2]].views[params[3]].state=params[4]; app.prepare(); pagination = parseInt(params[5]); filterLetter = params[6]; artist = params[7]; $('#navBrowse').addClass('active'); $('#cardBrowse').removeClass('hide'); $('#cardBrowseDatabase').removeClass('hide'); $('#cardBrowseNavDatabase').addClass('active'); socket.send('MPD_API_GET_ARTISTALBUMS,' + pagination+',' + filterLetter + ',' + decodeURI(artist)); doSetFilterLetter('#browseDatabaseFilter'); } else if (params=hash.match(/^\#\/(browse)(Filesystem)\/((\d+)\/(\w|\!)\/(.*))$/)) { app.prepare(); app.currentApp=params[1]; app.currentTab=params[2]; app.currentView=''; app.apps[app.currentApp].active=params[2]; app.apps[app.currentApp].tabs[params[2]].state=params[3]; pagination = parseInt(params[4]); filterLetter = params[5]; browsepath = params[6]; $('#navBrowse').addClass('active'); $('#cardBrowse').removeClass('hide'); $('#cardBrowseFilesystem').removeClass('hide'); $('#cardBrowseNavFilesystem').addClass('active'); $('#browseBreadcrumb').empty().append("
  • root
  • "); socket.send('MPD_API_GET_BROWSE,'+pagination+','+(browsepath ? browsepath : "/")+','+filterLetter); doSetFilterLetter('#browseFilesystemFilter'); // Don't add all songs from root var add_all_songs = $('#browseFilesystemAddAllSongs'); if (browsepath) { add_all_songs.off(); // remove previous binds add_all_songs.on('click', function() { socket.send('MPD_API_ADD_TRACK,'+browsepath); }); add_all_songs.removeAttr('disabled').removeClass('disabled'); } else { add_all_songs.attr('disabled','disabled').addClass('disabled'); } var path_array = browsepath.split('/'); var full_path = ""; $.each(path_array, function(index, chunk) { if(path_array.length - 1 == index) { $('#browseBreadcrumb').append("
  • "+ chunk + "
  • "); return; } full_path = full_path + chunk; $('#browseBreadcrumb').append("
  • "+chunk+"
  • "); full_path += "/"; }); } else if (params=hash.match(/^\#\/(search)\/((\d+)\/([^\/]+)\/(.*))$/)) { app.currentApp=params[1]; app.apps[app.currentApp].state=params[2]; app.currentTab=''; app.currentView=''; pagination = parseInt(params[3]); var mpdtag = params[4]; var searchstr = params[5]; if ($('#cardSearch').hasClass('hide')) { app.prepare(); if (searchstr != '') { $('#searchList > tbody').append( "search" + "Searching" + ""); } else { setPagination(pagination); } $('#search > input').val(searchstr); $('#searchstr2').val(searchstr); $('#searchtags2 > button').each(function() { if ($(this).text == mpdtag) { $(this).removeClass('btn-secondary').addClass('btn-success'); $('#searchtags2desc').text($(this).text); } }); $('#cardSearch').removeClass('hide'); $('#navSearch').addClass('active'); } if (searchstr.length >= 2) { socket.send('MPD_API_SEARCH,' + mpdtag + ','+pagination+',' + searchstr); } else { $('#searchList > tbody').empty(); $('#searchAddAllSongs').attr('disabled','disabled').addClass('disabled'); } } else { app.goto("playback"); } }; $(document).ready(function(){ webSocketConnect(); volumeBar=$('#volumebar').slider(); volumeBar.slider('setValue',0); volumeBar.slider('on','slideStop', function(value){ socket.send("MPD_API_SET_VOLUME,"+value); }); progressBar=$('#progressbar').slider(); progressBar.slider('setValue',0); progressBar.slider('on','slideStop', function(value){ if(current_song && current_song.currentSongId >= 0) { var seekVal = Math.ceil(current_song.totalTime*(value/100)); socket.send("MPD_API_SET_SEEK,"+current_song.currentSongId+","+seekVal); } }); $('#about').on('shown.bs.modal', function () { socket.send("MPD_API_GET_STATS"); }) $('#settings').on('shown.bs.modal', function () { socket.send("MPD_API_GET_SETTINGS"); }) $('#addstream').on('shown.bs.modal', function () { $('#streamurl').focus(); }) $('#addstream form').on('submit', function (e) { addStream(); }); $('#mainMenu').on('shown.bs.dropdown', function () { $('#search > input').val(''); $('#search > input').focus(); }); if(!notificationsSupported()) $('#btnnotifyWeb').addClass("disabled"); else if (Cookies.get('notificationWeb') === 'true') $('#btnnotifyWeb').removeClass('btn-secondary').addClass("btn-success") if (Cookies.get('notificationPage') === 'true') $('#btnnotifyPage').removeClass('btn-secondary').addClass("btn-success") add_filter('#browseFilesystemFilterLetters'); add_filter('#browseDatabaseFilterLetters'); add_filter('#browsePlaylistsFilterLetters'); window.addEventListener("hashchange", app.route, false); }); function webSocketConnect() { if (typeof MozWebSocket != "undefined") { socket = new MozWebSocket(get_appropriate_ws_url()); } else { socket = new WebSocket(get_appropriate_ws_url()); } try { socket.onopen = function() { console.log("connected"); /* emit request for mympd settings */ socket.send('MPD_API_GET_SETTINGS'); /* emit initial request for output names */ socket.send('MPD_API_GET_OUTPUTS'); showNotification('Connected to myMPD','','','success'); $('#modalConnectionError').modal('hide'); app.route(); } socket.onmessage = function got_packet(msg) { if(msg.data === last_state || msg.data.length == 0) return; try { var obj = JSON.parse(msg.data); } catch(e) { console.log('Invalid JSON data received: '+ msg.data); } switch (obj.type) { case 'queuesearch': //Do the same as queue case 'queue': if(app.currentApp !== 'queue') break; $('#panel-heading-queue').empty(); if (obj.totalEntities > 0) { $('#panel-heading-queue').text(obj.totalEntities+' Songs'); } if (typeof(obj.totalTime) != undefined && obj.totalTime > 0 ) { $('#panel-heading-queue').append(' – ' + beautifyDuration(obj.totalTime)); } var nrItems=0; var tr=document.getElementById(app.currentApp+'List').getElementsByTagName('tbody')[0].getElementsByTagName('tr'); for (var song in obj.data) { nrItems++; var minutes = Math.floor(obj.data[song].duration / 60); var seconds = obj.data[song].duration - minutes * 60; var row="" + (obj.data[song].pos + 1) + "" + ""+ obj.data[song].title +"" + ""+ obj.data[song].artist +"" + ""+ obj.data[song].album +"" + ""+ minutes + ":" + (seconds < 10 ? '0' : '') + seconds + ""; if (nrItems <= tr.length) { $(tr[nrItems-1]).replaceWith(row); } else { $('#'+app.currentApp+'List > tbody').append(row); } } for (var i=tr.length;i>nrItems;i--) { $(tr[tr.length-1]).remove(); } if (obj.type == 'queuesearch' && nrItems == 0) { $('#queueList > tbody').append( "error_outline" + "No results, please refine your search!" + "" ); } setPagination(obj.totalEntities); if ( isTouch ) { $('#queueList > tbody > tr > td:last-child').append( '' + 'delete'); } else { $('#queueList > tbody > tr').on({ mouseover: function(){ var doomed = $(this); if ( $('#btntrashmodeup').hasClass('btn-success') ) doomed = $('#queueList > tbody > tr:lt(' + ($(this).index() + 1) + ')'); if ( $('#btntrashmodedown').hasClass('btn-success') ) doomed = $('#queueList > tbody > tr:gt(' + ($(this).index() - 1) + ')'); $.each(doomed, function(){ if($(this).children().last().has('a').length == 0) $(this).children().last().append( '' + 'delete') .find('a').fadeTo('fast',1); }); }, mouseleave: function(){ var doomed = $(this); if ( $('#btntrashmodeup').hasClass('btn-success') ) doomed = $("#queueList > tbody > tr:lt(" + ($(this).index() + 1) + ")"); if ( $('#btntrashmodedown').hasClass('btn-success') ) doomed = $("#queueList > tbody > tr:gt(" + ($(this).index() - 1) + ")"); $.each(doomed, function(){$(this).children().last().find("a").stop().remove();}); } }); }; $('#queueList > tbody > tr').on({ click: function() { $('#queueList > tbody > tr').removeClass('active'); socket.send('MPD_API_PLAY_TRACK,'+$(this).attr('trackid')); $(this).addClass('active'); }, }); break; case 'playlists': if(app.currentApp !== 'browse' && app.currentTab !== 'Playlists') break; var nrItems=0; var tr=document.getElementById(app.currentApp+app.currentTab+'List').getElementsByTagName('tbody')[0].getElementsByTagName('tr'); for (var item in obj.data) { nrItems++; var d = new Date(obj.data[item].last_modified * 1000); var row='' + 'list' + '' + basename(obj.data[item].plist) + '' + ''+d.toUTCString()+''; if (nrItems <= tr.length) { $(tr[nrItems-1]).replaceWith(row); } else { $('#'+app.currentApp+app.currentTab+'List > tbody').append(row); } } for (var i=tr.length;i>nrItems;i--) { $(tr[tr.length-1]).remove(); } setPagination(obj.totalEntities); if ( isTouch ) { $('#'+app.currentApp+app.currentTab+'List > tbody > tr > td:last-child').append( '' + 'delete'); } else { $('#'+app.currentApp+app.currentTab+'List > tbody > tr').on({ mouseover: function(){ if($(this).children().last().has('a').length == 0) $(this).children().last().append( '' + 'delete'); }, mouseleave: function(){ var doomed = $(this); $(this).children().last().find("a").stop().remove(); } }); }; $('#'+app.currentApp+app.currentTab+'List > tbody > tr').on({ click: function() { socket.send('MPD_API_ADD_PLAYLIST,' + decodeURI($(this).attr('uri'))); showNotification('"' + $('td:nth-last-child(3)', this).text() + '" added','','','success'); } }); if (nrItems == 0) { $('#'+app.currentApp+app.currentTab+'List > tbody').append( 'error_outline' + 'No playlists found.' + '' ); } break; case 'listDBtags': if(app.currentApp !== 'browse' && app.currentTab !== 'Database') break; if (obj.tagtype == 'AlbumArtist') { $('#browseDatabaseAlbumCards').addClass('hide'); $('#browseDatabaseArtistList').removeClass('hide'); $('#btnBrowseDatabaseArtist').addClass('hide'); var nrItems=0; var tr=document.getElementById(app.currentApp+app.currentTab+app.currentView+'List').getElementsByTagName('tbody')[0].getElementsByTagName('tr'); for (var item in obj.data) { nrItems++; var row='' + 'album' + '' + obj.data[item].value + ''; if (nrItems <= tr.length) { $(tr[nrItems-1]).replaceWith(row); } else { $('#'+app.currentApp+app.currentTab+app.currentView+'List > tbody').append(row); } } for (var i=tr.length;i>nrItems;i--) { $(tr[tr.length-1]).remove(); } setPagination(obj.totalEntities); $('#'+app.currentApp+app.currentTab+app.currentView+'List > tbody > tr').on({ click: function() { app.goto('browse','Database','Album','0/!/'+$(this).attr('uri')); } }); if (nrItems == 0) { $('#'+app.currentApp+app.currentTab+app.currentView+'List > tbody').append( 'error_outline' + 'No entries found.' + '' ); } } else if (obj.tagtype == 'Album') { $('#browseDatabaseArtistList').addClass('hide'); $('#browseDatabaseAlbumCards').empty(); $('#browseDatabaseAlbumCards').removeClass('hide'); $('#btnBrowseDatabaseArtist').removeClass('hide'); var nrItems=0; for (var item in obj.data) { var card='
    '+ ' '+ '
    '+ '
    '+obj.searchstr+'
    '+ '

    '+obj.data[item].value+'

    '+ ' '+ ''; $('#browseDatabaseAlbumCards').append(card); socket.send('MPD_API_GET_ARTISTALBUMTITLES,' + obj.searchstr + ','+obj.data[item].value); } setPagination(obj.totalEntities); } break; case 'listTitles': var album=$('#'+genId(obj.album)+' > div > table > tbody'); $('#'+genId(obj.album)+' > img').attr('src','/library/'+obj.data[0].uri.replace(/\/[^\/]+$/,'\/')+coverImageFile); $('#'+genId(obj.album)+' > img').attr('uri',obj.data[0].uri.replace(/\/[^\/]+$/,'')); $('#'+genId(obj.album)+' > img').attr('data-album',encodeURI(obj.album)); var titleList=''; for (var item in obj.data) { titleList+=''+ ''; } album.append(titleList); $('#'+genId(obj.album)+' > img').on({ click: function() { socket.send('MPD_API_ADD_TRACK,' + decodeURI($(this).attr('uri'))); showNotification('"'+decodeURI($(this).attr('data-album'))+'" added','','','success'); } }); $('#tbl'+genId(obj.album)+' > tbody > tr').on({ click: function() { socket.send('MPD_API_ADD_TRACK,' + decodeURI($(this).attr('uri'))); showNotification('"' + $('td:nth-last-child(1)', this).text() + '" added','','','success'); } }); break; case 'search': $('#panel-heading-search').text(obj.totalEntities + ' Songs found'); if (obj.totalEntities > 0) { $('#searchAddAllSongs').removeAttr('disabled').removeClass('disabled'); } else { $('#searchAddAllSongs').attr('disabled','disabled').addClass('disabled'); } case 'browse': if(app.currentApp !== 'browse' && app.currentTab !== 'Filesystem' && app.currentApp !== 'search') break; /* The use of encodeURI() below might seem useless, but it's not. It prevents * some browsers, such as Safari, from changing the normalization form of the * URI from NFD to NFC, breaking our link with MPD. */ var nrItems=0; var tr=document.getElementById(app.currentApp+app.currentTab+'List').getElementsByTagName('tbody')[0].getElementsByTagName('tr'); for (var item in obj.data) { nrItems++; var row=''; switch(obj.data[item].type) { case 'directory': row ='' + '' + '' + ''; break; case 'song': var minutes = Math.floor(obj.data[item].duration / 60); var seconds = obj.data[item].duration - minutes * 60; row ='' + '' + '' + '' + '' + ''; break; case 'playlist': row ='' + '' + '' + ''; break; } if (nrItems <= tr.length) { $(tr[nrItems-1]).replaceWith(row); } else { $('#'+app.currentApp+app.currentTab+'List > tbody').append(row); } } for (var i=tr.length;i>nrItems;i--) { $(tr[tr.length-1]).remove(); } setPagination(obj.totalEntities); if (nrItems == 0) { $('#'+app.currentApp+app.currentTab+'List > tbody').append( '' + '' + ''); } function appendClickableIcon(appendTo, onClickAction, glyphicon) { $(appendTo).append( '' + '' + glyphicon + '') .find('a').click(function(e) { e.stopPropagation(); socket.send(onClickAction + "," + decodeURI($(this).parents("tr").attr("uri"))); showNotification('"' + $('td:nth-last-child(3)', $(this).parents('tr')).text() + '" added','','','success'); }); } if ( isTouch ) { appendClickableIcon($('#'+app.currentApp+app.currentTab+'List > tbody > tr.dir > td:last-child'), 'MPD_API_ADD_TRACK', 'playlist_add'); appendClickableIcon($('#'+app.currentApp+app.currentTab+'List > tbody > tr.song > td:last-child'), 'MPD_API_ADD_TRACK', 'playlist_add'); } else { $('#'+app.currentApp+app.currentTab+'List > tbody > tr').on({ mouseenter: function() { if($(this).is(".dir")) appendClickableIcon($(this).children().last(), 'MPD_API_ADD_TRACK', 'playlist_add'); else if($(this).is(".song")) appendClickableIcon($(this).children().last(), 'MPD_API_ADD_TRACK', 'playlist_add'); }, mouseleave: function(){ $(this).children().last().find("a").stop().remove(); } }); }; $('#'+app.currentApp+app.currentTab+'List > tbody > tr').on({ click: function() { switch($(this).attr('class')) { case 'dir': pagination = 0; browsepath = $(this).attr("uri"); $("#browseFilesystemList > a").attr("href", '#/browseFilesystem/'+pagination+'/'+filterLetter+'/'+browsepath); app.goto('browse','Filesystem',undefined,pagination+'/'+filterLetter+'/'+browsepath); break; case 'song': socket.send("MPD_API_ADD_TRACK," + decodeURI($(this).attr("uri"))); showNotification('"' + $('td:nth-last-child(5)', this).text() + '" added','','','success'); break; case 'plist': socket.send("MPD_API_ADD_PLAYLIST," + decodeURI($(this).attr("uri"))); showNotification('"' + $('td:nth-last-child(3)', this).text() + '" added','','','success'); break; } } }); $('#browseBreadcrumb > li > a').on({ click: function() { pagination = 0; browsepath = $(this).attr("uri"); $("#browseFilesystemList > a").attr("href", '#/browseFilesystem/'+pagination+'/'+filterLetter+'/'+browsepath); app.goto('browse','Filesystem',undefined,pagination+'/'+filterLetter+'/'+browsepath); } }); break; case 'state': updatePlayIcon(obj); updateVolumeIcon(obj.data.volume); if(JSON.stringify(obj) === JSON.stringify(last_state)) break; current_song.totalTime = obj.data.totalTime; current_song.currentSongId = obj.data.currentsongid; var total_minutes = Math.floor(obj.data.totalTime / 60); var total_seconds = obj.data.totalTime - total_minutes * 60; var elapsed_minutes = Math.floor(obj.data.elapsedTime / 60); var elapsed_seconds = obj.data.elapsedTime - elapsed_minutes * 60; volumeBar.slider('setValue',obj.data.volume); var progress = Math.floor(100*obj.data.elapsedTime/obj.data.totalTime); progressBar.slider('setValue',progress); var counterText=elapsed_minutes + ":" + (elapsed_seconds < 10 ? '0' : '') + elapsed_seconds + " / " + total_minutes + ":" + (total_seconds < 10 ? '0' : '') + total_seconds; $('#counter').text(counterText); if (last_state) { $('#queueList > tbody > tr[trackid='+last_state.data.currentsongid+'] > td').eq(4).text(last_state.data.totalTime); $('#queueList > tbody > tr[trackid='+last_state.data.currentsongid+'] > td').eq(0).removeClass('material-icons').text(last_state.data.songpos); } $('#queueList > tbody > tr').removeClass('active').removeClass("font-weight-bold"); $('#queueList > tbody > tr[trackid='+obj.data.currentsongid+'] > td').eq(4).text(counterText); $('#queueList > tbody > tr[trackid='+obj.data.currentsongid+'] > td').eq(0).addClass('material-icons').text('play_arrow'); $('#queueList > tbody > tr[trackid='+obj.data.currentsongid+']').addClass('active').addClass("font-weight-bold"); last_state = obj; break; case 'outputnames': $('#btn-outputs-block button').remove(); if ( Object.keys(obj.data).length ) { $.each(obj.data, function(id, name){ var btn = $(''); btn.appendTo($('#btn-outputs-block')); }); } else { $('#btn-outputs-block').addClass('hide'); } /* remove cache, since the buttons have been recreated */ last_outputs = ''; break; case 'outputs': if(JSON.stringify(obj) === JSON.stringify(last_outputs)) break; $.each(obj.data, function(id, enabled){ if (enabled) $('#btnoutput'+id).removeClass('btn-secondary').addClass("btn-success") else $('#btnoutput'+id).removeClass("btn-success").addClass("btn-secondary"); }); last_outputs = obj; break; case 'disconnected': showNotification('myMPD lost connection to MPD','','','danger'); break; case 'update_queue': if(app.currentApp === 'queue') socket.send('MPD_API_GET_QUEUE,'+pagination); break; case "song_change": songChange(obj.data.title, obj.data.artist, obj.data.album, obj.data.uri); break; case 'settings': if (!isNaN(obj.data.max_elements_per_page)) MAX_ELEMENTS_PER_PAGE=obj.data.max_elements_per_page; if(obj.data.random) $('#btnrandom').removeClass('btn-secondary').addClass("btn-success") else $('#btnrandom').removeClass("btn-success").addClass("btn-secondary"); if(obj.data.consume) $('#btnconsume').removeClass('btn-secondary').addClass("btn-success") else $('#btnconsume').removeClass("btn-success").addClass("btn-secondary"); if(obj.data.single) $('#btnsingle').removeClass('btn-secondary').addClass("btn-success") else $('#btnsingle').removeClass("btn-success").addClass("btn-secondary"); if(obj.data.crossfade != undefined) $('#inputCrossfade').removeAttr('disabled').val(obj.data.crossfade); else $('#inputCrossfade').attr('disabled', 'disabled'); if(obj.data.mixrampdb != undefined) $('#inputMixrampdb').removeAttr('disabled').val(obj.data.mixrampdb); else $('#inputMixrampdb').attr('disabled', 'disabled'); if(obj.data.mixrampdelay != undefined) $('#inputMixrampdelay').removeAttr('disabled').val(obj.data.mixrampdelay); else $('#inputMixrampdb').attr('disabled', 'disabled'); if(obj.data.repeat) $('#btnrepeat').removeClass('btn-secondary').addClass("btn-success") else $('#btnrepeat').removeClass("btn-success").addClass("btn-secondary"); $("#selectReplaygain").val(obj.data.replaygain); setLocalStream(obj.data.mpdhost,obj.data.streamport); coverImageFile=obj.data.coverimage; break; case 'mpdstats': $('#mpdstats_artists').text(obj.data.artists); $('#mpdstats_albums').text(obj.data.albums); $('#mpdstats_songs').text(obj.data.songs); $('#mpdstats_dbplaytime').text(beautifyDuration(obj.data.dbplaytime)); $('#mpdstats_playtime').text(beautifyDuration(obj.data.playtime)); $('#mpdstats_uptime').text(beautifyDuration(obj.data.uptime)); var d = new Date(obj.data.dbupdated * 1000); $('#mpdstats_dbupdated').text(d.toUTCString()); $('#mympdVersion').text(obj.data.mympd_version); $('#mpdVersion').text(obj.data.mpd_version); break; case 'error': showNotification(obj.data,'','','danger'); default: break; } } socket.onclose = function(){ console.log('disconnected'); $('#modalConnectionError').modal('show'); setTimeout(function() { console.log('reconnect'); webSocketConnect(); },3000); } } catch(exception) { alert('

    Error' + exception); } } function get_appropriate_ws_url() { var pcol; var u = document.URL; var separator; /* /* We open the websocket encrypted if this page came on an /* https:// url itself, otherwise unencrypted /*/ if (u.substring(0, 5) == "https") { pcol = "wss://"; u = u.substr(8); } else { pcol = "ws://"; if (u.substring(0, 4) == "http") u = u.substr(7); } u = u.split('#'); if (/\/$/.test(u[0])) { separator = ""; } else { separator = "/"; } return pcol + u[0] + separator + "ws"; } function setPagination(number) { var totalPages=Math.ceil(number / MAX_ELEMENTS_PER_PAGE); if (totalPages==0) { totalPages=1; } $('#'+app.currentApp+app.currentTab+'PaginationTopPage').text('Page '+(pagination / MAX_ELEMENTS_PER_PAGE + 1)+' / '+totalPages); $('#'+app.currentApp+app.currentTab+'PaginationBottomPage').text('Page '+(pagination / MAX_ELEMENTS_PER_PAGE + 1)+' / '+totalPages); if (totalPages > 1) { $('#'+app.currentApp+app.currentTab+'PaginationTopPage').removeClass('disabled').removeAttr('disabled'); $('#'+app.currentApp+app.currentTab+'PaginationBottomPage').removeClass('disabled').removeAttr('disabled'); $('#'+app.currentApp+app.currentTab+'PaginationTopPages').empty(); $('#'+app.currentApp+app.currentTab+'PaginationBottomPages').empty(); for (var i=0;i'+(i+1)+''); $('#'+app.currentApp+app.currentTab+'PaginationBottomPages').append(''); } } else { $('#'+app.currentApp+app.currentTab+'PaginationTopPage').addClass('disabled').attr('disabled','disabled'); $('#'+app.currentApp+app.currentTab+'PaginationBottomPage').addClass('disabled').attr('disabled','disabled'); } if(number > pagination + MAX_ELEMENTS_PER_PAGE) { $('#'+app.currentApp+app.currentTab+'PaginationTopNext').removeClass('disabled').removeAttr('disabled'); $('#'+app.currentApp+app.currentTab+'PaginationBottomNext').removeClass('disabled').removeAttr('disabled'); $('#'+app.currentApp+app.currentTab+'ButtonsBottom').removeClass('hide'); } else { $('#'+app.currentApp+app.currentTab+'PaginationTopNext').addClass('disabled').attr('disabled','disabled'); $('#'+app.currentApp+app.currentTab+'PaginationBottomNext').addClass('disabled').attr('disabled','disabled'); $('#'+app.currentApp+app.currentTab+'ButtonsBottom').addClass('hide'); } if(pagination > 0) { $('#'+app.currentApp+app.currentTab+'PaginationTopPrev').removeClass('disabled').removeAttr('disabled'); $('#'+app.currentApp+app.currentTab+'PaginationBottomPrev').removeClass('disabled').removeAttr('disabled'); } else { $('#'+app.currentApp+app.currentTab+'PaginationTopPrev').addClass('disabled').attr('disabled','disabled'); $('#'+app.currentApp+app.currentTab+'PaginationBottomPrev').addClass('disabled').attr('disabled','disabled'); } } function updateVolumeIcon(volume) { if (volume == -1) { $('#volumePrct').text('Volumecontrol disabled'); $('#volumeControl').addClass('hide'); } else { $('#volumeControl').removeClass('hidden'); $('#volumePrct').text(volume+' %'); if(volume == 0) { $("#volume-icon").text("volume_off"); } else if (volume < 50) { $("#volume-icon").text("volume_down"); } else { $("#volume-icon").text("volume_up"); } } } function updatePlayIcon(obj) { if(obj.data.state == 1) { // stop $('#btnPlay > span').text('play_arrow'); playstate = 'stop'; } else if(obj.data.state == 2) { // play $('#btnPlay > span').text('pause'); playstate = 'play'; } else { // pause $('#btnPlay > span').text('play_arrow'); playstate = 'pause'; } if (obj.data.nextsongpos == -1) { $('#btnNext').addClass('disabled').attr('disabled','disabled'); } else { $('#btnNext').removeClass('disabled').removeAttr('disabled'); } if (obj.data.songpos <= 0) { $('#btnPrev').addClass('disabled').attr('disabled','disabled'); } else { $('#btnPrev').removeClass('disabled').removeAttr('disabled'); } if (obj.data.queue_length == 0) { $('#btnPlay').addClass('disabled').attr('disabled','disabled'); } else { $('#btnPlay').removeClass('disabled').removeAttr('disabled'); } } function updateDB(event) { socket.send('MPD_API_UPDATE_DB'); showNotification('Updating MPD Database...','','','success'); event.preventDefault(); } function clickPlay() { if( playstate != 'play') socket.send('MPD_API_SET_PLAY'); else socket.send('MPD_API_SET_PAUSE'); } function setLocalStream(mpdhost,streamport) { var mpdstream = 'http://'; if ( mpdhost == '127.0.0.1' || mpdhost == 'localhost') mpdstream += window.location.hostname; else mpdstream += mpdhost; mpdstream += ':'+streamport+'/'; Cookies.set('mpdstream', mpdstream, { expires: 424242 }); } function delQueueSong(tr) { if ( $('#btntrashmodeup').hasClass('btn-success') ) { socket.send('MPD_API_RM_RANGE,0,' + (tr.index() + 1)); tr.remove(); } else if ( $('#btntrashmodesingle').hasClass('btn-success') ) { socket.send('MPD_API_RM_TRACK,' + tr.attr('trackid')); tr.remove(); } else if ( $('#btntrashmodedown').hasClass('btn-success') ) { socket.send('MPD_API_RM_RANGE,' + tr.index() + ',-1'); tr.remove(); }; } function delPlaylist(tr) { socket.send("MPD_API_RM_PLAYLIST," + decodeURI(tr.attr("uri"))); tr.remove(); } function basename(path) { return path.split('/').reverse()[0]; } function toggleBtn(btn) { if ($(btn).hasClass('btn-success')) { $(btn).removeClass('btn-success').addClass('btn-secondary'); } else { $(btn).removeClass('btn-secondary').addClass('btn-success'); } } $('#btnrandom').on('click', function (e) { toggleBtn(this); }); $('#btnconsume').on('click', function (e) { toggleBtn(this); }); $('#btnsingle').on('click', function (e) { toggleBtn(this); }); $('#btnrepeat').on('click', function (e) { toggleBtn(this); }); $('#cardBrowseNavFilesystem').on('click', function (e) { app.goto('browse','Filesystem'); e.preventDefault(); }); $('#cardBrowseNavDatabase').on('click', function (e) { app.goto('browse','Database'); e.preventDefault(); }); $('#btnBrowseDatabaseArtist').on('click', function (e) { app.goto('browse','Database','Artist'); e.preventDefault(); }); $('#cardBrowseNavPlaylists').on('click', function (e) { app.goto('browse','Playlists'); e.preventDefault(); }); $('#cardBrowseNavFilesystem').on('click', function (e) { app.goto('browse','Filesystem'); e.preventDefault(); }); $('#navPlayback').on('click', function (e) { app.goto('playback'); e.preventDefault(); }); $('#navQueue').on('click', function (e) { app.goto('queue'); e.preventDefault(); }); $('#navBrowse').on('click', function (e) { app.goto('browse'); e.preventDefault(); }); $('#navSearch').on('click', function (e) { app.goto('search'); e.preventDefault(); }); function confirmSettings() { var formOK=true; if (!$('#inputCrossfade').is(':disabled')) { var value=parseInt($('#inputCrossfade').val()); if (!isNaN(value)) { $('#inputCrossfade').val(value); } else { $('#inputCrossfade').popover({"content":"Must be a number","trigger":"manual"}); $('#inputCrossfade').popover('show'); $('#inputCrossfade').focus(); formOK=false; } } if (!$('#inputMixrampdb').is(':disabled')) { var value=parseFloat($('#inputMixrampdb').val()); if (!isNaN(value)) { $('#inputMixrampdb').val(value); } else { $('#inputMixrampdb').popover({"content":"Must be a number","trigger":"manual"}); $('#inputMixrampdb').popover('show'); $('#inputMixrampdb').focus(); formOK=false; } } if (!$('#inputMixrampdelay').is(':disabled')) { if ($('#inputMixrampdelay').val() == 'nan') $('#inputMixrampdelay').val('-1'); var value=parseFloat($('#inputMixrampdelay').val()); if (!isNaN(value)) { $('#inputMixrampdelay').val(value); } else { $('#inputMixrampdelay').popover({"content":"Must be a number, -1 to disable","trigger":"manual"}); $('#inputMixrampdelay').popover('show'); $('#inputMixrampdelay').focus(); formOK=false; } } if (formOK == true) { socket.send("MPD_API_TOGGLE_CONSUME," + ($('#btnconsume').hasClass('btn-success') ? 1 : 0)); socket.send("MPD_API_TOGGLE_RANDOM," + ($('#btnrandom').hasClass('btn-success') ? 1 : 0)); socket.send("MPD_API_TOGGLE_SINGLE," + ($('#btnsingle').hasClass('btn-success') ? 1 : 0)); socket.send("MPD_API_TOGGLE_REPEAT," + ($('#btnrepeat').hasClass('btn-success') ? 1 : 0)); socket.send("MPD_API_SET_REPLAYGAIN," + $('#selectReplaygain').val()); if (!$('#inputCrossfade').is(':disabled')) socket.send("MPD_API_SET_CROSSFADE," + $('#inputCrossfade').val()); if (!$('#inputMixrampdb').is(':disabled')) socket.send("MPD_API_SET_MIXRAMPDB," + $('#inputMixrampdb').val()); if (!$('#inputMixrampdelay').is(':disabled')) socket.send("MPD_API_SET_MIXRAMPDELAY," + $('#inputMixrampdelay').val()); $('#settings').modal('hide'); } } function toggleoutput(button, id) { socket.send("MPD_API_TOGGLE_OUTPUT,"+id+"," + ($(button).hasClass('btn-success') ? 0 : 1)); } $('#trashmodebtns > button').on('click', function(e) { $('#trashmodebtns').children("button").removeClass("btn-success").addClass('btn-secondary'); $(this).removeClass("btn-secondary").addClass("btn-success"); }); $('#btnnotifyWeb').on('click', function (e) { if(Cookies.get('notificationWeb') === 'true') { Cookies.set('notificationWeb', false, { expires: 424242 }); $('#btnnotifyWeb').removeClass('btn-success').addClass('btn-secondary'); } else { Notification.requestPermission(function (permission) { if(!('permission' in Notification)) { Notification.permission = permission; } if (permission === 'granted') { Cookies.set('notificationWeb', true, { expires: 424242 }); $('#btnnotifyWeb').removeClass('btn-secondary').addClass('btn-success'); } }); } }); $('#btnnotifyPage').on('click', function (e) { if(Cookies.get("notificationPage") === 'true') { Cookies.set("notificationPage", false, { expires: 424242 }); $('#btnnotifyPage').removeClass('btn-success').addClass('btn-secondary'); } else { Cookies.set('notificationPage', true, { expires: 424242 }); $('#btnnotifyPage').removeClass('btn-secondary').addClass('btn-success'); } }); $('#search > input').keypress(function (event) { if ( event.which == 13 ) { $('#mainMenu > a').dropdown('toggle'); } }); $('#search').submit(function () { app.goto('search',undefined,undefined,pagination + '/Any Tag/' + $('#search > input').val()); return false; }); function doSearch(searchstr) { var mpdtag='Any Tag'; $('#searchtags2 > button').each(function() { if ($(this).hasClass('btn-success')) { mpdtag=$(this).text(); } }); app.goto('search',undefined,undefined,pagination + '/' + mpdtag + '/' + searchstr); } $('#search2').submit(function () { return false; }); function addAllFromSearch() { var mpdtag='Any Tag'; $('#searchtags2 > button').each(function() { if ($(this).hasClass('btn-success')) { mpdtag=$(this).text(); } }); var searchstr=$('#searchstr2').val(); if (searchstr.length >= 2) { socket.send('MPD_API_SEARCH_ADD,' + mpdtag + ',' + searchstr); var rowCount = $('#searchList >tbody >tr').length; showNotification('Added '+rowCount+' songs from search','','','success'); } } $('#searchstr2').keyup(function (event) { pagination=0; doSearch($(this).val()); }); $('#searchtags2 > button').on('click',function (e) { $('#searchtags2 > button').removeClass('btn-success').addClass('btn-secondary'); $(this).removeClass('btn-secondary').addClass('btn-success'); $('#searchtags2desc').text($(this).text()); doSearch($('#searchstr2').val()); }); $('#searchqueuestr').keyup(function (event) { pagination=0; doQueueSearch(); }); $('#searchqueuetag > button').on('click',function (e) { $('#searchqueuetag > button').removeClass('btn-success').addClass('btn-secondary'); $(this).removeClass('btn-secondary').addClass('btn-success'); $('#searchqueuetagdesc').text($(this).text()); doQueueSearch(); }); function doQueueSearch() { var searchstr=$('#searchqueuestr').val(); var mpdtag='Any Tag'; $('#searchqueuetag > button').each(function() { if ($(this).hasClass('btn-success')) { mpdtag=$(this).text(); } }); app.goto('queue',undefined,undefined,pagination + '/' + mpdtag + '/' + searchstr); } $('#searchqueue').submit(function () { return false; }); $('#searchqueue').submit(function () { return false; }); function scrollToTop() { document.body.scrollTop = 0; // For Safari document.documentElement.scrollTop = 0; // For Chrome, Firefox, IE and Opera } function gotoPage(x,element,event) { switch (x) { case "next": pagination += MAX_ELEMENTS_PER_PAGE; break; case "prev": pagination -= MAX_ELEMENTS_PER_PAGE; if(pagination <= 0) pagination = 0; break; default: pagination = x; } switch(app.currentApp) { case "queue": if ($('#searchqueuestr').val().length >=3) { doQueueSearch(); } else { var mpdtag='Any Tag'; $('#searchqueuetag > button').each(function() { if ($(this).hasClass('btn-success')) { mpdtag=$(this).text(); } }); app.goto('queue',undefined,undefined, pagination+'/'+mpdtag+'/'); } break; case "search": doSearch($('#searchstr2').val()); break; case "browse": if (app.currentTab == 'Filesystem') app.goto('browse','Filesystem',undefined, pagination+'/'+filterLetter+'/'+browsepath) else if (app.currentTab == 'Playlists') app.goto('browse','Playlists',undefined, pagination+'/'+filterLetter) else if (app.currentTab == 'Database') { if (app.currentView == 'Artist') app.goto('browse','Database','Artist', pagination+'/'+filterLetter+'/') else if (app.currentView == 'Album') app.goto('browse','Database','Album', pagination+'/'+filterLetter+'/'+artist) } break; } event.preventDefault(); } function addStream() { if($('#streamurl').val().length > 0) { socket.send('MPD_API_ADD_TRACK,'+$('#streamurl').val()); } $('#streamurl').val(""); $('#addstream').modal('hide'); } function saveQueue() { if($('#playlistname').val().length > 0) { socket.send('MPD_API_SAVE_QUEUE,'+$('#playlistname').val()); } $('#savequeue').modal('hide'); } function showNotification(notificationTitle,notificationText,notificationHtml,notificationType) { if (Cookies.get('notificationWeb') === 'true') { var notification = new Notification(notificationTitle, {icon: 'assets/favicon.ico', body: notificationText}); setTimeout(function(notification) { notification.close(); }, 3000, notification); } if (Cookies.get('notificationPage') === 'true') { $.notify({ title: notificationTitle, message: notificationHtml},{ type: notificationType, offset: { y: 60, x:20 }, template: '

    ' }); } } function notificationsSupported() { return "Notification" in window; } function songChange(title, artist, album, uri) { var textNotification = ''; var htmlNotification = ''; var pageTitle = 'myMPD: '; if (typeof uri != 'undefined' && uri.length > 0) { var coverImg=''; if (uri.indexOf('http://') == 0 || uri.indexOf('https://') == 0 ) { coverImg='/assets/coverimage-httpstream.png'; } else if (coverImageFile != '') { coverImg='/library/'+uri.replace(/\/[^\/]+$/,'\/'+coverImageFile); } else { coverImg='/assets/coverimage-notavailable.png'; } $('#album-cover').css('backgroundImage','url("'+coverImg+'"),url("/assets/coverimage-notavailable.png")'); } if(typeof artist != 'undefined' && artist.length > 0 && artist != '-') { textNotification += artist; htmlNotification += '
    ' + artist; pageTitle += artist + ' - '; $('#artist').text(artist); } else { $('#artist').text(''); } if(typeof album != 'undefined' && album.length > 0 && album != '-') { textNotification += ' - ' + album; htmlNotification += '
    ' + album; $('#album').text(album); } else { $('#album').text(''); } if(typeof title != 'undefined' && title.length > 0) { pageTitle += title; $('#currenttrack').text(title); } else { $('#currenttrack').text(''); } document.title = pageTitle; showNotification(title,textNotification,htmlNotification,'success'); } $(document).keydown(function(e){ if (e.target.tagName == 'INPUT') { return; } switch (e.which) { case 37: //left socket.send('MPD_API_SET_PREV'); break; case 39: //right socket.send('MPD_API_SET_NEXT'); break; case 32: //space clickPlay(); break; default: return; } e.preventDefault(); }); function setFilterLetter(filter) { pagination = 0; if (app.currentApp == 'browse') { switch(app.currentTab) { case 'Filesystem': app.goto('browse','Filesystem',undefined, pagination+'/'+filter+'/'+browsepath); break; case 'Database': if (app.currentView == 'Artist') app.goto('browse','Database','Artist', pagination+'/'+filter+'/'); else if (app.currentView == 'Album') app.goto('browse','Database','Album', pagination+'/'+filter+'/'+artist); break; case 'Playlists': app.goto('browse','Playlists',undefined, pagination+'/'+filter+'/'); break; } } } function doSetFilterLetter(x) { $(x+'Letters > button').removeClass('btn-success').addClass('btn-secondary'); if (filterLetter == '0') { $(x).text('Filter: #'); $(x+'Letters > button').each(function() { if ($(this).text() == '#') { $(this).addClass('btn-success'); } }); } else if (filterLetter != '!') { $(x).text('Filter: '+filterLetter); $(x+'Letters > button').each(function() { if ($(this).text() == filterLetter) { $(this).addClass('btn-success'); } }); } else { $(x).text('Filter'); } } function add_filter (x) { $(x).append(''); $(x).append(''); for (i = 65; i <= 90; i++) { var c = String.fromCharCode(i); $(x).append(''); } } function chVolume (increment) { var aktValue=volumeBar.slider('getValue'); var newValue=aktValue+increment; if (newValue<0) { newValue=0; } else if (newValue > 100) { newValue=100; } volumeBar.slider('setValue',newValue); socket.send("MPD_API_SET_VOLUME,"+newValue); } function beautifyDuration(x) { var days = Math.floor(x / 86400); var hours = Math.floor(x / 3600) - days * 24; var minutes = Math.floor(x / 60) - hours * 60 - days * 1440; var seconds = x - days * 86400 - hours * 3600 - minutes * 60; return (days > 0 ? days + '\u2009d ' : '') + (hours > 0 ? hours + '\u2009h ' + (minutes < 10 ? '0' : '') : '') + minutes + '\u2009m ' + (seconds < 10 ? '0' : '') + seconds + '\u2009s'; } function genId(x) { return 'id'+x.replace(/[^\w]/g,''); }
    '+obj.data[item].track+''+obj.data[item].title+'
    folder_open' + basename(obj.data[item].dir) + '
    music_note' + obj.data[item].title + '' + obj.data[item].artist + '' + obj.data[item].album + '' + minutes + ':' + (seconds < 10 ? '0' : '') + seconds + '
    list' + basename(obj.data[item].plist) + '
    error_outlineNo results