<!doctype> <html> <!-- {{{ head --> <head> <link rel="icon" href="favicon.ico" /> <meta charset="utf-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>aria2 web client</title> <link rel="stylesheet" href="css/bootstrap.css"> <link rel="stylesheet" href="css/font-awesome.css"> <link rel="stylesheet" href="css/style.css"> <link rel="stylesheet" href="css/download.css"> <link rel="stylesheet" href="css/modals.css"> <!-- external javascript dependencies --> <script src="js/libs/jquery-1.8.3.js"></script> <script src="js/libs/underscore.js"></script> <script src="js/libs/angular.js"></script> <script src="js/libs/angularui-bootstrap.js"></script> <script src="js/libs/jquery.flot.js"></script> <!-- We do manual resize currently <script src="js/libs/jquery.flot.resize.js"></script> --> <!-- webui core --> <script src="js/init.js"></script> <script src="js/directives/chunkbar.js"></script> <script src="js/directives/dgraph.js"></script> <script src="js/directives/fselect.js"></script> <script src="js/filters/bytes.js"></script> <script src="js/filters/path.js"></script> <script src="js/services/constants.js"></script> <script src="js/services/deps.js"></script> <script src="js/services/base64.js"></script> <script src="js/services/utils.js"></script> <script src="js/services/modals.js"></script> <script src="js/services/alerts.js"></script> <script src="js/services/settings/settings.js"></script> <script src="js/services/settings/filters.js"></script> <script src="js/services/rpc/jsoncall.js"></script> <script src="js/services/rpc/sockcall.js"></script> <script src="js/services/rpc/syscall.js"></script> <script src="js/services/rpc/rpc.js"></script> <script src="js/services/rpc/helpers.js"></script> <script src="js/ctrls/modal.js"></script> <script src="js/ctrls/nav.js"></script> <script src="js/ctrls/download.js"></script> <script src="js/ctrls/alert.js"></script> </head> <!-- }}} --> <body ng-app="webui"> <!-- {{{ header --> <div class="navbar navbar-inverse navbar-fixed-top" ng-controller="NavCtrl"> <div class="navbar-inner"> <div class="container"> <!-- using a span for collapse rather than a to fix the angular majic on anchors --> <span class="btn btn-navbar" ng-click="collapsed = !collapsed"> <span class="icon-bar"></span> <span class="icon-bar"></span> <span class="icon-bar"></span> </span> <a class="brand">{{ name }}</a> <div class="nav-collapse" collapse="collapsed"> <ul class="nav"> <li class="dropdown"> <a class="dropdown-toggle" href="#"> Add <b class="caret"></b> </a> <ul class="dropdown-menu"> <li> <a href="#" ng-click="addUris()"> <i class="icon-download"></i> By URIs </a> </li> <li> <a href="#" ng-click="addTorrents()"> <i class="icon-file"></i> By Torrents </a> </li> <li> <a href="#" ng-click="addMetalinks()"> <i class="icon-file"></i> By Metalinks </a> </li> </ul> </li> <li class="dropdown"> <a class="dropdown-toggle" href="#"> Manage <b class="caret"></b></a> <ul class="dropdown-menu"> <li> <a href="#" ng-click="forcePauseAll()"><i class="icon-pause"></i> Pause All</a> </li> <li> <a href="#" ng-click="unpauseAll()"><i class="icon-play"></i> Resume Paused</a> </li> <li> <a href="#" ng-click="purgeDownloadResult()"><i class="icon-remove"></i> Purge Completed</a> </li> <!-- not adding remove all as requires many rpc syscalls to finish <li> <a href="#" ng-click="removeAll()"><i class="icon-fire"></i> Remove All</a> </li> --> </ul> </li> </ul> <ul class="nav"> <li class="dropdown"> <a href="#" class="dropdown-toggle">Settings <b class="caret"></b></a> <ul class="dropdown-menu"> <li> <a ng-click="changeCSettings()" href="#"><i class="icon-wrench"></i> Connection Settings</a> </li> <li> <a ng-click="changeGSettings()" href="#"><i class="icon-wrench"></i> Global Settings</a> </li> </ul> </li> </ul> <ul class="nav pull-right" ng-show="false"> <li class="dropdown"> <a class="dropdown-toggle" href="#">Miscellaneous <b class="caret"></b></a> <ul class="dropdown-menu"> <li> <a href="#"><i class="icon-list-alt"></i> Global Statistics</a> </li> <li> <a href="#"><i class="icon-info-sign"></i> About</a> </li> </ul> </li> </ul> </div><!--/.nav-collapse --> </div> </div> </div> <!-- }}} --> <div role="main" class="container" ng-controller="DownloadCtrl"> <!-- {{{ alerts --> <div ng-controller="AlertCtrl"> <div class="alert alert-{{alert.type}}" ng-repeat="alert in pendingAlerts"> <span ng-bind-html-unsafe="alert.msg"></span> <button type="button" class="close" ng-click="removeAlert($index)">x</button> </div> </div> <!-- }}} --> <!-- {{{ downloads --> <div ng-show="!getDownloads().length" class="hero-unit"> <h3>Currently no downloads in line to display, use the Add download button to start downloading files!</h3> </div> <!-- {{{ download template --> <table ng-repeat="download in getDownloads()" class="download" data-gid="{{download.gid}}"> <tbody> <tr> <td ng-click="download.collapsed = !download.collapsed" class="download-overview"> <b class="download-name">{{getName(download)}}</b> </td> <td class="download-controls" rowspan="2"> <!-- {{{ download control buttons --> <div class="btn-group"> <button ng-show="hasStatus(download, 'active')" class="btn" ng-click="pause(download)"> <i class="icon-pause"></i> </button> <button ng-show="hasStatus(download, 'paused')" class="btn" ng-click="resume(download)"> <i class="icon-play"></i> </button> <button ng-show="canRestart(download)" class="btn" ng-click="restart(download)"> <i class="icon-repeat"></i> </button> <button class="btn hidden-phone" ng-click="remove(download)"> <i class="icon-stop"></i> </button> <button class="btn hidden-phone" ng-show="['waiting', 'active'].indexOf( getType(download) )!= -1" ng-click="showSettings(download)"> <i class="icon-cog"></i> </button> <button class="btn dropdown-toggle"> <i class="caret"></i> </button> <ul class="dropdown-menu"> <li class="visible-phone"> <a ng-click="showSettings(download)" ng-show="['waiting', 'active'].indexOf( getType(download) )!= -1" href="#"><i class="icon-cog"></i> Settings</a> </li> <li ng-show="download.bittorrent"> <a href="#"><i class="icon-list-alt"></i> Peers</a> </li> <li> <a ng-click="download.collapsed = !download.collapsed" href="#"><i class="icon-info-sign"></i> More Info</a> </li> <li class="visible-phone"> <a ng-click="remove(download)" href="#"><i class="icon-remove"></i> Remove</a> </li> </ul> </div> <!-- }}} --> </td> </tr> <tr> <td ng-click="download.collapsed = !download.collapsed" class="download-overview"> <!-- {{{ statistics --> <ul class="stats pull-left"> <!-- {{{ active download statistics --> <li ng-show="hasStatus(download, 'active')" class="label label-success hidden-phone hidden-tablet"> <span>Status: {{download.status}}</span> </li> <li ng-show="hasStatus(download, 'active')" class="label label-success"> <span>ETA: {{getEta(download) | time}}</span> </li> <li ng-show="hasStatus(download, 'active')" class="label label-success hidden-phone"> <span>Size: {{download.totalLength | blength}}</span> </li> <li ng-show="hasStatus(download, 'active')" class="label label-success hidden-phone"> <span>Downloaded: {{download.completedLength | blength}}</span> </li> <li ng-show="hasStatus(download, 'active')" class="label label-success hidden-phone hidden-tablet"> <span>Progress: {{getProgress(download)}}%</span> </li> <li ng-show="hasStatus(download, 'active')" class="label label-success"> <span>Speed: {{download.downloadSpeed | bspeed}}</span> </li> <!-- TODO: figure out how to do multiple ng-show <li ng-show="download.bittorrent" class="label label-success hidden-phone">Upload Speed: <span class="download-uploadSpeed">{{download.uploadSpeed | bspeed}}</span></li> --> <!-- }}} --> <!-- {{{ paused download statistics --> <li ng-show="hasStatus(download, 'paused')" class="label label-info"> <span>Status: {{download.status}}</span> </li> <li ng-show="hasStatus(download, 'paused')" class="label label-info"> <span>Size: {{download.totalLength | blength}}</span> </li> <li ng-show="hasStatus(download, 'paused')" class="label label-info hidden-phone"> <span>Downloaded: {{download.completedLength | blength}}</span> </li> <li ng-show="hasStatus(download, 'paused')" class="label label-info hidden-phone"> <span>Path: {{download.dir}}</span> </li> <!-- }}} --> <!-- {{{ complete download statistics --> <li ng-show="hasStatus(download, 'complete')" class="label"> <span>Status: {{download.status}}</span> </li> <li ng-show="hasStatus(download, 'complete')" class="label hidden-phone"> <span>Path: {{download.dir}}</span> </li> <li ng-show="hasStatus(download, 'complete')" class="label"> <span>Size: {{download.totalLength | blength}}</span> </li> <!-- }}} --> <!-- {{{ removed download statistics --> <li ng-show="hasStatus(download, 'removed')" class="label label-warning"> <span>Status: {{download.status}}</span> </li> <li ng-show="hasStatus(download, 'removed')" class="label label-warning hidden-phone"> <span>Path: {{download.dir}}</span> </li> <li ng-show="hasStatus(download, 'removed')" class="label label-warning"> <span>Size: {{download.totalLength | blength}}</span> </li> <!-- }}} --> <!-- {{{ error download statistics --> <li ng-show="hasStatus(download, 'error')" class="label label-important"> <span>Status: {{download.status}}</span> </li> <li ng-show="hasStatus(download, 'error')" class="label label-important hidden-phone"> <span>Path: {{download.dir}}</span> </li> <li ng-show="hasStatus(download, 'error')" class="label label-important"> <span>Size: {{download.totalLength | blength}}</span> </li> <!-- }}} --> </ul> <!-- }}} --> </td> </tr> <tr> <td class="download-progress" colspan="2" ng-click="download.collapsed = !download.collapsed"> <div ng-show="hasStatus(download, 'active')" class="progress progress-striped" style="width: 100%; margin: 0; padding: 0;"> <div class="bar" style="width: {{getProgress(download)}}%;"></div> </div> <div ng-show="hasStatus(download, 'paused')" class="progress progress-info progress-striped" style="width: 100%; margin: 0; padding: 0;"> <div class="bar" style="width: {{getProgress(download)}}%;"></div> </div> <div ng-show="hasStatus(download, 'error')" class="progress progress-danger progress-striped" style="width: 100%; margin: 0; padding: 0;"> <div class="bar" style="width: {{getProgress(download)}}%;"></div> </div> <div ng-show="hasStatus(download, 'removed')" class="progress progress-warning progress-striped" style="width: 100%; margin: 0; padding: 0;"> <div class="bar" style="width: {{getProgress(download)}}%;"></div> </div> <div ng-show="hasStatus(download, 'complete')" class="progress progress-success progress-striped" style="width: 100%; margin: 0; padding: 0;"> <div class="bar" style="width: {{getProgress(download)}}%;"></div> </div> </td> </tr> <tr ng-click="download.collapsed = !download.collapsed" class="download-detail"> <td colspan="2"> <div collapse="download.collapsed"> <canvas bitfield="download.bitfield" draw="!download.collapsed" pieces="download.numPieces" class="progress chunk-canvas" width="1400" style="width: 100%; margin: 5px;" chunkbar></canvas> <ul class="stats"> <li class="label">Status: <span class="download-status">{{download.status}}</span></li> <li class="label">GID: <span class="download-gid">{{download.gid}}</span></li> <li class="label">Dir: <span class="download-dir">{{download.dir}}</span></li> <li class="label">Size: <span class="download-totalLength">{{download.totalLength | blength}}</span></li> <li class="label">Downloaded: <span class="download-completedLength">{{download.completedLength | blength}}</span></li> <li class="label">Num of Pieces: <span class="download-numPieces">{{download.numPieces}}</span></li> <li class="label">Piece Length: <span class="download-pieceLength">{{download.pieceLength | blength}}</span></li> <li class="label">ETA: <span class="download-eta">{{getEta(download) | time}}</span></li> <li class="label">Down Speed: <span class="download-downloadSpeed">{{download.downloadSpeed | bspeed}}</span></li> <li class="label" ng-show="download.bittorrent">Upload Speed: <span class="download-uploadSpeed">{{download.uploadSpeed | bspeed}}</span></li> <li class="label" ng-show="download.bittorrent">Upload Length: <span class="download-uploadLength">{{download.uploadLength | blength}}</span></li> <li class="label">Connections: <span class="download-connections">{{download.connections}}</span></li> </ul> <h4 class="hidden-phone">Download Files</h4> <ul class="download-files hidden-phone"> <li class="label" ng-repeat="file in download.files">{{file.path | prelative:download.dir}} ({{file.length | blength}})</li> </ul> <div ng-show="hasStatus(download, 'active')"> <div class="download-graph" dspeed="download.downloadSpeed" uspeed="download.uploadSpeed" dgraph draw="!download.collapsed"></div> </div> </div> </td> </tr> </tbody> </table> <!-- }}} --> <!-- }}} --> </div> <!-- {{{ modals --> <div ng-controller="ModalCtrl"> <!--{{{ add uri modal --> <div class="modal hide modal-adduris" modal="getUris.shown"> <div class="modal-header"> <button class="close" ng-click="getUris.close()">x</button> <h3>Add Downloads By URIs</h3> </div> <form class="modal-body"> <fieldset> <legend>Download URIs</legend> <textarea rows="6" ng-model="getUris.uris"></textarea> <p class="help-block"> You can add multiple downloads (files) at the same time by putting uris for each file on a separate line. <br> You can also add multiple uris (mirrors) for the *same* file. To do this separate the uris by a space. <br> Aria2 will use multiple uris (or mirrors) to boost the download speed for that file (download). <br> E.g. To add 2 files (downloads) f1.jpg and f2.mp4 with 2 uris (mirrors) each, add uris as following: <br> <!-- pre tags print tabs, so do not indent them! --> <pre> http://ex1.com/f1.jpg http://ex2.com/f1.jpg http://ex1.com/f2.mp4 http://ex2.com/f2.mp4 </pre> A URI can be HTTP(S)/FTP/BitTorrent Magnet URI. </p> </fieldset> </form> <div class="modal-footer"> <button class="btn" ng-click="getUris.close()">Cancel</button> <button class="btn btn-primary" ng-click="getUris.success()">Start</button> </div> </div> <!-- }}} --> <!-- {{{ add torrent modal --> <div class="modal hide" modal="getTorrents.shown"> <div class="modal-header"> <button class="close" ng-click="getTorrents.close()">x</button> <h3>Add Downloads By Torrents</h3> </div> <form class="modal-body"> <fieldset> <legend>Select Torrents</legend> <input type="file" class="input-xlarge" fselect="getTorrents.files" multiple/> <p class="help-block"> Select the torrent from the local filesystem to start the download. <br> You can select multiple torrents to start multiple downloads <br> To add a magnet torrent url, use the add url option and add it there. </p> </fieldset> </form> <div class="modal-footer"> <button class="btn" ng-click="getTorrents.close()">Cancel</button> <button class="btn btn-primary" ng-click="getTorrents.success()">Start</button> </div> </div <!-- }}} --> <!-- {{{ add metalink modal --> <div class="modal hide" modal="getMetalinks.shown"> <div class="modal-header"> <button class="close" ng-click="getMetalinks.close()">x</button> <h3>Add Downloads By Metalinks</h3> </div> <form class="modal-body"> <fieldset> <legend>Select Metalinks</legend> <input type="file" class="input-xlarge" fselect="getMetalinks.files" multiple/> <p class="help-block"> Select the metalink from the local filesystem to start the download. <br> You can select multiple metalinks to start multiple downloads </p> </fieldset> </form> <div class="modal-footer"> <button class="btn" ng-click="getMetalinks.close()">Cancel</button> <button class="btn btn-primary" ng-click="getMetalinks.success()">Start</button> </div> </div <!-- }}} --> <!-- {{{ settings modal --> <div class="modal hide" modal="settings.shown"> <div class="modal-header"> <button class="close" ng-click="settings.close()">x</button> <h3>{{settings.title}}</h3> </div> <form class="form-horizontal modal-body"> <fieldset> <div class="control-group"> <div ng-repeat="(name, set) in settings.settings"> <label class="control-label">{{name}}</label> <div class="controls"> <select ng-show="set.options.length" ng-options="opt for opt in set.options" ng-model="set.val"> </select> <input ng-show="!set.options.length" typ="text" class="input-xlarge" ng-model="set.val"/> <p class="help-block">{{set.desc}}</p> </div> <br><br> </div> </div> </fieldset> </form> <div class="modal-footer"> <button class="btn" ng-click="settings.close()">Cancel</button> <button class="btn btn-success" ng-click="settings.success()">Save</button> </div> </div <!-- }}} --> <!--{{{ connection modal --> <div class="modal hide" modal="connection.shown"> <div class="modal-header"> <button class="close" ng-click="connection.close()">x</button> <h3>Connection Settings</h3> </div> <div class="modal-body"> <form class="form-horizontal"> <fieldset> <legend>Aria2 RPC host and port</legend> <div class="control-group"> <label class="control-label">Enter the host:</label> <div class="controls"> <div class="input-prepend"> <span class="add-on">http(s)://</span><input type="text" class="input-xlarge"/> </div> <p class="help-block"> Enter the ip or dns name of the server on which the rpc for aria2 is running (default = localhost) </p> </div> <label class="control-label">Enter the port:</label> <div class="controls"> <input type="text" class="input-xlarge" ng-model="connection.port" placeholder="{{connection.port}}"/> <p class="help-block"> Enter the port of the server on which the rpc for aria2 is running (default = 6800) </p> </div> <label class="control-label">SSL/TLS encryption:</label> <div class="controls"> <button type="button" class="btn" data-toggle="button">ON</button> <p class="help-block">Enable SSL/TLS encryption.</p> </div> <label class="control-label">Enter the username (optional):</label> <div class="controls"> <input type="text" class="input-xlarge" ng-model="connection.auth.user"/> <p class="help-block"> Enter the aria2 RPC username (empty if authentication not enabled) </p> </div> <label class="control-label">Enter the password (optional):</label> <div class="controls"> <input type="password" class="input-xlarge" ng-model="connection.auth.pass"/> <p class="help-block"> Enter the aria2 RPC password (empty if authentication not enabled) </p> </div> </div> </fieldset> </form> </div> <div class="modal-footer"> <button href="#" class="btn" ng-click="connection.close()">Retry with default configuration</button> <button href="#" class="btn btn-primary" ng-click="connection.close()">Use custom IP and port settings</button> </div> </div> <!-- }}} --> </div> <!-- }}} --> </body> </html>