gtk tracker icons, some eye-candy
This commit is contained in:
parent
09fe2e696b
commit
cb8125aa1b
|
@ -69,7 +69,7 @@ class TorrentOptions(dict):
|
||||||
"download_location": "download_location",
|
"download_location": "download_location",
|
||||||
"add_paused": "add_paused"
|
"add_paused": "add_paused"
|
||||||
}
|
}
|
||||||
|
|
||||||
def items(self):
|
def items(self):
|
||||||
i = super(TorrentOptions, self).items()
|
i = super(TorrentOptions, self).items()
|
||||||
for k in self.default_keys:
|
for k in self.default_keys:
|
||||||
|
@ -77,27 +77,27 @@ class TorrentOptions(dict):
|
||||||
i.append((k, self.__getitem__(k)))
|
i.append((k, self.__getitem__(k)))
|
||||||
|
|
||||||
return i
|
return i
|
||||||
|
|
||||||
def keys(self):
|
def keys(self):
|
||||||
k = super(TorrentOptions, self).keys()
|
k = super(TorrentOptions, self).keys()
|
||||||
for key in self.default_keys.keys():
|
for key in self.default_keys.keys():
|
||||||
if key not in k:
|
if key not in k:
|
||||||
k.append(key)
|
k.append(key)
|
||||||
return k
|
return k
|
||||||
|
|
||||||
def iteritems(self):
|
def iteritems(self):
|
||||||
return self.items().itermitems()
|
return self.items().itermitems()
|
||||||
|
|
||||||
def has_key(self, key):
|
def has_key(self, key):
|
||||||
if super(TorrentOptions, self).has_key(key):
|
if super(TorrentOptions, self).has_key(key):
|
||||||
return True
|
return True
|
||||||
elif self.default_keys.has_key(key):
|
elif self.default_keys.has_key(key):
|
||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def __setitem__(self, key, value):
|
def __setitem__(self, key, value):
|
||||||
super(TorrentOptions, self).__setitem__(key, value)
|
super(TorrentOptions, self).__setitem__(key, value)
|
||||||
|
|
||||||
def __getitem__(self, key):
|
def __getitem__(self, key):
|
||||||
if super(TorrentOptions, self).has_key(key):
|
if super(TorrentOptions, self).has_key(key):
|
||||||
return super(TorrentOptions, self).__getitem__(key)
|
return super(TorrentOptions, self).__getitem__(key)
|
||||||
|
@ -106,7 +106,7 @@ class TorrentOptions(dict):
|
||||||
return self.config[self.default_keys[key]]
|
return self.config[self.default_keys[key]]
|
||||||
else:
|
else:
|
||||||
return self.default_keys[key]
|
return self.default_keys[key]
|
||||||
|
|
||||||
class Torrent:
|
class Torrent:
|
||||||
"""Torrent holds information about torrents added to the libtorrent session.
|
"""Torrent holds information about torrents added to the libtorrent session.
|
||||||
"""
|
"""
|
||||||
|
@ -124,22 +124,22 @@ class Torrent:
|
||||||
|
|
||||||
# Let's us know if we're waiting on a lt alert
|
# Let's us know if we're waiting on a lt alert
|
||||||
self.waiting_on_resume_data = False
|
self.waiting_on_resume_data = False
|
||||||
|
|
||||||
# Keep a list of file indexes we're waiting for file_rename alerts on
|
# Keep a list of file indexes we're waiting for file_rename alerts on
|
||||||
# This is so we can send one folder_renamed signal instead of multiple
|
# This is so we can send one folder_renamed signal instead of multiple
|
||||||
# file_renamed signals.
|
# file_renamed signals.
|
||||||
self.waiting_on_folder_rename = []
|
self.waiting_on_folder_rename = []
|
||||||
|
|
||||||
# We store the filename just in case we need to make a copy of the torrentfile
|
# We store the filename just in case we need to make a copy of the torrentfile
|
||||||
if not filename:
|
if not filename:
|
||||||
# If no filename was provided, then just use the infohash
|
# If no filename was provided, then just use the infohash
|
||||||
filename = self.torrent_id
|
filename = self.torrent_id
|
||||||
|
|
||||||
self.filename = filename
|
self.filename = filename
|
||||||
|
|
||||||
# Store the magnet uri used to add this torrent if available
|
# Store the magnet uri used to add this torrent if available
|
||||||
self.magnet = magnet
|
self.magnet = magnet
|
||||||
|
|
||||||
# Holds status info so that we don't need to keep getting it from lt
|
# Holds status info so that we don't need to keep getting it from lt
|
||||||
self.status = self.handle.status()
|
self.status = self.handle.status()
|
||||||
|
|
||||||
|
@ -147,17 +147,17 @@ class Torrent:
|
||||||
self.torrent_info = self.handle.get_torrent_info()
|
self.torrent_info = self.handle.get_torrent_info()
|
||||||
except RuntimeError:
|
except RuntimeError:
|
||||||
self.torrent_info = None
|
self.torrent_info = None
|
||||||
|
|
||||||
# Files dictionary
|
# Files dictionary
|
||||||
self.files = self.get_files()
|
self.files = self.get_files()
|
||||||
|
|
||||||
# Default total_uploaded to 0, this may be changed by the state
|
# Default total_uploaded to 0, this may be changed by the state
|
||||||
self.total_uploaded = 0
|
self.total_uploaded = 0
|
||||||
|
|
||||||
# Set the default options
|
# Set the default options
|
||||||
self.options = TorrentOptions()
|
self.options = TorrentOptions()
|
||||||
self.options.update(options)
|
self.options.update(options)
|
||||||
|
|
||||||
# We need to keep track if the torrent is finished in the state to prevent
|
# We need to keep track if the torrent is finished in the state to prevent
|
||||||
# some weird things on state load.
|
# some weird things on state load.
|
||||||
self.is_finished = False
|
self.is_finished = False
|
||||||
|
@ -188,7 +188,7 @@ class Torrent:
|
||||||
# Various torrent options
|
# Various torrent options
|
||||||
self.handle.resolve_countries(True)
|
self.handle.resolve_countries(True)
|
||||||
self.set_options(self.options)
|
self.set_options(self.options)
|
||||||
|
|
||||||
# Status message holds error info about the torrent
|
# Status message holds error info about the torrent
|
||||||
self.statusmsg = "OK"
|
self.statusmsg = "OK"
|
||||||
|
|
||||||
|
@ -216,13 +216,13 @@ class Torrent:
|
||||||
for (key, value) in options.items():
|
for (key, value) in options.items():
|
||||||
if OPTIONS_FUNCS.has_key(key):
|
if OPTIONS_FUNCS.has_key(key):
|
||||||
OPTIONS_FUNCS[key](value)
|
OPTIONS_FUNCS[key](value)
|
||||||
|
|
||||||
self.options.update(options)
|
self.options.update(options)
|
||||||
|
|
||||||
def get_options(self):
|
def get_options(self):
|
||||||
return self.options
|
return self.options
|
||||||
|
|
||||||
|
|
||||||
def set_max_connections(self, max_connections):
|
def set_max_connections(self, max_connections):
|
||||||
self.options["max_connections"] = int(max_connections)
|
self.options["max_connections"] = int(max_connections)
|
||||||
self.handle.set_max_connections(max_connections)
|
self.handle.set_max_connections(max_connections)
|
||||||
|
@ -286,7 +286,7 @@ class Torrent:
|
||||||
return
|
return
|
||||||
|
|
||||||
log.debug("setting %s's file priorities: %s", self.torrent_id, file_priorities)
|
log.debug("setting %s's file priorities: %s", self.torrent_id, file_priorities)
|
||||||
|
|
||||||
self.handle.prioritize_files(file_priorities)
|
self.handle.prioritize_files(file_priorities)
|
||||||
|
|
||||||
if 0 in self.options["file_priorities"]:
|
if 0 in self.options["file_priorities"]:
|
||||||
|
@ -303,7 +303,7 @@ class Torrent:
|
||||||
|
|
||||||
# Set the first/last priorities if needed
|
# Set the first/last priorities if needed
|
||||||
self.set_prioritize_first_last(self.options["prioritize_first_last_pieces"])
|
self.set_prioritize_first_last(self.options["prioritize_first_last_pieces"])
|
||||||
|
|
||||||
def set_trackers(self, trackers):
|
def set_trackers(self, trackers):
|
||||||
"""Sets trackers"""
|
"""Sets trackers"""
|
||||||
if trackers == None:
|
if trackers == None:
|
||||||
|
@ -315,7 +315,7 @@ class Torrent:
|
||||||
trackers.append(tracker)
|
trackers.append(tracker)
|
||||||
self.trackers = trackers
|
self.trackers = trackers
|
||||||
return
|
return
|
||||||
|
|
||||||
log.debug("Setting trackers for %s: %s", self.torrent_id, trackers)
|
log.debug("Setting trackers for %s: %s", self.torrent_id, trackers)
|
||||||
tracker_list = []
|
tracker_list = []
|
||||||
|
|
||||||
|
@ -339,7 +339,7 @@ class Torrent:
|
||||||
|
|
||||||
def set_save_path(self, save_path):
|
def set_save_path(self, save_path):
|
||||||
self.options["download_location"] = save_path
|
self.options["download_location"] = save_path
|
||||||
|
|
||||||
def set_tracker_status(self, status):
|
def set_tracker_status(self, status):
|
||||||
"""Sets the tracker status"""
|
"""Sets the tracker status"""
|
||||||
self.tracker_status = status
|
self.tracker_status = status
|
||||||
|
@ -430,10 +430,10 @@ class Torrent:
|
||||||
torrent_info = self.handle.get_torrent_info()
|
torrent_info = self.handle.get_torrent_info()
|
||||||
else:
|
else:
|
||||||
torrent_info = self.torrent_info
|
torrent_info = self.torrent_info
|
||||||
|
|
||||||
if not torrent_info:
|
if not torrent_info:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
ret = []
|
ret = []
|
||||||
files = torrent_info.files()
|
files = torrent_info.files()
|
||||||
for index, file in enumerate(files):
|
for index, file in enumerate(files):
|
||||||
|
@ -487,7 +487,7 @@ class Torrent:
|
||||||
"""Returns the file progress as a list of floats.. 0.0 -> 1.0"""
|
"""Returns the file progress as a list of floats.. 0.0 -> 1.0"""
|
||||||
if not self.handle.has_metadata():
|
if not self.handle.has_metadata():
|
||||||
return 0.0
|
return 0.0
|
||||||
|
|
||||||
file_progress = self.handle.file_progress()
|
file_progress = self.handle.file_progress()
|
||||||
ret = []
|
ret = []
|
||||||
for i,f in enumerate(self.files):
|
for i,f in enumerate(self.files):
|
||||||
|
@ -511,7 +511,7 @@ class Torrent:
|
||||||
if tracker:
|
if tracker:
|
||||||
url = urlparse(tracker)
|
url = urlparse(tracker)
|
||||||
if hasattr(url, "hostname"):
|
if hasattr(url, "hostname"):
|
||||||
host = (url.hostname or 'unknown?')
|
host = (url.hostname or 'DHT')
|
||||||
parts = host.split(".")
|
parts = host.split(".")
|
||||||
if len(parts) > 2:
|
if len(parts) > 2:
|
||||||
host = ".".join(parts[-2:])
|
host = ".".join(parts[-2:])
|
||||||
|
@ -575,7 +575,7 @@ class Torrent:
|
||||||
"move_on_completed": self.options["move_completed"],
|
"move_on_completed": self.options["move_completed"],
|
||||||
"move_on_completed_path": self.options["move_completed_path"]
|
"move_on_completed_path": self.options["move_completed_path"]
|
||||||
}
|
}
|
||||||
|
|
||||||
def ti_name():
|
def ti_name():
|
||||||
if self.handle.has_metadata():
|
if self.handle.has_metadata():
|
||||||
return self.torrent_info.name()
|
return self.torrent_info.name()
|
||||||
|
@ -600,7 +600,7 @@ class Torrent:
|
||||||
if self.handle.has_metadata():
|
if self.handle.has_metadata():
|
||||||
return self.torrent_info.piece_length()
|
return self.torrent_info.piece_length()
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
fns = {
|
fns = {
|
||||||
"name": ti_name,
|
"name": ti_name,
|
||||||
"private": ti_priv,
|
"private": ti_priv,
|
||||||
|
@ -633,7 +633,7 @@ class Torrent:
|
||||||
|
|
||||||
self.status = None
|
self.status = None
|
||||||
self.torrent_info = None
|
self.torrent_info = None
|
||||||
|
|
||||||
return status_dict
|
return status_dict
|
||||||
|
|
||||||
def apply_options(self):
|
def apply_options(self):
|
||||||
|
@ -721,7 +721,7 @@ class Torrent:
|
||||||
returned in a libtorrent alert"""
|
returned in a libtorrent alert"""
|
||||||
self.handle.save_resume_data()
|
self.handle.save_resume_data()
|
||||||
self.waiting_on_resume_data = True
|
self.waiting_on_resume_data = True
|
||||||
|
|
||||||
def write_resume_data(self, resume_data):
|
def write_resume_data(self, resume_data):
|
||||||
"""Writes the .fastresume file for the torrent"""
|
"""Writes the .fastresume file for the torrent"""
|
||||||
resume_data = lt.bencode(resume_data)
|
resume_data = lt.bencode(resume_data)
|
||||||
|
@ -736,7 +736,7 @@ class Torrent:
|
||||||
fastresume.close()
|
fastresume.close()
|
||||||
except IOError:
|
except IOError:
|
||||||
log.warning("Error trying to save fastresume file")
|
log.warning("Error trying to save fastresume file")
|
||||||
|
|
||||||
self.waiting_on_resume_data = False
|
self.waiting_on_resume_data = False
|
||||||
|
|
||||||
def delete_fastresume(self):
|
def delete_fastresume(self):
|
||||||
|
@ -765,7 +765,7 @@ class Torrent:
|
||||||
open(path, "wb").write(lt.bencode(torrent_file))
|
open(path, "wb").write(lt.bencode(torrent_file))
|
||||||
except Exception, e:
|
except Exception, e:
|
||||||
log.warning("Unable to save torrent file: %s", e)
|
log.warning("Unable to save torrent file: %s", e)
|
||||||
|
|
||||||
def delete_torrentfile(self):
|
def delete_torrentfile(self):
|
||||||
"""Deletes the .torrent file in the state"""
|
"""Deletes the .torrent file in the state"""
|
||||||
path = "%s/%s.torrent" % (
|
path = "%s/%s.torrent" % (
|
||||||
|
@ -807,7 +807,7 @@ class Torrent:
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def rename_files(self, filenames):
|
def rename_files(self, filenames):
|
||||||
"""Renames files in the torrent. 'filenames' should be a list of
|
"""Renames files in the torrent. 'filenames' should be a list of
|
||||||
(index, filename) pairs."""
|
(index, filename) pairs."""
|
||||||
for index, filename in filenames:
|
for index, filename in filenames:
|
||||||
self.handle.rename_file(index, filename)
|
self.handle.rename_file(index, filename)
|
||||||
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
from deluge.tracker_icons import TrackerIcons
|
||||||
|
|
||||||
|
def test():
|
||||||
|
trackericons = TrackerIcons()
|
||||||
|
|
||||||
|
print trackericons._fetch_icon("unknown?") #exception, Returns None
|
||||||
|
print trackericons._fetch_icon("deluge-torrent.org") #404 , returns False
|
||||||
|
print trackericons._fetch_icon("legaltorrents.com")
|
||||||
|
print trackericons.get("thepiratebay.com")
|
||||||
|
print trackericons.get("unknown2") #exception, returns None
|
||||||
|
print trackericons.get("legaltorrents.com") #logs cached
|
||||||
|
print trackericons.get("google.com")
|
||||||
|
|
||||||
|
|
||||||
|
test()
|
|
@ -0,0 +1,105 @@
|
||||||
|
#
|
||||||
|
# tracker_icons.py
|
||||||
|
#
|
||||||
|
# Copyright (C) 2008 Martijn Voncken <mvoncken@gmail.com>
|
||||||
|
#
|
||||||
|
# Deluge is free software.
|
||||||
|
#
|
||||||
|
# You may redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License, as published by the Free Software
|
||||||
|
# Foundation; either version 3 of the License, or (at your option)
|
||||||
|
# any later version.
|
||||||
|
#
|
||||||
|
# deluge is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with deluge. If not, write to:
|
||||||
|
# The Free Software Foundation, Inc.,
|
||||||
|
# 51 Franklin Street, Fifth Floor
|
||||||
|
# Boston, MA 02110-1301, USA.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
from urllib import urlopen
|
||||||
|
from deluge.log import LOG as log
|
||||||
|
from deluge.common import get_default_config_dir, get_pixmap
|
||||||
|
import os
|
||||||
|
|
||||||
|
#some servers don't have their favicon at the expected location
|
||||||
|
RENAMES = {"legaltorrents.com":"beta.legaltorrents.com"}
|
||||||
|
VALID_TYPES = ["octet-stream","x-icon"]
|
||||||
|
|
||||||
|
class TrackerIcons(object):
|
||||||
|
def __init__(self):
|
||||||
|
#set image cache dir
|
||||||
|
self.image_dir = get_default_config_dir("trackers")
|
||||||
|
if not os.path.exists(self.image_dir):
|
||||||
|
os.mkdir(self.image_dir)
|
||||||
|
|
||||||
|
#self.images : {tracker_host:filename}
|
||||||
|
self.images = {"DHT":get_pixmap("dht16.png" )}
|
||||||
|
|
||||||
|
#load image-names in cache-dir
|
||||||
|
for icon in os.listdir(self.image_dir):
|
||||||
|
if icon.endswith(".ico"):
|
||||||
|
self.add_icon(icon[:-4], os.path.join(self.image_dir, icon))
|
||||||
|
|
||||||
|
|
||||||
|
def _fetch_icon(self, tracker_host):
|
||||||
|
"""
|
||||||
|
gets new icon from the internet.
|
||||||
|
used by get().
|
||||||
|
calls add_icon()
|
||||||
|
returns True or False
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
host_name = RENAMES.get(tracker_host, tracker_host)
|
||||||
|
icon = urlopen("http://%s/favicon.ico" % host_name)
|
||||||
|
icon_data = icon.read()
|
||||||
|
|
||||||
|
#validate icon:
|
||||||
|
if icon.info().getsubtype() not in VALID_TYPES:
|
||||||
|
raise Exception("Unexpected type: %s" % icon.info().getsubtype())
|
||||||
|
if not icon_data:
|
||||||
|
raise Exception("No data")
|
||||||
|
except Exception, e:
|
||||||
|
log.debug("%s %s %s" % (tracker_host, e, e.message))
|
||||||
|
self.add_icon(tracker_host, None)
|
||||||
|
return False
|
||||||
|
|
||||||
|
filename = os.path.join(get_default_config_dir("trackers"),"%s.ico" % tracker_host)
|
||||||
|
|
||||||
|
f = open(filename,"wb")
|
||||||
|
f.write(icon_data)
|
||||||
|
f.close()
|
||||||
|
self.add_icon(tracker_host, filename)
|
||||||
|
return True
|
||||||
|
|
||||||
|
def add_icon(self, tracker_host, filename):
|
||||||
|
self.images[tracker_host] = filename
|
||||||
|
|
||||||
|
def get(self, tracker_host):
|
||||||
|
"""
|
||||||
|
use this method to get the filename of an icon.
|
||||||
|
"""
|
||||||
|
if not tracker_host in self.images:
|
||||||
|
self._fetch_icon(tracker_host)
|
||||||
|
else:
|
||||||
|
log.debug("cached tracker icon:%s" % tracker_host)
|
||||||
|
return self.images[tracker_host]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -38,6 +38,7 @@ import pkg_resources
|
||||||
|
|
||||||
import deluge.component as component
|
import deluge.component as component
|
||||||
import deluge.common
|
import deluge.common
|
||||||
|
from deluge.tracker_icons import TrackerIcons
|
||||||
from deluge.log import LOG as log
|
from deluge.log import LOG as log
|
||||||
from deluge.ui.client import aclient
|
from deluge.ui.client import aclient
|
||||||
from deluge.configmanager import ConfigManager
|
from deluge.configmanager import ConfigManager
|
||||||
|
@ -78,7 +79,7 @@ class FilterTreeView(component.Component):
|
||||||
self.scrolled = glade.get_widget("scrolledwindow_sidebar")
|
self.scrolled = glade.get_widget("scrolledwindow_sidebar")
|
||||||
self.sidebar = component.get("SideBar")
|
self.sidebar = component.get("SideBar")
|
||||||
self.config = ConfigManager("gtkui.conf")
|
self.config = ConfigManager("gtkui.conf")
|
||||||
|
self.tracker_icons = TrackerIcons()
|
||||||
|
|
||||||
self.filters = {}
|
self.filters = {}
|
||||||
self.label_view = gtk.TreeView()
|
self.label_view = gtk.TreeView()
|
||||||
|
@ -201,12 +202,13 @@ class FilterTreeView(component.Component):
|
||||||
value = model.get_value(row, 1)
|
value = model.get_value(row, 1)
|
||||||
label = model.get_value(row, 2)
|
label = model.get_value(row, 2)
|
||||||
count = model.get_value(row, 3)
|
count = model.get_value(row, 3)
|
||||||
|
pix = model.get_value(row, 4)
|
||||||
|
|
||||||
if (label == "") and cat == "label":
|
if (label == "") and cat == "label":
|
||||||
label = _("no label")
|
label = _("no label")
|
||||||
|
|
||||||
|
|
||||||
if cat == "state":
|
if pix:
|
||||||
self.renderpix.set_property("visible", True)
|
self.renderpix.set_property("visible", True)
|
||||||
else:
|
else:
|
||||||
self.renderpix.set_property("visible", False)
|
self.renderpix.set_property("visible", False)
|
||||||
|
@ -227,6 +229,15 @@ class FilterTreeView(component.Component):
|
||||||
if cat == "state":
|
if cat == "state":
|
||||||
pix = STATE_PIX.get(value, "dht")
|
pix = STATE_PIX.get(value, "dht")
|
||||||
return gtk.gdk.pixbuf_new_from_file(deluge.common.get_pixmap("%s16.png" % pix))
|
return gtk.gdk.pixbuf_new_from_file(deluge.common.get_pixmap("%s16.png" % pix))
|
||||||
|
|
||||||
|
elif cat == "tracker_host":
|
||||||
|
ico = self.tracker_icons.get(value)
|
||||||
|
if ico:
|
||||||
|
try: #assume we could get trashed images here..
|
||||||
|
return gtk.gdk.pixbuf_new_from_file(ico)
|
||||||
|
except:
|
||||||
|
log.debug(e.message)
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def on_selection_changed(self, selection):
|
def on_selection_changed(self, selection):
|
||||||
|
|
Loading…
Reference in New Issue