deluge/src/interface.py

1345 lines
51 KiB
Python
Raw Normal View History

2006-12-08 19:06:49 +00:00
#!/usr/bin/env python
2006-11-28 22:28:37 +00:00
#
2007-03-26 01:58:58 +00:00
# interface.py
#
# Copyright (C) Zach Tibbitts 2006 <zach@collegegeek.org>
2007-01-08 19:38:19 +00:00
#
# 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; either version 2, or (at your option)
# any later version.
#
2007-01-08 19:38:19 +00:00
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
2007-01-08 19:38:19 +00:00
# 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
2007-01-08 19:38:19 +00:00
# along with this program. If not, write to:
# The Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor
# Boston, MA 02110-1301, USA.
2007-06-12 16:33:32 +00:00
#
# In addition, as a special exception, the copyright holders give
# permission to link the code of portions of this program with the OpenSSL
# library.
# You must obey the GNU General Public License in all respects for all of
# the code used other than OpenSSL. If you modify file(s) with this
# exception, you may extend this exception to your version of the file(s),
# but you are not obligated to do so. If you do not wish to do so, delete
# this exception statement from your version. If you delete this exception
# statement from all source files in the program, then also delete it here.
2006-11-28 22:28:37 +00:00
2007-03-05 22:30:54 +00:00
import sys, os, os.path, urllib
2007-03-26 21:09:39 +00:00
import core, common, dgtk, ipc_manager, dialogs
import plugins, pref
import pygtk
2006-11-28 22:28:37 +00:00
pygtk.require('2.0')
2007-01-11 21:53:25 +00:00
import gtk, gtk.glade, gobject
2007-01-11 00:58:23 +00:00
import xdg, xdg.BaseDirectory
2007-03-05 22:30:54 +00:00
import gettext, locale
2007-02-21 01:58:35 +00:00
class DelugeGTK:
def __init__(self):
2007-03-05 22:30:54 +00:00
APP = 'deluge'
2007-03-26 21:09:39 +00:00
DIR = os.path.join(common.INSTALL_PREFIX, 'share', 'locale')
2007-03-06 01:40:46 +00:00
locale.setlocale(locale.LC_ALL, '')
2007-03-05 23:38:14 +00:00
locale.bindtextdomain(APP, DIR)
locale.textdomain(APP)
2007-03-05 22:30:54 +00:00
gettext.bindtextdomain(APP, DIR)
gettext.textdomain(APP)
gettext.install(APP, DIR)
2007-01-12 20:07:55 +00:00
self.is_running = False
2007-06-16 05:45:22 +00:00
self.update_queue = []
2007-02-22 20:51:22 +00:00
self.ipc_manager = ipc_manager.Manager(self)
2007-01-12 20:07:55 +00:00
self.torrent_file_queue = []
2007-02-15 21:54:55 +00:00
#Start the Deluge Manager:
2007-03-26 21:13:14 +00:00
self.manager = core.Manager(common.CLIENT_CODE, common.CLIENT_VERSION,
'%s %s'%(common.PROGRAM_NAME, common.PROGRAM_VERSION), common.CONFIG_DIR)
2007-03-22 18:04:15 +00:00
self.something_screwed_up = False
2007-03-26 04:59:48 +00:00
self.plugins = plugins.PluginManager(self.manager, self)
2007-03-26 21:09:39 +00:00
self.plugins.add_plugin_dir(common.PLUGIN_DIR)
if os.path.isdir(os.path.join(common.CONFIG_DIR , 'plugins')):
self.plugins.add_plugin_dir(os.path.join(common.CONFIG_DIR, 'plugins'))
2007-02-15 21:54:55 +00:00
self.plugins.scan_for_plugins()
2007-06-08 00:15:51 +00:00
self.config = self.manager.get_config()
2007-01-11 00:58:23 +00:00
#Set up the interface:
2007-03-26 21:09:39 +00:00
self.wtree = gtk.glade.XML(common.get_glade_file("delugegtk.glade"), domain=APP)
2006-11-28 22:28:37 +00:00
self.window = self.wtree.get_widget("main_window")
2006-12-22 23:24:42 +00:00
self.toolbar = self.wtree.get_widget("tb_middle")
2007-06-11 22:11:03 +00:00
self.window.drag_dest_set(gtk.DEST_DEFAULT_ALL,[('text/uri-list', 0, 80)], gtk.gdk.ACTION_COPY)
2007-02-16 18:38:14 +00:00
self.window.connect("delete_event", self.close)
2007-02-16 02:03:16 +00:00
self.window.connect("drag_data_received", self.on_drag_data)
self.window.connect("window-state-event", self.window_state_event)
self.window.connect("configure-event", self.window_configure_event)
2007-03-26 21:09:39 +00:00
self.window.set_title(common.PROGRAM_NAME)
self.window.set_icon_from_file(common.get_pixmap("deluge32.png"))
2007-03-01 21:02:40 +00:00
self.notebook = self.wtree.get_widget("torrent_info")
2007-02-21 02:43:09 +00:00
self.statusbar = self.wtree.get_widget("statusbar")
## Construct the Interface
2007-03-01 19:30:06 +00:00
try:
self.build_tray_icon()
except AttributeError:
#python-pygtk is < 2.9
self.tray_icon = dgtk.StupidTray()
self.has_tray = False
else:
self.has_tray = True
2007-03-26 01:03:00 +00:00
self.preferences_dialog = dialogs.PreferencesDlg(self, self.config)
self.plugin_dialog = dialogs.PluginDlg(self, self.plugins)
self.build_torrent_table()
self.build_summary_tab()
self.build_file_tab()
self.build_peer_tab()
# Set the Torrent menu bar sub-menu to the same as the right-click Torrent pop-up menu
self.wtree.get_widget("menu_torrent").set_submenu(self.torrent_menu)
2006-12-08 19:02:22 +00:00
self.connect_signals()
2006-12-08 19:02:22 +00:00
2007-02-15 22:41:53 +00:00
try:
self.load_window_settings()
except KeyError:
pass
2007-02-16 04:30:49 +00:00
self.apply_prefs()
2007-02-27 23:49:29 +00:00
self.load_window_geometry()
2007-06-16 05:45:22 +00:00
# Load plugins after GTK is initialised
self.update_queue.append(self.load_plugins)
2007-02-22 20:51:22 +00:00
def external_add_torrent(self, torrent_file):
print "Ding!"
print "Got torrent externally:", os.path.basename(torrent_file)
print "Here's the raw data:", torrent_file
print "\tNow, what to do with it?"
if self.is_running:
print "\t\tthe client seems to already be running, i'll try and add the torrent"
uid = self.interactive_add_torrent(torrent_file)
else:
print "\t\tthe client hasn't started yet, I'll queue the torrent"
self.torrent_file_queue.append(torrent_file)
def connect_signals(self):
2007-01-12 20:07:55 +00:00
self.wtree.signal_autoconnect({
2006-11-28 22:28:37 +00:00
## File Menu
2007-01-12 20:07:55 +00:00
"add_torrent": self.add_torrent_clicked,
2007-02-12 18:14:15 +00:00
"add_torrent_url": self.add_torrent_url_clicked,
2007-02-16 18:38:14 +00:00
"menu_quit": self.quit,
## Edit Menu
"pref_clicked": self.show_pref_dialog,
"plugins_clicked": self.show_plugin_dialog,
2007-01-11 21:53:25 +00:00
## View Menu
2007-02-21 05:29:47 +00:00
"toolbar_toggle": self.toolbar_toggle,
2007-01-12 20:40:56 +00:00
"infopane_toggle": self.infopane_toggle,
2007-01-11 21:53:25 +00:00
"size_toggle": self.size_toggle,
"status_toggle": self.status_toggle,
"seeders_toggle": self.seeders_toggle,
"peers_toggle": self.peers_toggle,
"dl_toggle": self.dl_toggle,
"ul_toggle": self.ul_toggle,
"eta_toggle": self.eta_toggle,
"share_toggle": self.share_toggle,
2006-11-28 22:28:37 +00:00
## Help Menu
"show_about_dialog": self.show_about_dialog,
2007-01-12 20:07:55 +00:00
## Toolbar
2007-02-27 00:47:22 +00:00
"start_pause": self.start_pause,
2007-01-12 20:07:55 +00:00
"update_tracker": self.update_tracker,
"clear_finished": self.clear_finished,
"queue_up": self.q_torrent_up,
"queue_down": self.q_torrent_down,
2007-02-21 05:38:02 +00:00
"queue_bottom": self.q_to_bottom,
2007-06-16 05:45:22 +00:00
"queue_top": self.q_to_top,
2007-01-12 20:07:55 +00:00
})
2007-02-14 18:08:33 +00:00
def build_tray_icon(self):
2007-03-26 21:09:39 +00:00
self.tray_icon = gtk.status_icon_new_from_file(common.get_pixmap("deluge32.png"))
2007-06-20 19:08:49 +00:00
self.tray_glade = gtk.glade.XML(common.get_glade_file("tray_menu.glade"), domain='deluge')
self.tray_menu = self.tray_glade.get_widget("tray_menu")
self.tray_glade.signal_autoconnect({
"quit": self.quit,
"plugins": self.show_plugin_dialog,
"preferences": self.show_pref_dialog,
"add_torrent": self.add_torrent_clicked,
"clear_finished": self.clear_finished,
"show_hide_window": self.force_show_hide,
})
self.tray_glade.get_widget("download-limit-image").set_from_file(common.get_pixmap('downloading22.png'))
self.tray_glade.get_widget("upload-limit-image").set_from_file(common.get_pixmap('seeding22.png'))
self.build_tray_bwsetsubmenu()
2007-02-27 20:47:46 +00:00
self.tray_icon.connect("activate", self.tray_clicked)
self.tray_icon.connect("popup-menu", self.tray_popup)
def tray_popup(self, status_icon, button, activate_time):
self.tray_menu.popup(None, None, gtk.status_icon_position_menu,
button, activate_time, status_icon)
def build_tray_bwsetsubmenu(self):
self.submenu_bwdownset = gtk.Menu()
self.submenu_bwupset = gtk.Menu()
group = None
for value in sorted(self.config.get("tray_downloadspeedlist")):
subitem = gtk.RadioMenuItem(group, str(value) + " " + _("KiB/s"))
group = subitem
self.submenu_bwdownset.append(subitem)
if value == self.config.get("max_download_rate"):
subitem.set_active(True)
subitem.connect("toggled", self.tray_setbwdown)
subitem = gtk.RadioMenuItem(group, _("Unlimited"))
self.submenu_bwdownset.append(subitem)
if self.config.get("max_download_rate") < 0:
subitem.set_active(True)
subitem.connect("toggled", self.tray_setbwdown)
subitem = gtk.SeparatorMenuItem()
self.submenu_bwdownset.append(subitem)
subitem = gtk.MenuItem(_("Other..."))
subitem.connect("activate", self.tray_setbwdown)
self.submenu_bwdownset.append(subitem)
group = None
for value in sorted(self.config.get("tray_uploadspeedlist")):
subitem = gtk.RadioMenuItem(group, str(value) + " " + _("KiB/s"))
group = subitem
if value == self.config.get("max_upload_rate"):
subitem.set_active(True)
self.submenu_bwupset.append(subitem)
subitem.connect("toggled", self.tray_setbwup)
subitem = gtk.RadioMenuItem(group, _("Unlimited"))
self.submenu_bwupset.append(subitem)
if self.config.get("max_upload_rate") < 0:
subitem.set_active(True)
subitem.connect("toggled", self.tray_setbwup)
subitem = gtk.SeparatorMenuItem()
self.submenu_bwupset.append(subitem)
subitem = gtk.MenuItem(_("Other..."))
subitem.connect("activate", self.tray_setbwup)
self.submenu_bwupset.append(subitem)
self.tray_glade.get_widget("download_limit").set_submenu(self.submenu_bwdownset)
self.tray_glade.get_widget("upload_limit").set_submenu(self.submenu_bwupset)
2007-06-20 19:08:49 +00:00
self.submenu_bwdownset.show_all()
self.submenu_bwupset.show_all()
def tray_setbwdown(self, widget, data=None):
str_bwdown = widget.get_children()[0].get_text().rstrip(" "+_("KiB/s"))
if str_bwdown == _("Unlimited"):
str_bwdown = -1
if str_bwdown == _("Other..."):
dialog_glade = gtk.glade.XML(common.get_glade_file("dgtkpopups.glade"))
rate_dialog = dialog_glade.get_widget("rate_dialog")
spin_rate = dialog_glade.get_widget("spin_rate")
spin_rate.set_value(self.config.get("max_download_rate"))
spin_rate.select_region(0, -1)
response = rate_dialog.run()
if response == 1: # OK Response
str_bwdown = spin_rate.get_value()
else:
rate_dialog.destroy()
return
rate_dialog.destroy()
self.config.set("max_download_rate", float(str_bwdown))
self.apply_prefs()
def tray_setbwup(self, widget, data=None):
str_bwup = widget.get_children()[0].get_text().rstrip(" "+_("KiB/s"))
if str_bwup == _("Unlimited"):
str_bwup = -1
if str_bwup == _("Other..."):
dialog_glade = gtk.glade.XML(common.get_glade_file("dgtkpopups.glade"))
rate_dialog = dialog_glade.get_widget("rate_dialog")
spin_rate = dialog_glade.get_widget("spin_rate")
spin_rate.set_value(self.config.get("max_upload_rate"))
spin_rate.select_region(0, -1)
response = rate_dialog.run()
if response == 1: # OK Response
str_bwup = spin_rate.get_value()
else:
rate_dialog.destroy()
return
rate_dialog.destroy()
self.config.set("max_upload_rate", float(str_bwup))
self.apply_prefs()
2007-05-25 08:17:22 +00:00
def unlock_tray(self,comingnext):
entered_pass = gtk.Entry(25)
entered_pass.set_activates_default(True)
entered_pass.set_width_chars(25)
entered_pass.set_visibility(False)
entered_pass.show()
tray_lock = gtk.Dialog(title=_("Deluge is locked"), parent=self.window,
buttons=(gtk.STOCK_CANCEL, gtk.RESPONSE_REJECT, gtk.STOCK_OK, gtk.RESPONSE_ACCEPT))
label = gtk.Label(_("Deluge is password protected.\nTo show the Deluge window, please enter your password"))
label.set_line_wrap(True)
label.set_justify(gtk.JUSTIFY_CENTER)
tray_lock.set_position(gtk.WIN_POS_CENTER_ALWAYS)
tray_lock.set_size_request(400, 200)
tray_lock.set_default_response(gtk.RESPONSE_ACCEPT)
tray_lock.vbox.pack_start(label)
tray_lock.vbox.pack_start(entered_pass)
tray_lock.show_all()
if tray_lock.run() == gtk.RESPONSE_ACCEPT:
if self.config.get("tray_passwd", default="") == entered_pass.get_text():
if comingnext == "mainwinshow":
self.window.show()
elif comingnext == "prefwinshow":
self.preferences_dialog.show()
self.apply_prefs()
self.config.save_to_file()
elif comingnext == "quitus":
self.window.hide()
self.shutdown()
tray_lock.destroy()
return True
2007-02-27 20:47:46 +00:00
def tray_clicked(self, status_icon):
2007-02-14 18:08:33 +00:00
if self.window.get_property("visible"):
if self.window.is_active():
self.window.hide()
else:
self.window.present()
else:
2007-05-25 08:17:22 +00:00
if self.config.get("lock_tray", bool, default=False) == True:
self.unlock_tray("mainwinshow")
else:
self.load_window_geometry()
2007-05-25 08:17:22 +00:00
self.window.show()
2007-02-14 18:08:33 +00:00
def force_show_hide(self, arg=None):
if self.window.get_property("visible"):
self.window.hide()
else:
2007-05-25 08:17:22 +00:00
if self.config.get("lock_tray", bool, default=False) == True:
self.unlock_tray("mainwinshow")
else:
self.window.show()
def build_torrent_table(self):
## Create the torrent listview
2007-02-21 01:18:37 +00:00
self.torrent_view = self.wtree.get_widget("torrent_view")
2007-03-26 21:09:39 +00:00
self.torrent_glade = gtk.glade.XML(common.get_glade_file("torrent_menu.glade"), domain='deluge')
2007-02-27 00:28:18 +00:00
self.torrent_menu = self.torrent_glade.get_widget("torrent_menu")
self.torrent_glade.signal_autoconnect({ "remove_torrent": self.remove_torrent_clicked,
"start_pause": self.start_pause,
2007-03-26 05:33:16 +00:00
"update_tracker": self.update_tracker,
"clear_finished": self.clear_finished,
"queue_up": self.q_torrent_up,
"queue_down": self.q_torrent_down,
"queue_bottom": self.q_to_bottom,
2007-06-16 05:45:22 +00:00
"queue_top": self.q_to_top,
"set_ratio0": self.set_ratio_0,
"set_ratio1": self.set_ratio_1,
"set_ratio2": self.set_ratio_2,
"set_ratio3": self.set_ratio_3,
2007-03-26 05:33:16 +00:00
})
self.torrent_menu.connect("focus", self.torrent_menu_focus)
# UID, Q#, Status Icon, Name, Size, Progress, Message, Seeders, Peers, DL, UL, ETA, Share
self.torrent_model = gtk.ListStore(int, int, gtk.gdk.Pixbuf, str, str, float, str, int, int, int, int, int, int, str, float)
2007-02-21 01:18:37 +00:00
self.torrent_view.set_model(self.torrent_model)
self.torrent_view.set_rules_hint(True)
self.torrent_view.set_reorderable(True)
2007-06-10 20:00:16 +00:00
self.torrent_view.get_selection().set_mode(gtk.SELECTION_MULTIPLE)
self.torrent_selected = None
2007-02-28 01:51:52 +00:00
def size(column, cell, model, iter, data):
size = long(model.get_value(iter, data))
2007-03-26 21:09:39 +00:00
size_str = common.fsize(size)
2007-02-28 01:51:52 +00:00
cell.set_property('text', size_str)
def rate(column, cell, model, iter, data):
rate = int(model.get_value(iter, data))
2007-03-26 21:09:39 +00:00
rate_str = common.frate(rate)
2007-02-28 01:51:52 +00:00
cell.set_property('text', rate_str)
def peer(column, cell, model, iter, data):
c1, c2 = data
a = int(model.get_value(iter, c1))
b = int(model.get_value(iter, c2))
cell.set_property('text', '%d (%d)'%(a, b))
def time(column, cell, model, iter, data):
time = int(model.get_value(iter, data))
if time < 0 or time == 0:
time_str = _("Done")
else:
2007-03-26 21:09:39 +00:00
time_str = common.ftime(time)
2007-02-28 01:51:52 +00:00
cell.set_property('text', time_str)
2007-03-06 21:17:41 +00:00
def ratio(column, cell, model, iter, data):
ratio = float(model.get_value(iter, data))
if ratio == -1:
ratio_str = _("Unknown")
else:
ratio_str = "%.2f"%ratio
cell.set_property('text', ratio_str)
2006-12-08 19:02:22 +00:00
2006-12-24 04:08:34 +00:00
## Initializes the columns for the torrent_view
(TORRENT_VIEW_COL_UID, TORRENT_VIEW_COL_QUEUE, TORRENT_VIEW_COL_STATUSICON, TORRENT_VIEW_COL_NAME,
TORRENT_VIEW_COL_SIZE, TORRENT_VIEW_COL_PROGRESS, TORRENT_VIEW_COL_STATUS,
TORRENT_VIEW_COL_CONNECTED_SEEDS, TORRENT_VIEW_COL_SEEDS,
TORRENT_VIEW_COL_CONNECTED_PEERS, TORRENT_VIEW_COL_PEERS, TORRENT_VIEW_COL_DOWNLOAD,
TORRENT_VIEW_COL_UPLOAD, TORRENT_VIEW_COL_ETA, TORRENT_VIEW_COL_RATIO) = range(15)
self.queue_column = dgtk.add_text_column(self.torrent_view, "#", TORRENT_VIEW_COL_QUEUE)
2007-06-19 05:41:03 +00:00
self.name_column = dgtk.add_texticon_column(self.torrent_view, _("Name"), TORRENT_VIEW_COL_STATUSICON, TORRENT_VIEW_COL_NAME)
self.size_column = dgtk.add_func_column(self.torrent_view, _("Size"), size, TORRENT_VIEW_COL_SIZE)
self.status_column = dgtk.add_progress_column(self.torrent_view, _("Status"), TORRENT_VIEW_COL_PROGRESS, TORRENT_VIEW_COL_STATUS)
self.seed_column = dgtk.add_func_column(self.torrent_view, _("Seeders"), peer, (TORRENT_VIEW_COL_CONNECTED_SEEDS, TORRENT_VIEW_COL_SEEDS))
self.peer_column = dgtk.add_func_column(self.torrent_view, _("Peers"), peer, (TORRENT_VIEW_COL_CONNECTED_PEERS, TORRENT_VIEW_COL_PEERS))
self.dl_column = dgtk.add_func_column(self.torrent_view, _("Download"), rate, TORRENT_VIEW_COL_DOWNLOAD)
self.ul_column = dgtk.add_func_column(self.torrent_view, _("Upload"), rate, TORRENT_VIEW_COL_UPLOAD)
self.eta_column = dgtk.add_func_column(self.torrent_view, _("ETA"), time, TORRENT_VIEW_COL_ETA)
self.share_column = dgtk.add_func_column(self.torrent_view, _("Ratio"), ratio, TORRENT_VIEW_COL_RATIO)
2006-11-28 22:28:37 +00:00
2006-12-24 04:08:34 +00:00
self.status_column.set_expand(True)
2007-06-19 05:41:03 +00:00
self.name_column.set_sort_column_id(TORRENT_VIEW_COL_NAME)
self.seed_column.set_sort_column_id(TORRENT_VIEW_COL_CONNECTED_SEEDS)
self.peer_column.set_sort_column_id(TORRENT_VIEW_COL_CONNECTED_PEERS)
2007-02-17 18:39:59 +00:00
2007-03-01 22:04:22 +00:00
def long_sort(model, iter1, iter2, data):
value1 = long(model.get_value(iter1, data))
value2 = long(model.get_value(iter2, data))
if value1 < value2:
return -1
elif value1 > value2:
return 1
else:
return 0
self.torrent_model.set_sort_func(TORRENT_VIEW_COL_QUEUE, long_sort, TORRENT_VIEW_COL_QUEUE)
self.torrent_model.set_sort_func(TORRENT_VIEW_COL_SIZE, long_sort, TORRENT_VIEW_COL_SIZE)
self.torrent_model.set_sort_func(TORRENT_VIEW_COL_ETA, long_sort, TORRENT_VIEW_COL_ETA)
self.torrent_model.set_sort_column_id(TORRENT_VIEW_COL_QUEUE, gtk.SORT_ASCENDING)
2007-03-02 19:14:50 +00:00
try:
self.torrent_view.get_selection().set_select_function(self.torrent_clicked, full=True)
except TypeError:
self.torrent_view.get_selection().set_select_function(self.old_t_click)
2007-02-27 00:28:18 +00:00
self.torrent_view.connect("button-press-event", self.torrent_view_clicked)
2007-06-10 20:00:16 +00:00
self.right_click = False
2007-03-02 19:14:50 +00:00
def old_t_click(self, path):
return self.torrent_clicked(self.torrent_view.get_selection(), self.torrent_model, path, False)
2007-02-27 00:28:18 +00:00
2007-02-17 18:39:59 +00:00
def torrent_clicked(self, selection, model, path, is_selected):
if is_selected:
# Torrent is already selected, we don't need to do anything
2007-06-10 20:00:16 +00:00
return not self.right_click
2007-02-17 18:39:59 +00:00
unique_id = model.get_value(model.get_iter(path), 0)
state = self.manager.get_torrent_state(unique_id)
# A new torrent has been selected, need to update parts of interface
2007-03-26 21:09:39 +00:00
self.text_summary_total_size.set_text(common.fsize(state["total_size"]))
2007-06-08 07:16:39 +00:00
self.text_summary_pieces.set_text(str(state["num_pieces"]))
2007-02-17 18:39:59 +00:00
self.text_summary_tracker.set_text(str(state["tracker"]))
#self.text_summary_compact_allocation.set_text(str(state[""]))
# Now for the File tab
self.file_store.clear()
all_files = self.manager.get_torrent_file_info(unique_id)
2007-02-21 03:26:48 +00:00
file_filter = self.manager.get_file_filter(unique_id)
if file_filter is None:
2007-02-22 21:47:21 +00:00
file_filter = [False] * len(all_files)
2007-02-21 03:26:48 +00:00
assert(len(all_files) == len(file_filter))
2007-05-25 07:31:31 +00:00
i=0
for f in all_files:
2007-06-12 02:27:29 +00:00
self.file_store.append([not file_filter[i], f['path'], common.fsize(f['size']),
2007-06-13 03:58:27 +00:00
f['offset'], round(f['progress'],2)])
i=i+1
2007-05-25 07:31:31 +00:00
return True
2007-02-27 00:28:18 +00:00
def torrent_view_clicked(self, widget, event):
if event.button == 3:
x = int(event.x)
y = int(event.y)
data = self.torrent_view.get_path_at_pos(x, y)
if data is None:
return True
path, col, cellx, celly = data
2007-06-10 20:00:16 +00:00
self.right_click = self.torrent_view.get_selection().path_is_selected(path)
2007-02-27 00:28:18 +00:00
self.torrent_view.grab_focus()
self.torrent_view.set_cursor(path, col, 0)
unique_id = self.torrent_model.get_value(self.torrent_model.get_iter(path), 0)
# Get the torrent state so we can check if the torrent is paused.
torrent_state = self.manager.get_torrent_state(unique_id)
2007-03-26 03:57:40 +00:00
widget = self.torrent_glade.get_widget("menu_pause")
if torrent_state["is_paused"]:
2007-03-26 03:57:40 +00:00
widget.set_image(gtk.image_new_from_stock(gtk.STOCK_MEDIA_PLAY, gtk.ICON_SIZE_MENU))
widget.get_children()[0].set_text(_("Resume"))
2007-03-26 03:57:40 +00:00
else:
widget.set_image(gtk.image_new_from_stock(gtk.STOCK_MEDIA_PAUSE, gtk.ICON_SIZE_MENU))
widget.get_children()[0].set_text(_("Pause"))
2007-02-27 00:28:18 +00:00
self.torrent_menu.popup(None, None, None, event.button, event.time)
return True
else:
2007-06-10 20:00:16 +00:00
self.right_click = False
2007-02-27 00:28:18 +00:00
return False
2007-02-27 00:47:22 +00:00
def start_pause(self, widget):
2007-06-10 20:00:16 +00:00
unique_ids = self.get_selected_torrent_rows()
try:
2007-06-10 20:00:16 +00:00
for uid in unique_ids:
self.manager.set_user_pause(uid, not self.manager.is_user_paused(uid))
# We need to force an update so the GUI looks more responsive
self.update()
except KeyError:
pass
def set_ratio_0(self, widget):
unique_ids = self.get_selected_torrent_rows()
for uid in unique_ids:
self.manager.set_ratio(uid, 0)
def set_ratio_1(self, widget):
unique_ids = self.get_selected_torrent_rows()
for uid in unique_ids:
self.manager.set_ratio(uid, 1)
def set_ratio_2(self, widget):
unique_ids = self.get_selected_torrent_rows()
for uid in unique_ids:
self.manager.set_ratio(uid, 2)
def set_ratio_3(self, widget):
unique_ids = self.get_selected_torrent_rows()
for uid in unique_ids:
self.manager.set_ratio(uid, 3)
def torrent_menu_focus(self, widget, direction):
menuitem = self.torrent_glade.get_widget("menu_pause")
# Check if we are selecting multiple torrents
if len(self.get_selected_torrent_rows()) > 1:
menuitem.set_image(gtk.image_new_from_stock(gtk.STOCK_MEDIA_PAUSE, gtk.ICON_SIZE_MENU))
menuitem.get_children()[0].set_text(_("Pause/Resume"))
return
# Get the selected torrent state so we can check if the torrent is paused.
unique_id = self.get_selected_torrent()
torrent_state = self.manager.get_torrent_state(unique_id)
if torrent_state["is_paused"]:
menuitem.set_image(gtk.image_new_from_stock(gtk.STOCK_MEDIA_PLAY, gtk.ICON_SIZE_MENU))
menuitem.get_children()[0].set_text(_("Resume"))
else:
menuitem.set_image(gtk.image_new_from_stock(gtk.STOCK_MEDIA_PAUSE, gtk.ICON_SIZE_MENU))
menuitem.get_children()[0].set_text(_("Pause"))
def build_summary_tab(self):
2007-01-11 23:15:22 +00:00
#Torrent Summary tab
2007-01-12 20:07:55 +00:00
# Look into glade's widget prefix function
2007-01-11 23:15:22 +00:00
self.text_summary_title = self.wtree.get_widget("summary_title")
self.text_summary_total_size = self.wtree.get_widget("summary_total_size")
self.text_summary_pieces = self.wtree.get_widget("summary_pieces")
self.text_summary_total_downloaded = self.wtree.get_widget("summary_total_downloaded")
self.text_summary_total_uploaded = self.wtree.get_widget("summary_total_uploaded")
2007-06-09 01:15:00 +00:00
self.text_summary_download_rate = self.wtree.get_widget("summary_download_rate")
self.text_summary_upload_rate = self.wtree.get_widget("summary_upload_rate")
self.text_summary_seeders = self.wtree.get_widget("summary_seeders")
self.text_summary_peers = self.wtree.get_widget("summary_peers")
2007-01-11 23:15:22 +00:00
self.text_summary_percentage_done = self.wtree.get_widget("summary_percentage_done")
self.text_summary_share_ratio = self.wtree.get_widget("summary_share_ratio")
self.text_summary_downloaded_this_session = self.wtree.get_widget("summary_downloaded_this_session")
2007-02-22 21:15:56 +00:00
self.text_summary_uploaded_this_session = self.wtree.get_widget("summary_uploaded_this_session")
2007-01-11 23:15:22 +00:00
self.text_summary_tracker = self.wtree.get_widget("summary_tracker")
self.text_summary_tracker_response = self.wtree.get_widget("summary_tracker_response")
2007-02-06 21:21:09 +00:00
self.text_summary_tracker_status = self.wtree.get_widget("summary_tracker_status")
2007-01-11 23:15:22 +00:00
self.text_summary_next_announce = self.wtree.get_widget("summary_next_announce")
self.text_summary_compact_allocation = self.wtree.get_widget("summary_compact_allocation")
2007-06-09 01:15:00 +00:00
self.text_summary_eta = self.wtree.get_widget("summary_eta")
def build_peer_tab(self):
2007-06-13 03:58:27 +00:00
self.peer_view = self.wtree.get_widget("peer_view")
self.peer_store = gtk.ListStore(str, str, float, str, str)
def percent(column, cell, model, iter, data):
percent = float(model.get_value(iter, data))
percent_str = "%.2f%%"%percent
cell.set_property("text", percent_str)
self.peer_view.set_model(self.peer_store)
self.peer_ip_column = dgtk.add_text_column(self.peer_view, _("IP Address"), 0)
2007-02-16 02:21:36 +00:00
self.peer_client_column = dgtk.add_text_column(self.peer_view, _("Client"), 1)
self.peer_complete_column = dgtk.add_func_column(self.peer_view, _("Percent Complete"), percent, 2)
2007-02-16 02:21:36 +00:00
self.peer_download_column = dgtk.add_text_column(self.peer_view, _("Download Rate"), 3)
self.peer_upload_column = dgtk.add_text_column(self.peer_view, _("Upload Rate"), 4)
def build_file_tab(self):
2007-06-13 03:58:27 +00:00
def percent(column, cell, model, iter, data):
percent = float(model.get_value(iter, data))
percent_str = "%.2f%%"%percent
cell.set_property("text", percent_str)
self.file_view = self.wtree.get_widget("file_view")
self.file_glade = gtk.glade.XML(common.get_glade_file("file_tab_menu.glade"), domain='deluge')
self.file_menu = self.file_glade.get_widget("file_tab_menu")
self.file_glade.signal_autoconnect({ "select_all": self.file_select_all,
"unselect_all": self.file_unselect_all,
"check_selected": self.file_check_selected,
"uncheck_selected": self.file_uncheck_selected,
})
2007-06-13 03:58:27 +00:00
self.file_store = gtk.ListStore(bool, str, str, str, float)
self.file_view.set_model(self.file_store)
self.file_view.get_selection().set_mode(gtk.SELECTION_MULTIPLE)
self.file_view.get_selection().set_select_function(self.file_clicked)
self.file_selected = []
self.file_view.connect("button-press-event", self.file_view_clicked)
2007-01-11 23:15:22 +00:00
2007-02-21 03:26:48 +00:00
dgtk.add_toggle_column(self.file_view, _("Download"), 0, toggled_signal=self.file_toggled)
dgtk.add_text_column(self.file_view, _("Filename"), 1).set_expand(True)
dgtk.add_text_column(self.file_view, _("Size"), 2)
dgtk.add_text_column(self.file_view, _("Offset"), 3)
2007-06-13 03:58:27 +00:00
dgtk.add_func_column(self.file_view, _("Progress"), percent, 4)
def file_select_all(self, widget):
self.file_view.get_selection().select_all()
def file_unselect_all(self, widget):
self.file_view.get_selection().unselect_all()
def file_check_selected(self, widget):
self.file_view.get_selection().selected_foreach(self.file_toggle_selected, True)
def file_uncheck_selected(self, widget):
self.file_view.get_selection().selected_foreach(self.file_toggle_selected, False)
2007-02-21 03:26:48 +00:00
def file_clicked(self, path):
return not self.file_selected
def file_view_clicked(self, widget, event):
if event.button == 3:
self.file_menu.popup(None, None, None, event.button, event.time)
return True
else:
self.file_selected = False
return False
def file_toggle_selected(self, treemodel, path, selected_iter, value):
self.file_store.set_value(selected_iter, 0, value)
2007-02-19 21:19:23 +00:00
def file_toggled(self, renderer, path):
self.file_selected = True
2007-02-19 21:19:23 +00:00
file_iter = self.file_store.get_iter_from_string(path)
value = not renderer.get_active()
selection = self.file_view.get_selection()
if selection.iter_is_selected(file_iter):
selection.selected_foreach(self.file_toggle_selected, value)
else:
self.file_store.set_value(file_iter, 0, value)
2007-02-19 21:19:23 +00:00
file_filter = []
itr = self.file_store.get_iter_first()
while itr is not None:
2007-02-22 21:15:56 +00:00
file_filter.append(not self.file_store.get_value(itr, 0))
2007-02-19 21:19:23 +00:00
itr = self.file_store.iter_next(itr)
self.manager.set_file_filter(self.get_selected_torrent(), file_filter)
def file_get_iter_from_name(self, name):
iter = self.file_store.get_iter_first()
while iter:
if self.file_store.get_value(iter, 1) == name:
return iter
iter = self.file_store.iter_next(iter)
return None
def show_about_dialog(self, arg=None):
2007-03-26 01:03:00 +00:00
dialogs.show_about_dialog()
def show_pref_dialog(self, arg=None):
if self.window.get_property("visible"):
2007-05-25 08:17:22 +00:00
self.preferences_dialog.show()
self.apply_prefs()
2007-06-08 00:15:51 +00:00
self.config.save()
2007-05-25 08:17:22 +00:00
else:
if self.config.get("lock_tray", bool, default=False) == True:
self.unlock_tray("prefwinshow")
else:
self.preferences_dialog.show()
self.apply_prefs()
self.config.save()
2007-05-25 07:31:31 +00:00
def show_plugin_dialog(self, arg=None):
2007-03-26 01:03:00 +00:00
self.plugin_dialog.show()
2007-02-14 22:43:48 +00:00
def apply_prefs(self):
# Show tray icon if necessary
self.tray_icon.set_visible(self.config.get("enable_system_tray"))
# Update the max_*_rate_bps prefs
ulrate = self.config.get("max_upload_rate") * 1024
dlrate = self.config.get("max_download_rate") * 1024
if not (ulrate < 0):
self.config.set("max_upload_rate_bps", ulrate)
if not (dlrate < 0):
self.config.set("max_download_rate_bps", dlrate)
# Update the tray download speed limits
if self.config.get("max_download_rate") not in self.config.get("tray_downloadspeedlist") and self.config.get("max_download_rate") >= 0:
# We need to prepend this value and remove the last value in the list
self.config.get("tray_downloadspeedlist").insert(0, self.config.get("max_download_rate"))
self.config.get("tray_downloadspeedlist").pop()
# Re-build the sub-menu to display new option
self.build_tray_bwsetsubmenu()
# Do the same for the upload speed limits
if self.config.get("max_upload_rate") not in self.config.get("tray_uploadspeedlist") and self.config.get("max_upload_rate") >= 0:
# We need to prepend this value and remove the last value in the list
self.config.get("tray_uploadspeedlist").insert(0, self.config.get("max_upload_rate"))
self.config.get("tray_uploadspeedlist").pop()
# Re-build the sub-menu to display new option
self.build_tray_bwsetsubmenu()
# Apply the preferences in the core
self.manager.apply_prefs()
2007-06-16 10:56:09 +00:00
self.manager.pe_settings(self.config.get("encout_state", int, default=common.EncState.enabled), self.config.get("encin_state", int, default=common.EncState.enabled), self.config.get("enclevel_type", int, default=common.EncLevel.both), self.config.get("pref_rc4", bool, default=True))
2007-06-16 12:26:43 +00:00
self.manager.proxy_settings(self.config.get("proxy_hostname"), self.config.get("proxy_username"), self.config.get("proxy_password"), int(self.config.get("proxy_port")), self.config.get("proxy_type", int, default=common.ProxyType.none))
2007-03-06 20:57:49 +00:00
def get_message_from_state(self, torrent_state):
state = torrent_state['state']
is_paused = torrent_state['is_paused']
progress = torrent_state['progress']
progress = '%d%%'%int(progress * 100)
if is_paused:
message = 'Paused %s'%progress
else:
2007-03-08 06:49:53 +00:00
try:
2007-03-26 01:58:58 +00:00
message = core.STATE_MESSAGES[state]
2007-03-08 06:49:53 +00:00
if state in (1, 3, 4, 7):
message = '%s %s'%(message, progress)
except IndexError:
message = ''
2007-03-06 20:57:49 +00:00
return message
2007-02-14 22:43:48 +00:00
# UID, Q#, Name, Size, Progress, Message, Seeders, Peers, DL, UL, ETA, Share
def get_list_from_unique_id(self, unique_id):
2007-05-25 07:31:31 +00:00
state = self.manager.get_torrent_state(unique_id)
queue = int(state['queue_pos']) + 1
2007-05-25 07:31:31 +00:00
name = state['name']
2007-02-28 01:51:52 +00:00
size = long(state['total_size'])
2007-02-14 22:43:48 +00:00
progress = float(state['progress'] * 100)
2007-03-06 20:57:49 +00:00
message = self.get_message_from_state(state)
2007-02-28 01:51:52 +00:00
seeds = int(state['num_seeds'])
seeds_t = int(state['total_seeds'])
peers = int(state['num_peers'])
peers_t = int(state['total_peers'])
dlrate = int(state['download_rate'])
ulrate = int(state['upload_rate'])
try:
eta = common.get_eta(state["total_size"], state["total_done"], state["download_rate"])
2007-02-28 01:51:52 +00:00
except ZeroDivisionError:
eta = 0
2007-02-28 01:51:52 +00:00
share = float(self.calc_share_ratio(unique_id, state))
2007-06-19 05:41:03 +00:00
# Set the appropriate status icon
if state["is_paused"]:
status_icon = gtk.gdk.pixbuf_new_from_file(common.get_pixmap("inactive22.png"))
2007-06-19 05:41:03 +00:00
elif state["is_seed"]:
status_icon = gtk.gdk.pixbuf_new_from_file(common.get_pixmap("seeding22.png"))
2007-06-19 05:41:03 +00:00
else:
status_icon = gtk.gdk.pixbuf_new_from_file(common.get_pixmap("downloading22.png"))
2007-06-19 05:41:03 +00:00
rlist = [int(unique_id), int(queue), status_icon, str(name), long(size), float(progress), str(message),
2007-02-28 01:51:52 +00:00
int(seeds), int(seeds_t), int(peers), int(peers_t), int(dlrate), int(ulrate), int(eta), float(share)]
return rlist
2007-02-14 22:43:48 +00:00
2007-01-11 00:58:23 +00:00
## Start the timer that updates the interface
def start(self):
if not (self.window.flags() & gtk.VISIBLE):
print "Showing window"
2007-01-12 20:07:55 +00:00
self.window.show()
# go through torrent files to add
#dummy preferences values:
use_default_download_location = True
default_download_location = "."
for torrent_file in self.torrent_file_queue:
print "adding torrent", torrent_file
try:
self.interactive_add_torrent(torrent_file, append=False)
2007-03-26 01:58:58 +00:00
except core.DelugeError:
2007-01-12 20:07:55 +00:00
print "duplicate torrent found, ignoring", torrent_file
## add torrents in manager to interface
2007-02-28 01:51:52 +00:00
# self.torrent_model.append([0, 1, "Hello, World", 2048, 50.0, "Hi", 1, 2, 1, 2, 2048, 2048, 120, 1.0])
2007-01-12 20:07:55 +00:00
for uid in self.manager.get_unique_IDs():
2007-02-21 01:18:37 +00:00
self.torrent_model.append(self.get_list_from_unique_id(uid))
2007-01-11 00:58:23 +00:00
gobject.timeout_add(1000, self.update)
2007-01-12 20:07:55 +00:00
try:
self.is_running = True
gtk.main()
except KeyboardInterrupt:
self.manager.quit()
2007-06-16 05:45:22 +00:00
def load_plugins(self):
enable_plugins = self.config.get('enabled_plugins', str, default="").split(':')
for plugin in enable_plugins:
try:
self.plugins.enable_plugin(plugin)
except KeyError:
pass
2007-01-11 00:58:23 +00:00
## Call via a timer to update the interface
def update(self):
2007-06-16 05:45:22 +00:00
self.update_queue.reverse()
while len(self.update_queue) > 0:
f = self.update_queue.pop()
f()
# We need to apply the queue changes
self.manager.apply_queue()
2007-01-11 21:53:25 +00:00
# Make sure that the interface still exists
try:
2007-01-11 05:01:11 +00:00
tab = self.wtree.get_widget("torrent_info").get_current_page()
2007-01-11 21:53:25 +00:00
except AttributeError:
return False
2007-02-21 02:43:09 +00:00
if self.something_screwed_up:
2007-03-26 01:03:00 +00:00
dialogs.show_popup_warning(self.window,
_("For some reason, the previous state could not be loaded, so a blank state has been loaded for you."))
2007-03-26 01:03:00 +00:00
restore_torrents = dialogs.show_popup_question(self.window,
_("Would you like to attempt to reload the previous session's downloads?"))
if restore_torrents:
2007-03-26 01:58:58 +00:00
torrent_subdir = os.path.join(self.manager.base_dir, core.TORRENTS_SUBDIR)
for torrent in os.listdir(torrent_subdir):
if torrent.endswith('.torrent'):
2007-03-08 22:49:03 +00:00
self.interactive_add_torrent(torrent)
self.something_screwed_up = False
2007-02-21 02:43:09 +00:00
# Update Statusbar and Tray Tips
core_state = self.manager.get_state()
connections = core_state['num_peers']
max_connections = int(self.config.get("max_connections"))
2007-03-26 21:09:39 +00:00
dlrate = common.frate(core_state['download_rate'])
ulrate = common.frate(core_state['upload_rate'])
if self.config.get("max_download_rate") < 0:
dlrate_max = _("Unlimited")
else:
dlrate_max = common.frate(self.config.get("max_download_rate_bps"))
if self.config.get("max_upload_rate") < 0:
ulrate_max = _("Unlimited")
else:
ulrate_max = common.frate(self.config.get("max_upload_rate_bps"))
self.statusbar_temp_msg = '%s: %s (%s) %s: %s (%s) %s: %s (%s)'%(
_('Connections'), connections, max_connections, _('Download'),
dlrate, dlrate_max, _('Upload'), ulrate, ulrate_max)
2007-02-21 02:43:09 +00:00
if 'DHT_nodes' in core_state.keys():
2007-05-25 07:31:31 +00:00
dht_peers = core_state['DHT_nodes']
if dht_peers == -1:
dht_peers = '?'
else:
dht_peers = str(dht_peers)
self.statusbar_temp_msg = self.statusbar_temp_msg + ' [DHT: %s]'%(dht_peers)
msg = _("Deluge Bittorrent Client") + "\n" + \
_("Connections") + ": " + str(connections) + " (" + str(max_connections) + ")" + "\n" + _("Download") + ": " + \
dlrate + " (" + dlrate_max + ")" + "\n" + _("Upload") + ": " + ulrate + " (" + ulrate_max + ")"
2007-05-25 07:10:22 +00:00
self.tray_icon.set_tooltip(msg)
2007-05-25 07:31:31 +00:00
2007-02-21 02:43:09 +00:00
#Update any active plugins
2007-02-14 22:43:48 +00:00
self.plugins.update_active_plugins()
2007-02-21 02:43:09 +00:00
# Put the generated message into the statusbar
# This gives plugins a chance to write to the
# statusbar if they want
self.statusbar.pop(1)
self.statusbar.push(1, self.statusbar_temp_msg)
2007-02-14 22:43:48 +00:00
2007-01-11 23:15:22 +00:00
# If no torrent is selected, select the first torrent:
2007-06-10 20:00:16 +00:00
if self.torrent_selected is None:
2007-02-21 01:18:37 +00:00
self.torrent_view.get_selection().select_path("0")
2007-01-12 20:40:56 +00:00
#Torrent List
2007-02-21 01:18:37 +00:00
itr = self.torrent_model.get_iter_first()
2007-01-12 20:40:56 +00:00
if itr is None:
return True
2007-02-14 22:43:48 +00:00
2007-01-12 20:40:56 +00:00
while itr is not None:
2007-02-21 01:18:37 +00:00
uid = self.torrent_model.get_value(itr, 0)
2007-01-12 20:40:56 +00:00
try:
state = self.manager.get_torrent_state(uid)
tlist = self.get_list_from_unique_id(uid)
2007-02-28 01:51:52 +00:00
for i in range(len(tlist)):
try:
self.torrent_model.set_value(itr, i, tlist[i])
except:
2007-03-01 19:12:06 +00:00
print "ERR", i, type(tlist[i]), tlist[i]
2007-02-21 01:18:37 +00:00
itr = self.torrent_model.iter_next(itr)
2007-03-26 01:58:58 +00:00
except core.InvalidUniqueIDError:
2007-02-21 01:18:37 +00:00
self.torrent_model.remove(itr)
if not self.torrent_model.iter_is_valid(itr):
2007-01-12 20:40:56 +00:00
itr = None
2007-02-27 19:38:55 +00:00
try:
2007-06-19 03:13:29 +00:00
if self.manager.get_torrent_state(self.get_selected_torrent())["is_paused"]:
2007-02-27 19:38:55 +00:00
self.wtree.get_widget("toolbutton_pause").set_stock_id(gtk.STOCK_MEDIA_PLAY)
self.wtree.get_widget("toolbutton_pause").set_label(_("Resume"))
2007-02-27 19:38:55 +00:00
else:
self.wtree.get_widget("toolbutton_pause").set_stock_id(gtk.STOCK_MEDIA_PAUSE)
self.wtree.get_widget("toolbutton_pause").set_label(_("Pause"))
2007-02-27 19:38:55 +00:00
except KeyError:
2007-03-02 06:40:14 +00:00
pass
2007-02-12 20:41:06 +00:00
try:
state = self.manager.get_torrent_state(self.get_selected_torrent())
2007-03-26 01:58:58 +00:00
except core.InvalidUniqueIDError:
return True
2007-02-14 22:43:48 +00:00
if tab == 0: #Details Pane
2007-03-23 00:27:21 +00:00
self.wtree.get_widget("summary_name").set_text(state['name'])
2007-03-26 21:09:39 +00:00
self.text_summary_total_size.set_text(common.fsize(state["total_size"]))
2007-06-08 07:16:39 +00:00
self.text_summary_pieces.set_text(str(state["num_pieces"]))
self.text_summary_total_downloaded.set_text(common.fsize(state["total_done"]) + " (" + common.fsize(state["total_download"]) + ")")
self.text_summary_total_uploaded.set_text(common.fsize(self.manager.unique_IDs[self.get_selected_torrent()].uploaded_memory + state["total_payload_upload"]) + " (" + common.fsize(state["total_upload"]) + ")")
2007-03-26 21:09:39 +00:00
self.text_summary_download_rate.set_text(common.frate(state["download_rate"]))
self.text_summary_upload_rate.set_text(common.frate(state["upload_rate"]))
self.text_summary_seeders.set_text(common.fseed(state))
self.text_summary_peers.set_text(common.fpeer(state))
2007-02-12 23:12:08 +00:00
self.wtree.get_widget("progressbar").set_fraction(float(state['progress']))
2007-03-26 21:09:39 +00:00
self.wtree.get_widget("progressbar").set_text(common.fpcnt(state["progress"]))
self.text_summary_share_ratio.set_text('%.3f'%(self.calc_share_ratio(self.get_selected_torrent(), state)))
2007-01-11 23:15:22 +00:00
self.text_summary_tracker.set_text(str(state["tracker"]))
2007-02-06 21:21:09 +00:00
self.text_summary_tracker_status.set_text(str(state["tracker_ok"]))
2007-01-11 23:15:22 +00:00
self.text_summary_next_announce.set_text(str(state["next_announce"]))
2007-03-26 21:09:39 +00:00
self.text_summary_eta.set_text(common.estimate_eta(state))
2007-01-12 20:40:56 +00:00
elif tab == 1: #Peers List
2007-02-19 20:39:23 +00:00
def biographer(model, path, iter, dictionary):
assert(model.get_value(iter, 0) not in dictionary.keys())
dictionary[model.get_value(iter, 0)] = model.get_string_from_iter(iter)
class remover_data:
def __init__(self, new_ips):
self.new_ips = new_ips
self.removed = False
def remover(model, path, iter, data):
if model.get_value(iter, 0) not in data.new_ips:
model.remove(iter)
data.removed = True
return True
else:
return False
2007-02-12 20:41:06 +00:00
unique_id = self.get_selected_torrent()
2007-03-02 06:40:14 +00:00
new_peer_info = self.manager.get_torrent_peer_info(unique_id)
2007-02-12 20:41:06 +00:00
new_ips = {}
for index in range(len(new_peer_info)):
if not new_peer_info[index]['client'] == "":
assert(new_peer_info[index]['ip'] not in new_ips.keys())
new_ips[new_peer_info[index]['ip']] = index
2007-02-19 20:39:23 +00:00
while True:
data = remover_data(new_ips.keys())
self.peer_store.foreach(remover, data)
if not data.removed:
break
curr_ips = {}
self.peer_store.foreach(biographer, curr_ips)
assert(self.peer_store.iter_n_children(None) == len(curr_ips.keys()))
for peer in new_peer_info:
if peer['ip'] in curr_ips.keys():
self.peer_store.set(self.peer_store.get_iter_from_string(curr_ips[peer['ip']]),
1, unicode(peer['client'], 'Latin-1'),
2, round(peer["peer_has"],2),
2007-03-26 21:09:39 +00:00
3, common.frate(peer["download_speed"]),
4, common.frate(peer["upload_speed"]))
2007-02-19 20:39:23 +00:00
if peer['ip'] not in curr_ips.keys() and peer['client'] is not "":
self.peer_store.append([peer["ip"],
unicode(peer["client"], 'Latin-1'),
round(peer["peer_has"],2),
common.frate(peer["download_speed"]),
common.frate(peer["upload_speed"])])
2007-03-02 06:40:14 +00:00
del new_peer_info
del new_ips
del curr_ips
2007-02-12 20:41:06 +00:00
2007-06-13 03:58:27 +00:00
elif tab == 2: #file tab
unique_id = self.get_selected_torrent()
new_file_info = self.manager.get_torrent_file_info(unique_id)
for file in new_file_info:
iter = self.file_get_iter_from_name(file['path'])
if (iter != None) and (round(self.file_store.get_value(iter, 4),2) != round(file['progress'],2)):
self.file_store.set_value(iter, 4, round(file['progress'],2))
2007-06-13 03:58:27 +00:00
return True
2007-01-11 21:53:25 +00:00
else:
pass
return True
2007-01-11 00:58:23 +00:00
def calc_share_ratio(self, unique_id, torrent_state):
2007-02-28 01:51:52 +00:00
r = float(self.manager.calc_ratio(unique_id, torrent_state))
return r
2007-01-11 23:15:22 +00:00
2007-06-10 20:00:16 +00:00
# Return the id of the last single selected torrent
2007-01-11 05:01:11 +00:00
def get_selected_torrent(self):
try:
2007-06-10 20:00:16 +00:00
if self.torrent_view.get_selection().count_selected_rows() == 1:
self.torrent_selected = self.torrent_view.get_selection().get_selected_rows()[1][0]
selected_torrent = self.torrent_model.get_value(self.torrent_model.get_iter(self.torrent_selected), 0)
return selected_torrent
except TypeError, ValueError:
2007-06-10 20:00:16 +00:00
return None
# Return a list of ids of the selected torrents
def get_selected_torrent_rows(self):
selected_ids = []
selected_paths = self.torrent_view.get_selection().get_selected_rows()[1]
try:
for path in selected_paths:
selected_ids.append(self.torrent_model.get_value(self.torrent_model.get_iter(path), 0))
return selected_ids
except ValueError:
return None
2007-02-16 02:03:16 +00:00
def on_drag_data(self, widget, drag_context, x, y, selection_data, info, timestamp):
2007-06-11 22:21:02 +00:00
uri_split = selection_data.data.strip().split()
for uri in uri_split:
path = urllib.url2pathname(uri).strip('\r\n\x00')
if path.startswith('file:\\\\\\'):
path = path[8:]
elif path.startswith('file://'):
path = path[7:]
elif path.startswith('file:'):
path = path[5:]
if path.endswith('.torrent'):
self.interactive_add_torrent(path)
2006-12-22 23:24:42 +00:00
def interactive_add_torrent(self, torrent, append=True):
2007-02-21 21:21:42 +00:00
if self.config.get('use_default_dir', bool, default=False):
path = self.config.get('default_download_path', default=os.path.expandvars('$HOME'))
else:
2007-03-26 01:03:00 +00:00
path = dialogs.show_directory_chooser_dialog(self.window)
if path is None:
return
2007-03-23 01:28:53 +00:00
try:
unique_id = self.manager.add_torrent(torrent, path, self.config.get('use_compact_storage', bool, default=False))
if append:
self.torrent_model.append(self.get_list_from_unique_id(unique_id))
except core.InvalidEncodingError, e:
print "InvalidEncodingError", e
dialogs.show_popup_warning(self.window, _("An error occured while trying to add the torrent. It's possible your .torrent file is corrupted."))
2007-03-27 23:19:37 +00:00
except core.DuplicateTorrentError, e:
dialogs.show_popup_warning(self.window, _("The torrent you've added seems to already be in Deluge."))
except core.InsufficientFreeSpaceError, e:
nice_need = common.fsize(e.needed_space)
nice_free = common.fsize(e.free_space)
2007-05-25 07:31:31 +00:00
dialogs.show_popup_warning(self.window, _("There is not enough free disk space to complete your download.") + "\n" + \
_("Space Needed:") + " " + nice_need + "\n" + \
_("Available Space:") + " " + nice_free)
2007-01-12 20:07:55 +00:00
def add_torrent_clicked(self, obj=None):
2007-03-26 01:03:00 +00:00
torrent = dialogs.show_file_open_dialog()
2007-01-11 00:58:23 +00:00
if torrent is not None:
self.interactive_add_torrent(torrent)
2007-02-12 18:14:15 +00:00
def add_torrent_url_clicked(self, obj=None):
2007-02-16 02:21:36 +00:00
dlg = gtk.Dialog(title=_("Add torrent from URL"), parent=self.window,
buttons=(gtk.STOCK_CANCEL, 0, gtk.STOCK_OK, 1))
2007-03-26 21:09:39 +00:00
dlg.set_icon_from_file(common.get_pixmap("deluge32.png"))
2007-02-16 02:21:36 +00:00
label = gtk.Label(_("Enter the URL of the .torrent to download"))
entry = gtk.Entry()
dlg.vbox.pack_start(label)
dlg.vbox.pack_start(entry)
clip = gtk.clipboard_get(selection='PRIMARY')
text = clip.wait_for_text()
entry.set_text(text)
dlg.show_all()
result = dlg.run()
url = entry.get_text()
dlg.destroy()
if result == 1:
2007-06-11 22:11:03 +00:00
add_torrent_url(url)
2007-06-09 23:50:10 +00:00
def external_add_url(self, url):
print "Got URL externally:", url
if self.is_running:
print "\t\tthe client seems to already be running, i'll try and add the URL"
self.add_torrent_url(url)
else:
print "\t\tthe client hasn't started yet, I'll queue the URL torrent file"
self.queue_torrent_url(url)
def add_torrent_url(self, url):
filename, headers = self.fetch_url(url)
if filename:
self.interactive_add_torrent(filename)
def queue_torrent_url(self, url):
filename, headers = self.fetch_url(url)
if filename:
self.torrent_file_queue.append(filename)
def fetch_url(self, url):
filename, headers = urllib.urlretrieve(url)
if filename.endswith(".torrent") or headers["content-type"]=="application/x-bittorrent":
return filename, headers
else:
print "URL doesn't appear to be a valid torrent file:", url
return None, None
2007-02-12 18:14:15 +00:00
2007-01-12 20:07:55 +00:00
def remove_torrent_clicked(self, obj=None):
2007-06-10 20:00:16 +00:00
torrent_list = self.get_selected_torrent_rows()
glade = gtk.glade.XML(common.get_glade_file("dgtkpopups.glade"), domain='deluge')
asker = glade.get_widget("remove_torrent_dlg")
asker.set_icon_from_file(common.get_pixmap("deluge32.png"))
warning = glade.get_widget("warning")
warning.set_text(" ")
2007-02-12 18:14:15 +00:00
torrent_also = glade.get_widget("torrent_also")
2007-06-10 20:00:16 +00:00
data_also = glade.get_widget("data_also")
data_also.connect("toggled", self.remove_toggle_warning, warning)
2007-02-12 18:14:15 +00:00
2007-06-10 20:00:16 +00:00
response = asker.run()
asker.destroy()
if response == 1:
for torrent in torrent_list:
if torrent is not None:
if torrent == self.get_selected_torrent():
first = self.torrent_model.get_iter_first()
if first:
self.torrent_view.get_selection().select_path("0")
else:
self.torrent_selected = None
2007-02-12 18:14:15 +00:00
self.manager.remove_torrent(torrent, data_also.get_active(), torrent_also.get_active())
2007-06-10 20:00:16 +00:00
self.clear_details_pane()
def clear_details_pane(self):
self.wtree.get_widget("progressbar").set_text("")
2007-06-01 10:16:10 +00:00
self.wtree.get_widget("summary_name").set_text("")
self.text_summary_total_size.set_text("")
self.text_summary_pieces.set_text("")
self.text_summary_total_downloaded.set_text("")
self.text_summary_total_uploaded.set_text("")
self.text_summary_download_rate.set_text("")
self.text_summary_upload_rate.set_text("")
self.text_summary_seeders.set_text("")
self.text_summary_peers.set_text("")
self.wtree.get_widget("progressbar").set_fraction(0.0)
self.text_summary_share_ratio.set_text("")
self.text_summary_tracker.set_text("")
self.text_summary_tracker_status.set_text("")
self.text_summary_next_announce.set_text("")
self.text_summary_eta.set_text("")
self.peer_store.clear()
self.file_store.clear()
2007-02-12 18:14:15 +00:00
def remove_toggle_warning(self, args, warning):
if not args.get_active():
warning.set_text(" ")
2007-05-25 07:31:31 +00:00
else:
warning.set_markup("<i>" + _("Warning - all downloaded files for this torrent will be deleted!") + "</i>")
return False
2007-01-12 20:07:55 +00:00
def update_tracker(self, obj=None):
torrent = self.get_selected_torrent()
if torrent is not None:
self.manager.update_tracker(torrent)
def clear_finished(self, obj=None):
2007-02-27 20:47:46 +00:00
print "Clearing Completed Torrents"
self.manager.clear_completed()
def q_torrent_up(self, obj=None):
torrent = self.get_selected_torrent()
if torrent is not None:
self.manager.queue_up(torrent)
self.update()
def q_torrent_down(self, obj=None):
torrent = self.get_selected_torrent()
if torrent is not None:
2007-02-23 03:46:01 +00:00
self.manager.queue_down(torrent)
self.update()
2007-02-21 05:38:02 +00:00
def q_to_bottom(self, widget):
torrent = self.get_selected_torrent()
if torrent is not None:
self.manager.queue_bottom(torrent)
self.update()
2007-06-16 05:45:22 +00:00
def q_to_top(self, widget):
torrent = self.get_selected_torrent()
if torrent is not None:
self.manager.queue_top(torrent)
self.update()
2007-02-21 05:29:47 +00:00
def toolbar_toggle(self, widget):
if widget.get_active():
self.wtree.get_widget("tb_left").show()
self.wtree.get_widget("tb_middle").show()
self.wtree.get_widget("tb_right").show()
else:
self.wtree.get_widget("tb_left").hide()
self.wtree.get_widget("tb_middle").hide()
self.wtree.get_widget("tb_right").hide()
2007-01-12 20:40:56 +00:00
def infopane_toggle(self, widget):
if widget.get_active():
self.wtree.get_widget("torrent_info").show()
else:
2007-02-28 06:20:12 +00:00
self.wtree.get_widget("torrent_info").hide()
2007-01-11 21:53:25 +00:00
def size_toggle(self, obj):
self.size_column.set_visible(obj.get_active())
def status_toggle(self, obj):
self.status_column.set_visible(obj.get_active())
def seeders_toggle(self, obj):
self.seed_column.set_visible(obj.get_active())
def peers_toggle(self, obj):
self.peer_column.set_visible(obj.get_active())
def dl_toggle(self, obj):
self.dl_column.set_visible(obj.get_active())
def ul_toggle(self, obj):
self.ul_column.set_visible(obj.get_active())
def eta_toggle(self, obj):
self.eta_column.set_visible(obj.get_active())
def share_toggle(self, obj):
self.share_column.set_visible(obj.get_active())
2007-02-15 22:41:53 +00:00
def load_window_settings(self):
2007-02-21 21:21:42 +00:00
self.wtree.get_widget("chk_infopane").set_active(self.config.get("show_infopane", bool, default=True))
self.wtree.get_widget("chk_toolbar").set_active(self.config.get("show_toolbar", bool, default=True))
self.wtree.get_widget("chk_size").set_active(self.config.get("show_size", bool, default=True))
self.wtree.get_widget("chk_status").set_active(self.config.get("show_status", bool, default=True))
self.wtree.get_widget("chk_seed").set_active(self.config.get("show_seeders", bool, default=True))
self.wtree.get_widget("chk_peer").set_active(self.config.get("show_peers", bool, default=True))
self.wtree.get_widget("chk_download").set_active(self.config.get("show_dl", bool, default=True))
self.wtree.get_widget("chk_upload").set_active(self.config.get("show_ul", bool, default=True))
self.wtree.get_widget("chk_eta").set_active(self.config.get("show_eta", bool, default=True))
self.wtree.get_widget("chk_ratio").set_active(self.config.get("show_share", bool, default=True))
self.wtree.get_widget("vpaned1").set_position(self.config.get("window_height") - self.config.get("window_pane_position"))
2007-02-15 22:41:53 +00:00
def save_window_settings(self):
2007-02-21 01:18:37 +00:00
self.config.set("show_infopane", self.wtree.get_widget("chk_infopane").get_active())
2007-02-21 05:29:47 +00:00
self.config.set("show_toolbar", self.wtree.get_widget("chk_toolbar").get_active())
2007-02-21 01:18:37 +00:00
self.config.set("show_size", self.size_column.get_visible())
self.config.set("show_status", self.status_column.get_visible())
self.config.set("show_seeders", self.seed_column.get_visible())
self.config.set("show_peers", self.peer_column.get_visible())
self.config.set("show_dl", self.dl_column.get_visible())
self.config.set("show_ul", self.ul_column.get_visible())
self.config.set("show_eta", self.eta_column.get_visible())
self.config.set("show_share", self.share_column.get_visible())
2007-06-12 05:06:13 +00:00
self.config.set("window_pane_position", self.config.get("window_height") - self.wtree.get_widget("vpaned1").get_position())
2007-02-27 23:49:29 +00:00
def window_configure_event(self, widget, event):
if self.config.get("window_maximized") == False:
self.config.set("window_x_pos", self.window.get_position()[0])
self.config.set("window_y_pos", self.window.get_position()[1])
self.config.set("window_width", event.width)
self.config.set("window_height", event.height)
def window_state_event(self, widget, event):
if event.changed_mask & gtk.gdk.WINDOW_STATE_MAXIMIZED:
if event.new_window_state & gtk.gdk.WINDOW_STATE_MAXIMIZED:
self.config.set("window_maximized", True)
else:
self.config.set("window_maximized", False)
return False
2007-02-27 23:49:29 +00:00
def load_window_geometry(self):
x = self.config.get('window_x_pos')
y = self.config.get('window_y_pos')
w = self.config.get('window_width')
h = self.config.get('window_height')
2007-02-27 23:49:29 +00:00
self.window.move(x, y)
self.window.resize(w, h)
if self.config.get("window_maximized") == True:
self.window.maximize()
2007-02-16 18:38:14 +00:00
def close(self, widget, event):
2007-03-01 19:30:06 +00:00
if self.config.get("close_to_tray", bool, default=False) and self.config.get("enable_system_tray", bool, default=True) and self.has_tray:
2007-02-16 18:38:14 +00:00
self.window.hide()
return True
else:
self.quit()
def quit(self, widget=None):
if self.window.get_property("visible"):
self.window.hide()
self.shutdown()
else:
if self.config.get("lock_tray", bool, default=False) == True:
self.unlock_tray("quitus")
else:
2007-05-25 08:17:22 +00:00
self.window.hide()
self.shutdown()
2007-02-15 21:30:17 +00:00
def shutdown(self):
2007-02-17 18:39:59 +00:00
enabled_plugins = ':'.join(self.plugins.get_enabled_plugins())
2007-02-21 01:18:37 +00:00
self.config.set('enabled_plugins', enabled_plugins)
2007-02-15 22:41:53 +00:00
self.save_window_settings()
2007-06-08 00:15:51 +00:00
self.config.save()
2007-02-15 21:30:17 +00:00
self.plugins.shutdown_all_plugins()
2007-01-11 00:58:23 +00:00
self.manager.quit()
2007-01-11 21:53:25 +00:00
gtk.main_quit()
2007-05-25 07:31:31 +00:00
2006-12-08 20:26:35 +00:00
## For testing purposes, create a copy of the interface
2006-11-28 22:28:37 +00:00
if __name__ == "__main__":
2007-01-11 00:58:23 +00:00
interface = DelugeGTK()
2007-02-06 21:21:09 +00:00
interface.start()
2007-05-25 07:31:31 +00:00