/* * Copyright © Dave Perrett and Malcolm Jarvis * This code is licensed under the GPL version 2. * For details, see http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * * Class Transmission */ function Transmission(){ this.initialize(); } Transmission.prototype = { /*-------------------------------------------- * * C O N S T R U C T O R * *--------------------------------------------*/ initialize: function() { // Initialize the helper classes this.remote = new TransmissionRemote(this); // Initialize the implementation fields this._current_search = ''; this._torrents = { }; this._rows = [ ]; this._prefs = { }; // Initialize the clutch preferences Prefs.getClutchPrefs( this ); this.preloadImages(); // Set up user events var tr = this; $('#pause_all_link').bind('click', function(e){ tr.stopAllClicked(e); }); $('#resume_all_link').bind('click', function(e){ tr.startAllClicked(e); }); $('#pause_selected_link').bind('click', function(e){ tr.stopSelectedClicked(e); } ); $('#resume_selected_link').bind('click', function(e){ tr.startSelectedClicked(e); }); $('#remove_link').bind('click', function(e){ tr.removeClicked(e); }); $('#filter_all_link').parent().bind('click', function(e){ tr.showAllClicked(e); }); $('#filter_active_link').parent().bind('click', function(e){ tr.showActiveClicked(e); }); $('#filter_downloading_link').parent().bind('click', function(e){ tr.showDownloadingClicked(e); }); $('#filter_seeding_link').parent().bind('click', function(e){ tr.showSeedingClicked(e); }); $('#filter_paused_link').parent().bind('click', function(e){ tr.showPausedClicked(e); }); $('.inspector_tab').bind('click', function(e){ tr.inspectorTabClicked(e, this); }); $('.file_wanted_control').live('click', function(e){ tr.fileWantedClicked(e, this); }); $('.file_priority_control').live('click', function(e){ tr.filePriorityClicked(e, this); }); $('#files_select_all').live('click', function(e){ tr.filesSelectAllClicked(e, this); }); $('#files_deselect_all').live('click', function(e){ tr.filesDeselectAllClicked(e, this); }); $('#open_link').bind('click', function(e){ tr.openTorrentClicked(e); }); $('#upload_confirm_button').bind('click', function(e){ tr.confirmUploadClicked(e); return false;}); $('#upload_cancel_button').bind('click', function(e){ tr.cancelUploadClicked(e); return false; }); $('#turtle_button').bind('click', function(e){ tr.toggleTurtleClicked(e); return false; }); $('#prefs_tab_general_tab').click(function(e){ changeTab(this, 'prefs_tab_general') }); $('#prefs_tab_speed_tab').click(function(e){ changeTab(this, 'prefs_tab_speed') }); if (iPhone) { $('#inspector_close').bind('click', function(e){ tr.hideInspector(); }); $('#preferences_link').bind('click', function(e){ tr.releaseClutchPreferencesButton(e); }); } else { $(document).bind('keydown', function(e){ tr.keyDown(e); }); $('#torrent_container').bind('click', function(e){ tr.deselectAll( true ); }); $('#filter_toggle_link').bind('click', function(e){ tr.toggleFilterClicked(e); }); $('#inspector_link').bind('click', function(e){ tr.toggleInspectorClicked(e); }); this.setupSearchBox(); this.createContextMenu(); this.createSettingsMenu(); } this.initTurtleDropDowns(); this.initPrefs(); this._torrent_list = $('#torrent_list')[0]; this._inspector_file_list = $('#inspector_file_list')[0]; this._inspector_peers_list = $('#inspector_peers_list')[0]; this._inspector_trackers_list = $('#inspector_trackers_list')[0]; this._inspector_tab_files = $('#inspector_tab_files')[0]; this._toolbar_buttons = $('#torrent_global_menu ul li'); this._toolbar_pause_button = $('li#pause_selected')[0]; this._toolbar_pause_all_button = $('li#pause_all')[0]; this._toolbar_start_button = $('li#resume_selected')[0]; this._toolbar_start_all_button = $('li#resume_all')[0]; this._toolbar_remove_button = $('li#remove')[0]; this._context_pause_button = $('li#context_pause_selected')[0]; this._context_start_button = $('li#context_resume_selected')[0]; var ti = '#torrent_inspector_'; this._inspector = { }; this._inspector._info_tab = { }; this._inspector._info_tab.availability = $(ti+'availability')[0]; this._inspector._info_tab.comment = $(ti+'comment')[0]; this._inspector._info_tab.creator_date = $(ti+'creator_date')[0]; this._inspector._info_tab.creator = $(ti+'creator')[0]; this._inspector._info_tab.download_dir = $(ti+'download_dir')[0]; this._inspector._info_tab.downloaded = $(ti+'downloaded')[0]; this._inspector._info_tab.download_from = $(ti+'download_from')[0]; this._inspector._info_tab.download_speed = $(ti+'download_speed')[0]; this._inspector._info_tab.error = $(ti+'error')[0]; this._inspector._info_tab.hash = $(ti+'hash')[0]; this._inspector._info_tab.have = $(ti+'have')[0]; this._inspector._info_tab.name = $(ti+'name')[0]; this._inspector._info_tab.progress = $(ti+'progress')[0]; this._inspector._info_tab.ratio = $(ti+'ratio')[0]; this._inspector._info_tab.secure = $(ti+'secure')[0]; this._inspector._info_tab.size = $(ti+'size')[0]; this._inspector._info_tab.state = $(ti+'state')[0]; this._inspector._info_tab.pieces = $(ti+'pieces')[0]; this._inspector._info_tab.uploaded = $(ti+'uploaded')[0]; this._inspector._info_tab.upload_speed = $(ti+'upload_speed')[0]; this._inspector._info_tab.upload_to = $(ti+'upload_to')[0]; // Setup the preference box this.setupPrefConstraints(); // Setup the prefs gui this.initializeSettings( ); // Get preferences & torrents from the daemon var tr = this; var async = false; this.loadDaemonPrefs( async ); this.loadDaemonStats( async ); this.initializeAllTorrents(); this.togglePeriodicRefresh( true ); this.togglePeriodicSessionRefresh( true ); }, loadDaemonPrefs: function( async ){ var tr = this; this.remote.loadDaemonPrefs( function(data){ var o = data.arguments; Prefs.getClutchPrefs( o ); tr.updatePrefs( o ); }, async ); }, loadDaemonStats: function( async ){ var tr = this; this.remote.loadDaemonStats( function(data){ var o = data.arguments; tr.updateStats( o ); }, async ); }, preloadImages: function() { if (iPhone) { this.loadImages( 'images/buttons/info_general.png', 'images/buttons/info_activity.png', 'images/buttons/info_files.png', 'images/buttons/toolbar_buttons.png', 'images/graphics/filter_bar.png', 'images/graphics/iphone_chrome.png', 'images/graphics/logo.png' ); } else { this.loadImages( 'images/buttons/info_general.png', 'images/buttons/info_activity.png', 'images/buttons/info_files.png', 'images/buttons/tab_backgrounds.png', 'images/buttons/toolbar_buttons.png', 'images/buttons/torrent_buttons.png', 'images/buttons/file_wanted_buttons.png', 'images/buttons/file_priority_buttons.png', 'images/graphics/chrome.png', 'images/graphics/filter_bar.png', 'images/graphics/logo.png', 'images/graphics/transfer_arrows.png', 'images/progress/progress.png' ); } }, loadImages: function() { for( var i=0, row; row=arguments[i]; ++i ) jQuery("").attr("src", row); }, /* * Set up the preference validation */ setupPrefConstraints: function() { // only allow integers for speed limit & port options $('div.preference input[@type=text]:not(#download_location)').blur( function() { this.value = this.value.replace(/[^0-9]/gi, ''); if (this.value == '') { if ($(this).is('#refresh_rate')) { this.value = 5; } else { this.value = 0; } } }); }, /* * Load the clutch prefs and init the GUI according to those prefs */ initializeSettings: function( ) { Prefs.getClutchPrefs( this ); // iPhone conditions in the section allow us to not // include transmenu js to save some bandwidth; if we // start using prefs on iPhone we need to weed // transmenu refs out of that too. $('#filter_' + this[Prefs._FilterMode] + '_link').parent().addClass('selected'); if (!iPhone) $('#sort_by_' + this[Prefs._SortMethod] ).selectMenuItem(); if (!iPhone && ( this[Prefs._SortDirection] == Prefs._SortDescending ) ) $('#reverse_sort_order').selectMenuItem(); if( this[Prefs._ShowFilter] ) this.showFilter( ); if( !iPhone && this[Prefs._ShowInspector] ) this.showInspector( ); if( !iPhone && this[Prefs._CompactDisplayState] ) $('#compact_view').selectMenuItem(); }, /* * Set up the search box */ setupSearchBox: function() { var tr = this; var search_box = $('#torrent_search'); search_box.bind('keyup click', {transmission: this}, function(event) { tr.setSearch(this.value); }); if (!$.browser.safari) { search_box.addClass('blur'); search_box[0].value = 'Filter'; search_box.bind('blur', {transmission: this}, function(event) { if (this.value == '') { $(this).addClass('blur'); this.value = 'Filter'; tr.setSearch(null); } }).bind('focus', {}, function(event) { if ($(this).is('.blur')) { this.value = ''; $(this).removeClass('blur'); } }); } }, contextStopSelected: function( ) { this.stopSelectedTorrents( ); }, contextStartSelected: function( ) { this.startSelectedTorrents( ); }, contextRemoveSelected: function( ) { this.removeSelectedTorrents( ); }, contextRemoveDataSelected: function( ) { this.removeSelectedTorrentsAndData( ); }, contextVerifySelected: function( ) { this.verifySelectedTorrents( ); }, contextToggleInspector: function( ) { this.toggleInspector( ); }, contextSelectAll: function( ) { this.selectAll( true ); }, contextDeselectAll: function( ) { this.deselectAll( true ); }, /* * Create the torrent right-click menu */ createContextMenu: function() { var tr = this; var bindings = { context_pause_selected: function(e){ tr.contextStopSelected(e); }, context_resume_selected: function(e){ tr.contextStartSelected(e); }, context_remove: function(e){ tr.contextRemoveSelected(e); }, context_removedata: function(e){ tr.contextRemoveDataSelected(e); }, context_verify: function(e){ tr.contextVerifySelected(e); }, context_toggle_inspector: function(e){ tr.contextToggleInspector(e); }, context_select_all: function(e){ tr.contextSelectAll(e); }, context_deselect_all: function(e){ tr.contextDeselectAll(e); } }; // Setup the context menu $('ul#torrent_list').contextMenu('torrent_context_menu', { bindings: bindings, menuStyle: Menu.context.menu_style, itemStyle: Menu.context.item_style, itemHoverStyle: Menu.context.item_hover_style, itemDisabledStyle: Menu.context.item_disabled_style, shadow: false, boundingElement: $('div#torrent_container'), boundingRightPad: 20, boundingBottomPad: 5, onContextMenu: function(e) { var closestRow = $(e.target).closest('.torrent')[0]._torrent; if(!closestRow.isSelected()) tr.setSelectedTorrent( closestRow, true ); return true; } }); }, /* * Create the footer settings menu */ createSettingsMenu: function() { var tr = this; $('#settings_menu').transMenu({ selected_char: '✔', direction: 'up', onClick: function(e){ return tr.processSettingsMenuEvent(e); } }); $('#unlimited_download_rate').selectMenuItem(); $('#unlimited_upload_rate').selectMenuItem(); }, initTurtleDropDowns: function() { var i, out, hour, mins, start, end, value, content; // Build the list of times out = ""; start = $('#alt-speed-time-begin')[0]; end = $('#alt-speed-time-end')[0]; for (i = 0; i < 24 * 4; i++) { hour = parseInt(i / 4); mins = ((i % 4) * 15); value = (i * 15); content = hour + ":" + (mins == 0 ? "00" : mins); start.options[i] = new Option(content, value); end.options[i] = new Option(content, value); } }, /*-------------------------------------------- * * U T I L I T I E S * *--------------------------------------------*/ getAllTorrents: function() { var torrents = []; for(var key in this._torrents) torrents.push(this._torrents[key]); return torrents; }, getVisibleTorrents: function() { var torrents = [ ]; for( var i=0, row; row=this._rows[i]; ++i ) if( row._torrent && ( row[0].style.display != 'none' ) ) torrents.push( row._torrent ); return torrents; }, getSelectedTorrents: function() { var v = this.getVisibleTorrents( ); var s = [ ]; for( var i=0, row; row=v[i]; ++i ) if( row.isSelected( ) ) s.push( row ); return s; }, getDeselectedTorrents: function() { var visible_torrent_ids = jQuery.map(this.getVisibleTorrents(), function(t) { return t.id(); } ); var s = [ ]; jQuery.each( this.getAllTorrents( ), function() { var visible = (-1 != jQuery.inArray(this.id(), visible_torrent_ids)); if (!this.isSelected() || !visible) s.push( this ); } ); return s; }, getVisibleRows: function() { var rows = [ ]; for( var i=0, row; row=this._rows[i]; ++i ) if( row[0].style.display != 'none' ) rows.push( row ); return rows; }, getTorrentIndex: function( rows, torrent ) { for( var i=0, row; row=rows[i]; ++i ) if( row._torrent == torrent ) return i; return null; }, setPref: function( key, val ) { this[key] = val; Prefs.setValue( key, val ); }, scrollToElement: function( e ) { if( iPhone ) return; var container = $('#torrent_container'); var scrollTop = container.scrollTop( ); var innerHeight = container.innerHeight( ); var offsetTop = e[0].offsetTop; var offsetHeight = e.outerHeight( ); if( offsetTop < scrollTop ) container.scrollTop( offsetTop ); else if( innerHeight + scrollTop < offsetTop + offsetHeight ) container.scrollTop( offsetTop + offsetHeight - innerHeight ); }, seedRatioLimit: function(){ if(this._prefs && this._prefs['seedRatioLimited']) return this._prefs['seedRatioLimit']; else return -1; }, /*-------------------------------------------- * * S E L E C T I O N * *--------------------------------------------*/ setSelectedTorrent: function( torrent, doUpdate ) { this.deselectAll( ); this.selectTorrent( torrent, doUpdate ); }, selectElement: function( e, doUpdate ) { e.addClass('selected'); this.scrollToElement( e ); if( doUpdate ) this.selectionChanged( ); }, selectRow: function( rowIndex, doUpdate ) { this.selectElement( this._rows[rowIndex], doUpdate ); }, selectTorrent: function( torrent, doUpdate ) { if( torrent._element ) this.selectElement( torrent._element, doUpdate ); }, deselectElement: function( e, doUpdate ) { e.removeClass('selected'); if( doUpdate ) this.selectionChanged( ); }, deselectTorrent: function( torrent, doUpdate ) { if( torrent._element ) this.deselectElement( torrent._element, doUpdate ); }, selectAll: function( doUpdate ) { var tr = this; for( var i=0, row; row=tr._rows[i]; ++i ) tr.selectElement( row ); if( doUpdate ) tr.selectionChanged(); }, deselectAll: function( doUpdate ) { var tr = this; for( var i=0, row; row=tr._rows[i]; ++i ) tr.deselectElement( row ); tr._last_torrent_clicked = null; if( doUpdate ) tr.selectionChanged( ); }, /* * Select a range from this torrent to the last clicked torrent */ selectRange: function( torrent, doUpdate ) { if( !this._last_torrent_clicked ) { this.selectTorrent( torrent ); } else // select the range between the prevous & current { var rows = this.getVisibleRows( ); var i = this.getTorrentIndex( rows, this._last_torrent_clicked ); var end = this.getTorrentIndex( rows, torrent ); var step = i < end ? 1 : -1; for( ; i!=end; i+=step ) this.selectRow( i ); this.selectRow( i ); } if( doUpdate ) this.selectionChanged( ); }, selectionChanged: function() { this.updateButtonStates(); this.updateInspector(); this.updateSelectedData(); }, /*-------------------------------------------- * * E V E N T F U N C T I O N S * *--------------------------------------------*/ /* * Process key event */ keyDown: function(event) { var tr = this; var sel = tr.getSelectedTorrents( ); var rows = tr.getVisibleRows( ); var i = -1; if( event.keyCode == 40 ) // down arrow { var t = sel.length ? sel[sel.length-1] : null; i = t==null ? null : tr.getTorrentIndex(rows,t)+1; if( i == rows.length || i == null ) i = 0; } else if( event.keyCode == 38 ) // up arrow { var t = sel.length ? sel[0] : null i = t==null ? null : tr.getTorrentIndex(rows,t)-1; if( i == -1 || i == null ) i = rows.length - 1; } if( 0<=i && i 0 ) this.periodicTorrentUpdate( ids ); else this.periodicTorrentUpdate( false ); }, updateTurtleButton: function() { var w = $('#turtle_button'); var t; if ( this._prefs[RPC._TurtleState] ) { w.addClass('turtleEnabled'); w.removeClass('turtleDisabled'); t = "Click to disable Temporary Speed Limits"; } else { w.removeClass('turtleEnabled'); w.addClass('turtleDisabled'); t = "Click to enable Temporary Speed Limits"; } t += " (" + Transmission.fmt.speed(this._prefs[RPC._TurtleUpSpeedLimit]) + " up, " + Transmission.fmt.speed(this._prefs[RPC._TurtleDownSpeedLimit]) + " down)"; w.attr( 'title', t ); }, /*-------------------------------------------- * * I N T E R F A C E F U N C T I O N S * *--------------------------------------------*/ showPrefsDialog: function( ) { var tr = this; var preferredWidth = 450; var preferredHeight = 400; var windowWidth = $(window).width(); var windowHeight = $(window).height(); var fullscreen = (windowWidth<=preferredWidth) || (windowHeight<=preferredHeight); var args = { close: function(event, ui) { tr.togglePeriodicSessionRefresh( true ); }, width: fullscreen ? windowWidth : preferredWidth, height: fullscreen ? windowHeight : preferredHeight, resizable: !fullscreen, draggable: !fullscreen, title: 'Preferences', hide: 'blind', buttons: { } }; if( fullscreen ) args.position = [ 0, 0 ]; else args.buttons = { 'Close': function() { $(this).dialog('close'); } }; $("#prefs_dialog" ).dialog( args ); tr.togglePeriodicSessionRefresh( false ); }, prefsKeys: [ 'alt-speed-down', 'alt-speed-enabled', 'alt-speed-time-begin', 'alt-speed-time-enabled', 'alt-speed-time-end', 'alt-speed-time-day', 'alt-speed-up', 'blocklist-enabled', 'blocklist-size', 'cache-size', 'config-dir', 'download-dir', 'dht-enabled', 'encryption', 'idle-seeding-limit', 'idle-seeding-limit-enabled', 'incomplete-dir', 'incomplete-dir-enabled', 'lpd-enabled', 'peer-limit-global', 'peer-limit-per-torrent', 'pex-enabled', 'peer-port', 'peer-port-random-on-start', 'port-forwarding-enabled', 'rename-partial-files', 'rpc-version', 'rpc-version-minimum', 'script-torrent-done-filename', 'script-torrent-done-enabled', 'seedRatioLimit', 'seedRatioLimited', 'speed-limit-down', 'speed-limit-down-enabled', 'speed-limit-up', 'speed-limit-up-enabled', 'start-added-torrents', 'trash-original-torrent-files', 'version' ], setPrefIfDifferent: function( key, value ) { var tr = this; if( tr._prefs[key] != value ) { var args = { }; args[key] = value; tr.updatePrefs( args ); // tell our local copy about the change tr.remote.savePrefs( args ); // tell our server about the change } }, initPrefs: function( ) { var tr = this; $("#update_blocklist_button").button(); $("#prefs_tabs").tabs(); $.each( this.prefsKeys, function( index, key ) { $(':input#'+key).change( function( e ) { var control = e.target; var value; if( control.type == 'checkbox' ) value = control.checked; else { value = parseFloat( e.target.value ); if( value == NaN ) value = parseInt( e.target.value ); if( value == NaN ) value = e.target.value; } tr.setPrefIfDifferent( key, value ); if( key == 'peer-port' ) { // changing the port invalidates the port status delete tr._prefs['port-is-open']; tr.updatePrefs( tr._prefs ); } }); }); $('#port-status-spinner').hide(); $(':button#port-test').click(function(e) { e.target.disabled = true; $('#port-status-label' ).html( 'Testing...' ); var img = $('#port-status-spinner'); img.show(); var args = { }; tr.remote.portTest( function(data){ //console.log(data.arguments); e.target.disabled = false; img.hide(); tr.updatePrefs( data.arguments ); }, true ); }); $('#blocklist-update-spinner').hide(); $(':button#blocklist-update').click(function(e) { e.target.disabled = true; var img = $('#blocklist-update-spinner'); img.show(); var args = { }; tr.remote.blocklistUpdate( function(data){ console.log(data.arguments); e.target.disabled = false; img.hide(); tr.updatePrefs( data.arguments ); }, true ); }); }, /* * Process got some new session data from the server */ updatePrefs: function( prefs ) { var tr = this; var updateTurtle = false; // remember them for later $.extend(this._prefs, prefs); // update the prefs dialog var key; for( key in prefs ) { $(':input#'+key).each(function(index,e) { if( e.type=='checkbox') e.checked = prefs[key]; else e.value = prefs[key]; }); // extra pieces if( key === RPC._DownSpeedLimit ) $('#limited_download_rate').html('Limit (' + Transmission.fmt.speed(this._prefs[RPC._DownSpeedLimit]) + ')' ); else if( key === RPC._DownSpeedLimited ) $(this._prefs[RPC._DownSpeedLimited] ? '#limited_download_rate' : '#unlimited_download_rate').deselectMenuSiblings().selectMenuItem(); else if( key == RPC._UpSpeedLimit ) $('#limited_upload_rate').html('Limit (' + Transmission.fmt.speed(this._prefs[RPC._UpSpeedLimit]) + ')' ); else if( key === RPC._UpSpeedLimited ) $(this._prefs[RPC._UpSpeedLimited] ? '#limited_upload_rate' : '#unlimited_upload_rate').deselectMenuSiblings().selectMenuItem(); else if( key === RPC._TurtleState ) updateTurtle = true; else if( key === RPC._TurtleUpSpeedLimit ) updateTurtle = true; else if( key === RPC._TurtleDownSpeedLimit ) updateTurtle = true; else if( key == 'blocklist-size' ) $('#' + key).html( '' + prefs[key] ); else if( key == 'port-is-open' ) $('#port-status-label' ).html( 'Port is ' + (prefs[key] ? 'open' : 'closed') ); } if( this._prefs['port-is-open'] == undefined ) $('#port-status-label').html( 'Status unknown' ); if( updateTurtle ) this.updateTurtleButton(); }, showStatsDialog: function( ) { var tr = this; tr.loadDaemonStats(); tr.togglePeriodicStatsRefresh(true); $("#stats_dialog" ).dialog( { title: 'Statistics', hide: 'blind', width: 266, buttons: { 'Close': function() { tr.togglePeriodicStatsRefresh(false); $(this).dialog('close'); } } }); }, /* * Process got some new session stats from the server */ updateStats: function( stats ) { // can't think of a reason to remember this //this._stats = stats; var session = stats["current-stats"]; var total = stats["cumulative-stats"]; setInnerHTML( $('#stats_session_uploaded')[0], Transmission.fmt.size(session["uploadedBytes"]) ); setInnerHTML( $('#stats_session_downloaded')[0], Transmission.fmt.size(session["downloadedBytes"]) ); setInnerHTML( $('#stats_session_ratio')[0], Transmission.fmt.ratioString(Math.ratio(session["uploadedBytes"],session["downloadedBytes"]))); setInnerHTML( $('#stats_session_duration')[0], Transmission.fmt.timeInterval(session["secondsActive"]) ); setInnerHTML( $('#stats_total_count')[0], 'Started ' + total["sessionCount"] + " times" ); setInnerHTML( $('#stats_total_uploaded')[0], Transmission.fmt.size(total["uploadedBytes"]) ); setInnerHTML( $('#stats_total_downloaded')[0], Transmission.fmt.size(total["downloadedBytes"]) ); setInnerHTML( $('#stats_total_ratio')[0], Transmission.fmt.ratioString(Math.ratio(total["uploadedBytes"],total["downloadedBytes"]))); setInnerHTML( $('#stats_total_duration')[0], Transmission.fmt.timeInterval(total["secondsActive"]) ); }, setSearch: function( search ) { this._current_search = search ? search.trim() : null; this.refilter( ); }, setSortMethod: function( sort_method ) { this.setPref( Prefs._SortMethod, sort_method ); this.refilter( ); }, setSortDirection: function( direction ) { this.setPref( Prefs._SortDirection, direction ); this.refilter( ); }, /* * Process an event in the footer-menu */ processSettingsMenuEvent: function(event) { var tr = this; var $element = $(event.target); // Figure out which menu has been clicked switch ($element.parent()[0].id) { // Display the preferences dialog case 'footer_super_menu': if ($element[0].id == 'preferences') { $('div#prefs_container div#pref_error').hide(); $('div#prefs_container h2.dialog_heading').show(); tr.showPrefsDialog( ); } else if ($element[0].id == 'statistics') { $('div#stats_container div#stats_error').hide(); $('div#stats_container h2.dialog_heading').show(); tr.showStatsDialog( ); } else if ($element[0].id == 'compact_view') { this.setPref( Prefs._CompactDisplayState, !this[Prefs._CompactDisplayState]) if(this[Prefs._CompactDisplayState]) $element.selectMenuItem(); else $element.deselectMenuItem(); this.refreshDisplay( ); } break; // Limit the download rate case 'footer_download_rate_menu': //var args = { }; if ($element.is('#unlimited_download_rate')) { $element.deselectMenuSiblings().selectMenuItem(); tr.setPrefIfDifferent( RPC._DownSpeedLimited, false ); } else { var rate_str = ($element[0].innerHTML).replace(/[^0-9]/ig, ''); var rate_val = parseInt( rate_str ); setInnerHTML( $('#limited_download_rate')[0], 'Limit (' + Transmission.fmt.speed(rate_val) + ')' ); $('#limited_download_rate').deselectMenuSiblings().selectMenuItem(); tr.setPrefIfDifferent( RPC._DownSpeedLimit, rate_val ); tr.setPrefIfDifferent( RPC._DownSpeedLimited, true ); } break; // Limit the upload rate case 'footer_upload_rate_menu': var args = { }; if ($element.is('#unlimited_upload_rate')) { $element.deselectMenuSiblings().selectMenuItem(); tr.setPrefIfDifferent( RPC._UpSpeedLimited, false ); } else { var rate_str = ($element[0].innerHTML).replace(/[^0-9]/ig, ''); var rate_val = parseInt( rate_str ); setInnerHTML( $('#limited_upload_rate')[0], 'Limit (' + Transmission.fmt.speed(rate_val) + ')' ); $('#limited_upload_rate').deselectMenuSiblings().selectMenuItem(); tr.setPrefIfDifferent( RPC._UpSpeedLimit, rate_val ); tr.setPrefIfDifferent( RPC._UpSpeedLimited, true ); } break; // Sort the torrent list case 'footer_sort_menu': // The 'reverse sort' option state can be toggled independently of the other options if ($element.is('#reverse_sort_order')) { if(!$element.is('#reverse_sort_order.active')) break; var dir; if ($element.menuItemIsSelected()) { $element.deselectMenuItem(); dir = Prefs._SortAscending; } else { $element.selectMenuItem(); dir = Prefs._SortDescending; } tr.setSortDirection( dir ); // Otherwise, deselect all other options (except reverse-sort) and select this one } else { $element.parent().find('span.selected').each( function() { if (! $element.parent().is('#reverse_sort_order')) { $element.parent().deselectMenuItem(); } }); $element.selectMenuItem(); var method = $element[0].id.replace(/sort_by_/, ''); tr.setSortMethod( method ); } break; } $('#settings_menu').trigger('closemenu'); return false; // to prevent the event from bubbling up }, setLastTorrentClicked: function( torrent ) { this._last_torrent_clicked = torrent; }, /* * Update the inspector with the latest data for the selected torrents */ updateInspector: function() { if( !this[Prefs._ShowInspector] ) return; var torrents = this.getSelectedTorrents( ); if( !torrents.length && iPhone ) { this.hideInspector(); return; } var creator = 'N/A'; var comment = 'N/A'; var download_dir = 'N/A'; var date_created = 'N/A'; var error = 'None'; var hash = 'N/A'; var have_public = false; var have_private = false; var name; var sizeWhenDone = 0; var sizeDone = 0; var total_completed = 0; var total_download = 0; var total_download_peers = 0; var total_download_speed = 0; var total_availability = 0; var total_have = 0; var total_size = 0; var total_state = null; var pieces = 'N/A'; var total_upload = 0; var total_upload_peers = 0; var total_upload_speed = 0; var total_verified = 0; var na = 'N/A'; var tab = this._inspector._info_tab; $("#torrent_inspector_size, .inspector_row div").css('color', '#222'); if( torrents.length == 0 ) { setInnerHTML( tab.name, 'No Selection' ); setInnerHTML( tab.size, na ); setInnerHTML( tab.pieces, na ); setInnerHTML( tab.hash, na ); setInnerHTML( tab.state, na ); setInnerHTML( tab.download_speed, na ); setInnerHTML( tab.upload_speed, na ); setInnerHTML( tab.uploaded, na ); setInnerHTML( tab.downloaded, na ); setInnerHTML( tab.availability, na ); setInnerHTML( tab.ratio, na ); setInnerHTML( tab.have, na ); setInnerHTML( tab.upload_to, na ); setInnerHTML( tab.download_from, na ); setInnerHTML( tab.secure, na ); setInnerHTML( tab.creator_date, na ); setInnerHTML( tab.progress, na ); setInnerHTML( tab.comment, na ); setInnerHTML( tab.creator, na ); setInnerHTML( tab.download_dir, na ); setInnerHTML( tab.error, na ); this.updateVisibleFileLists(); this.updatePeersLists(); this.updateTrackersLists(); $("#torrent_inspector_size, .inspector_row > div:contains('N/A')").css('color', '#666'); return; } name = torrents.length == 1 ? torrents[0].name() : torrents.length+' Transfers Selected'; if( torrents.length == 1 ) { var t = torrents[0]; var err = t.getErrorMessage( ); if( err ) error = err; if( t._comment) comment = t._comment ; if( t._creator ) creator = t._creator ; if( t._download_dir) download_dir = t._download_dir; hash = t.hash(); pieces = t._pieceCount + ' pieces @ ' + Transmission.fmt.mem(t._pieceSize); date_created = Transmission.fmt.timestamp( t._creator_date ); } for( var i=0, t; t=torrents[i]; ++i ) { sizeWhenDone += t._sizeWhenDone; sizeDone += t._sizeWhenDone - t._leftUntilDone; total_completed += t.completed(); total_verified += t._verified; total_size += t.size(); total_upload += t.uploadTotal(); total_download += t.downloadTotal(); total_upload_speed += t.uploadSpeed(); total_download_speed += t.downloadSpeed(); total_upload_peers += t.peersGettingFromUs(); total_download_peers += t.peersSendingToUs(); total_availability += t._sizeWhenDone - t._leftUntilDone + t._desiredAvailable; if( total_state == null ) total_state = t.stateStr(); else if ( total_state.search ( t.stateStr() ) == -1 ) total_state += '/' + t.stateStr(); if( t._is_private ) have_private = true; else have_public = true; } var private_string = ''; if( have_private && have_public ) private_string = 'Mixed'; else if( have_private ) private_string = 'Private Torrent'; else if( have_public ) private_string = 'Public Torrent'; setInnerHTML( tab.name, name ); setInnerHTML( tab.size, torrents.length ? Transmission.fmt.size( total_size ) : na ); setInnerHTML( tab.pieces, pieces ); setInnerHTML( tab.hash, hash ); setInnerHTML( tab.state, total_state ); setInnerHTML( tab.download_speed, torrents.length ? Transmission.fmt.speed( total_download_speed ) : na ); setInnerHTML( tab.upload_speed, torrents.length ? Transmission.fmt.speed( total_upload_speed ) : na ); setInnerHTML( tab.uploaded, torrents.length ? Transmission.fmt.size( total_upload ) : na ); setInnerHTML( tab.downloaded, torrents.length ? Transmission.fmt.size( total_download ) : na ); setInnerHTML( tab.availability, torrents.length ? Transmission.fmt.percentString(Math.ratio( total_availability*100, sizeWhenDone )) + '%' : na ); setInnerHTML( tab.ratio, torrents.length ? Transmission.fmt.ratioString(Math.ratio( total_upload, total_download )) : na ); setInnerHTML( tab.have, torrents.length ? Transmission.fmt.size(total_completed) + ' (' + Transmission.fmt.size(total_verified) + ' verified)' : na ); setInnerHTML( tab.upload_to, torrents.length ? total_upload_peers : na ); setInnerHTML( tab.download_from, torrents.length ? total_download_peers : na ); setInnerHTML( tab.secure, private_string ); setInnerHTML( tab.creator_date, date_created ); setInnerHTML( tab.progress, torrents.length ? Transmission.fmt.percentString(Math.ratio( sizeDone*100, sizeWhenDone )) + '%' : na ); setInnerHTML( tab.comment, comment == na ? comment : comment.replace(/\//g, '/​') ); setInnerHTML( tab.creator, creator ); setInnerHTML( tab.download_dir, download_dir == na ? download_dir : download_dir.replace(/([\/_\.])/g, "$1​") ); setInnerHTML( tab.error, error ); this.updatePeersLists(); this.updateTrackersLists(); $(".inspector_row > div:contains('N/A')").css('color', '#666'); this.updateVisibleFileLists(); }, fileListIsVisible: function() { return this._inspector_tab_files.className.indexOf('selected') != -1; }, updateVisibleFileLists: function() { if( this.fileListIsVisible( ) === true ) { var selected = this.getSelectedTorrents(); jQuery.each( selected, function() { this.showFileList(); } ); jQuery.each( this.getDeselectedTorrents(), function() { this.hideFileList(); } ); // Check if we need to display the select all buttions if ( !selected.length ) { if ( $("#select_all_button_container").is(':visible') ) $("#select_all_button_container").hide(); } else { if ( !$("#select_all_button_container").is(':visible') ) $("#select_all_button_container").show(); } } }, updatePeersLists: function() { var tr = this; var html = ''; var torrents = this.getSelectedTorrents( ); if( $(this._inspector_peers_list).is(':visible') ) { for( var k=0, torrent; torrent=torrents[k]; ++k ) { html += '
'; if( torrents.length > 1 ) { html += '
'; html += torrent._name + '
'; } if( torrent._peers.length == 0 ) { html += '
'; // firefox won't paint the top border if the div is empty continue; } html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; for( var i=0, peer; peer=torrent._peers[i]; ++i ) { var parity = ((i+1) % 2 == 0 ? 'even' : 'odd'); html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; html += ''; } html += '
UpDown%StatusAddressClient
' + (peer.isEncrypted ? 'Encrypted' : '') + '' + ( peer.rateToPeer ? Transmission.fmt.speed(peer.rateToPeer) : '' ) + '' + ( peer.rateToClient ? Transmission.fmt.speed(peer.rateToClient) : '' ) + '' + Math.floor(peer.progress*100) + '%' + '' + peer.flagStr + '' + peer.address + '' + peer.clientName + '
'; } } setInnerHTML(this._inspector_peers_list, html); }, updateTrackersLists: function() { // By building up the HTML as as string, then have the browser // turn this into a DOM tree, this is a fast operation. var tr = this; var html = ''; var na = 'N/A'; var torrents = this.getSelectedTorrents( ); if( $(this._inspector_trackers_list).is(':visible') ) { for( var k=0, torrent; torrent = torrents[k]; ++k ) { html += '
'; if( torrents.length > 1 ) { html += '
'; html += torrent._name + '
'; } for( var i=0, tier; tier=torrent._trackerStats[i]; ++i ) { html += '
'; html += 'Tier ' + (i + 1) + '
'; html += ''; } html += '
'; } } setInnerHTML(this._inspector_trackers_list, html); }, lastAnnounceStatus: function(tracker){ var lastAnnounceLabel = 'Last Announce'; var lastAnnounce = 'N/A'; if (tracker.hasAnnounced) { var lastAnnounceTime = Transmission.fmt.timestamp(tracker.lastAnnounceTime); if (tracker.lastAnnounceSucceeded) { lastAnnounce = lastAnnounceTime; lastAnnounce += ' (got ' + tracker.lastAnnouncePeerCount + ' peer'; if (tracker.lastAnnouncePeerCount != 1){ lastAnnounce += 's'; } lastAnnounce += ')'; } else { lastAnnounceLabel = 'Announce error'; lastAnnounce = (tracker.lastAnnounceResult ? tracker.lastAnnounceResult + ' - ' : '') + lastAnnounceTime; } } return {'label':lastAnnounceLabel, 'value':lastAnnounce}; }, announceState: function(tracker){ var announceState = ''; switch (tracker.announceState) { case Torrent._TrackerActive: announceState = 'Announce in progress'; break; case Torrent._TrackerWaiting: var timeUntilAnnounce = tracker.nextAnnounceTime - ((new Date()).getTime() / 1000); if(timeUntilAnnounce < 0){ timeUntilAnnounce = 0; } announceState = 'Next announce in ' + Transmission.fmt.timeInterval(timeUntilAnnounce); break; case Torrent._TrackerQueued: announceState = 'Announce is queued'; break; case Torrent._TrackerInactive: announceState = tracker.isBackup ? 'Tracker will be used as a backup' : 'Announce not scheduled'; break; default: announceState = 'unknown announce state: ' + tracker.announceState; } return announceState; }, lastScrapeStatus: function(tracker){ var lastScrapeLabel = 'Last Scrape'; var lastScrape = 'N/A'; if (tracker.hasScraped) { var lastScrapeTime = Transmission.fmt.timestamp(tracker.lastScrapeTime); if (tracker.lastScrapeSucceeded) { lastScrape = lastScrapeTime; } else { lastScrapeLabel = 'Scrape error'; lastScrape = (tracker.lastScrapeResult ? tracker.lastScrapeResult + ' - ' : '') + lastScrapeTime; } } return {'label':lastScrapeLabel, 'value':lastScrape} }, /* * Toggle the visibility of the inspector (used by the context menu) */ toggleInspector: function() { if( this[Prefs._ShowInspector] ) this.hideInspector( ); else this.showInspector( ); }, showInspector: function() { $('#torrent_inspector').show(); if (iPhone) { $('body').addClass('inspector_showing'); $('#inspector_close').show(); this.hideiPhoneAddressbar(); } else { var w = $('#torrent_inspector').width() + 1 + 'px'; $('#torrent_filter_bar')[0].style.right = w; $('#torrent_container')[0].style.right = w; } setInnerHTML( $('ul li#context_toggle_inspector')[0], 'Hide Inspector' ); this.setPref( Prefs._ShowInspector, true ); this.updateInspector( ); this.refreshDisplay( ); }, /* * Hide the inspector */ hideInspector: function() { $('#torrent_inspector').hide(); if (iPhone) { this.deselectAll( ); $('body.inspector_showing').removeClass('inspector_showing'); $('#inspector_close').hide(); this.hideiPhoneAddressbar(); } else { $('#torrent_filter_bar')[0].style.right = '0px'; $('#torrent_container')[0].style.right = '0px'; setInnerHTML( $('ul li#context_toggle_inspector')[0], 'Show Inspector' ); } this.setPref( Prefs._ShowInspector, false ); this.refreshDisplay( ); }, /* * Toggle the visibility of the filter bar */ toggleFilter: function() { if( this[Prefs._ShowFilter] ) this.hideFilter(); else this.showFilter(); }, showFilter: function( ) { var container_top = parseInt($('#torrent_container').position().top) + $('#torrent_filter_bar').height() + 1; $('#torrent_container').css('top', container_top + 'px'); $('#torrent_filter_bar').show(); this.setPref( Prefs._ShowFilter, true ); }, hideFilter: function() { var container_top = parseInt($('#torrent_container').css('top')) - $('#torrent_filter_bar').height() - 1; $('#torrent_container').css('top', container_top + 'px'); $('#torrent_filter_bar').hide(); this.setPref( Prefs._ShowFilter, false ); this.setFilter( Prefs._FilterAll ); }, refreshMetaData: function(ids) { var tr = this; this.remote.getMetaDataFor(ids, function(active, removed){ tr.updateMetaData(active); }); }, updateMetaData: function( torrents ) { var tr = this; var refresh_files_for = [ ]; jQuery.each( torrents, function( ) { var t = tr._torrents[ this.id ]; if( t ) { t.refreshMetaData( this ); if( t.isSelected( ) ) refresh_files_for.push( t.id( ) ); } } ); if( refresh_files_for.length > 0 ) tr.remote.loadTorrentFiles( refresh_files_for ); }, refreshTorrents: function(ids) { var tr = this; if (!ids) ids = 'recently-active'; this.remote.getUpdatedDataFor(ids, function(active, removed){ tr.updateTorrentsData(active, removed); }); }, updateTorrentsData: function( updated, removed_ids ) { var tr = this; var new_torrent_ids = []; var refresh_files_for = []; jQuery.each( updated, function() { var t = tr._torrents[this.id]; if (t){ t.refresh(this); if(t.isSelected()) refresh_files_for.push(t.id()); } else new_torrent_ids.push(this.id); } ); if(refresh_files_for.length > 0) tr.remote.loadTorrentFiles( refresh_files_for ); if(new_torrent_ids.length > 0) tr.remote.getInitialDataFor(new_torrent_ids, function(torrents){ tr.addTorrents(torrents) } ); var removedAny = tr.deleteTorrents(removed_ids); if( ( new_torrent_ids.length != 0 ) || removedAny ) { tr.hideiPhoneAddressbar(); tr.deselectAll( true ); } this.refilter(); }, updateTorrentsFileData: function( torrents ){ var tr = this; var listIsVisible = tr.fileListIsVisible( ); jQuery.each( torrents, function() { var t = tr._torrents[this.id]; if (t) { t.refreshFileModel(this); if( listIsVisible && t.isSelected()) t.refreshFileView(); } } ); }, initializeAllTorrents: function(){ var tr = this; this.remote.getInitialDataFor( null ,function(torrents) { tr.addTorrents(torrents); } ); }, addTorrents: function( new_torrents ) { var transferFragment = document.createDocumentFragment( ); var fileFragment = document.createDocumentFragment( ); for( var i=0, row; row=new_torrents[i]; ++i ) { var new_torrent = new Torrent( transferFragment, fileFragment, this, row ); this._torrents[new_torrent.id()] = new_torrent; } this._inspector_file_list.appendChild( fileFragment ); this._torrent_list.appendChild( transferFragment ); this.refilter( ); }, deleteTorrents: function(torrent_ids){ if(typeof torrent_ids == 'undefined') return false; var tr = this; var removedAny = false; $.each( torrent_ids, function(index, id){ var torrent = tr._torrents[id]; if(torrent) { removedAny = true; var e = torrent.element(); if( e ) { var row_index; for( var i=0, row; row = tr._rows[i]; ++i ) { if( row._id == torrent._id ) { row_index = i; e = tr._rows[row_index]; break; } } delete e._torrent; //remove circular refernce to help IE garbage collect tr._rows.splice(row_index, 1) e.remove(); } torrent.hideFileList(); torrent.deleteFiles(); delete tr._torrents[torrent.id()]; } }); return removedAny; }, refreshDisplay: function( ) { var torrents = this.getVisibleTorrents(); for( var i=0; torrents[i]; ++i ) torrents[i].refreshHTML(); }, /* * Set the alternating background colors for torrents */ setTorrentBgColors: function( ) { var rows = this.getVisibleRows( ); for( var i=0, row; row=rows[i]; ++i ) { var wasEven = row[0].className.indexOf('even') != -1; var isEven = ((i+1) % 2 == 0); if( wasEven != isEven ) row.toggleClass('even', isEven); } }, updateStatusbar: function() { var torrents = this.getAllTorrents(); var torrentCount = torrents.length; var visibleCount = this.getVisibleTorrents().length; // calculate the overall speed var upSpeed = 0; var downSpeed = 0; for( var i=0, row; row=torrents[i]; ++i ) { upSpeed += row.uploadSpeed( ); downSpeed += row.downloadSpeed( ); } // update torrent count label var s; if( torrentCount == visibleCount ) s = torrentCount + ' Transfers'; else s = visibleCount + ' of ' + torrentCount + ' Transfers'; setInnerHTML( $('#torrent_global_transfer')[0], s ); // update the speeds s = Transmission.fmt.speed( upSpeed ); if( iPhone ) s = 'UL: ' + s; setInnerHTML( $('#torrent_global_upload')[0], s ); // download speeds s = Transmission.fmt.speed( downSpeed ); if( iPhone ) s = 'DL: ' + s; setInnerHTML( $('#torrent_global_download')[0], s ); }, /* * Select a torrent file to upload * FIXME */ uploadTorrentFile: function(confirmed) { // Display the upload dialog if (! confirmed) { $('input#torrent_upload_file').attr('value', ''); $('input#torrent_upload_url').attr('value', ''); $('input#torrent_auto_start').attr('checked', $('#prefs_form #auto_start')[0].checked); $('#upload_container').show(); if (!iPhone && Safari3) { setTimeout("$('div#upload_container div.dialog_window').css('top', '0px');",10); } // Submit the upload form } else { var tr = this; var args = { }; var paused = !$('#torrent_auto_start').is(':checked'); if ('' != $('#torrent_upload_url').val()) { tr.remote.addTorrentByUrl($('#torrent_upload_url').val(), { paused: paused }); } else { args.url = '/transmission/upload?paused=' + paused; args.type = 'POST'; args.data = { 'X-Transmission-Session-Id' : tr.remote._token }; args.dataType = 'xml'; args.iframe = true; args.success = function( data ) { tr.refreshTorrents(); tr.togglePeriodicRefresh( true ); }; tr.togglePeriodicRefresh( false ); $('#torrent_upload_form').ajaxSubmit( args ); } } }, removeSelectedTorrents: function() { var torrents = this.getSelectedTorrents( ); if( torrents.length ) this.promptToRemoveTorrents( torrents ); }, removeSelectedTorrentsAndData: function() { var torrents = this.getSelectedTorrents( ); if( torrents.length ) this.promptToRemoveTorrentsAndData( torrents ); }, promptToRemoveTorrents:function( torrents ) { if( torrents.length == 1 ) { var torrent = torrents[0]; var header = 'Remove ' + torrent.name() + '?'; var message = 'Once removed, continuing the transfer will require the torrent file. Are you sure you want to remove it?'; askUser( header, message, 'Remove', transmission.removeTorrents, torrents ); } else { var header = 'Remove ' + torrents.length + ' transfers?'; var message = 'Once removed, continuing the transfers will require the torrent files. Are you sure you want to remove them?'; askUser( header, message, 'Remove', transmission.removeTorrents, torrents ); } }, removeTorrents: function( torrents ) { console.log( "in removeTorrents" ); console.log( torrents ); var torrent_ids = jQuery.map(torrents, function(t) { return t.id(); } ); var tr = this; //this.remote.removeTorrents( torrent_ids, function(){ tr.refreshTorrents() } ); }, promptToRemoveTorrentsAndData:function( torrents ) { if( torrents.length == 1 ) { var torrent = torrents[0], header = 'Remove ' + torrent.name() + ' and delete data?', message = 'All data downloaded for this torrent will be deleted. Are you sure you want to remove it?'; askUser( header, message, 'Remove', transmission.removeTorrentsAndData, torrents ); } else { var header = 'Remove ' + torrents.length + ' transfers and delete data?', message = 'All data downloaded for these torrents will be deleted. Are you sure you want to remove them?'; askUser( header, message, 'Remove', transmission.removeTorrentsAndData, torrents ); } }, removeTorrentsAndData: function( torrents ) { this.remote.removeTorrentsAndData( torrents ); }, verifySelectedTorrents: function() { this.verifyTorrents( this.getSelectedTorrents( ) ); }, startSelectedTorrents: function( ) { this.startTorrents( this.getSelectedTorrents( ) ); }, startAllTorrents: function( ) { this.startTorrents( this.getAllTorrents( ) ); }, startTorrent: function( torrent ) { this.startTorrents( [ torrent ] ); }, startTorrents: function( torrents ) { var torrent_ids = jQuery.map(torrents, function(t) { return t.id(); } ); var tr = this; this.remote.startTorrents( torrent_ids, function(){ tr.refreshTorrents(torrent_ids) } ); }, verifyTorrent: function( torrent ) { this.verifyTorrents( [ torrent ] ); }, verifyTorrents: function( torrents ) { var torrent_ids = jQuery.map(torrents, function(t) { return t.id(); } ); var tr = this; this.remote.verifyTorrents( torrent_ids, function(){ tr.refreshTorrents(torrent_ids) } ); }, stopSelectedTorrents: function( ) { this.stopTorrents( this.getSelectedTorrents( ) ); }, stopAllTorrents: function( ) { this.stopTorrents( this.getAllTorrents( ) ); }, stopTorrent: function( torrent ) { this.stopTorrents( [ torrent ] ); }, stopTorrents: function( torrents ) { var torrent_ids = jQuery.map(torrents, function(t) { return t.id(); } ); var tr = this; this.remote.stopTorrents( torrent_ids, function(){ tr.refreshTorrents(torrent_ids )} ); }, changeFileCommand: function(command, torrent, file) { this.remote.changeFileCommand(command, torrent, file) }, hideiPhoneAddressbar: function(timeInSeconds) { var tr = this; if( iPhone ) { var delayLength = timeInSeconds ? timeInSeconds*1000 : 150; // not currently supported on iPhone if(/*document.body.scrollTop!=1 && */scroll_timeout==null) { scroll_timeout = setTimeout(function(){ tr.doToolbarHide(); }, delayLength); } } }, doToolbarHide: function() { window.scrollTo(0,1); scroll_timeout=null; }, /*** **** ***/ refilter: function() { // decide which torrents to keep showing var allTorrents = this.getAllTorrents( ); var keep = [ ]; for( var i=0, t; t=allTorrents[i]; ++i ) if( t.test( this[Prefs._FilterMode], this._current_search ) ) keep.push( t ); // sort the keepers Torrent.sortTorrents( keep, this[Prefs._SortMethod], this[Prefs._SortDirection] ); // make a backup of the selection var sel = this.getSelectedTorrents( ); this.deselectAll( ); // hide the ones we're not keeping for( var i=keep.length, e; e=this._rows[i]; ++i ) { delete e._torrent; e[0].style.display = 'none'; } // show the ones we're keeping sel.sort( Torrent.compareById ); for( var i=0, len=keep.length; i