Transmission-XBMC/resources/lib/gui.py

288 lines
11 KiB
Python
Raw Permalink Normal View History

2010-05-28 02:38:06 +00:00
# -*- coding: utf-8 -*-
2010-07-13 01:09:42 +00:00
# Copyright (c) 2010 Correl J. Roush
2010-05-28 02:38:06 +00:00
import sys
import base64
import threading
2010-05-28 02:38:06 +00:00
import xbmc
import xbmcgui
from basictypes.bytes import Bytes
import transmissionrpc
2011-09-13 05:27:04 +00:00
import search
import common
2010-05-28 02:38:06 +00:00
_ = sys.modules[ "__main__" ].__language__
__settings__ = sys.modules[ "__main__" ].__settings__
2010-05-28 02:38:06 +00:00
KEY_BUTTON_BACK = 275
KEY_KEYBOARD_ESC = 61467
KEY_MENU_ID = 92
2010-05-28 02:38:06 +00:00
EXIT_SCRIPT = ( 6, 10, 247, 275, 61467, 216, 257, 61448, )
CANCEL_DIALOG = EXIT_SCRIPT + ( 216, 257, 61448, )
UPDATE_INTERVAL = 1.0
STATUS_ICONS = {'stopped': 'pause.png',
'seeding': 'ok.png',
'downloading': 'down.png'}
2010-05-28 02:38:06 +00:00
class TransmissionGUI(xbmcgui.WindowXMLDialog):
def __init__(self, strXMLname, strFallbackPath, strDefaultName, bforeFallback=0):
self.list = {}
self.torrents = {}
self.timer = None
def set_settings(self, params):
__settings__.setSetting('rpc_host', params['address'])
__settings__.setSetting('rpc_port', params['port'])
__settings__.setSetting('rpc_user', params['user'])
__settings__.setSetting('rpc_password', params['password'])
def onInit(self):
p = xbmcgui.DialogProgress()
p.create(_(32000), _(32001)) # 'Transmission', 'Connecting to Transmission'
try:
self.transmission = common.get_rpc_client()
2011-03-29 03:22:40 +00:00
except:
p.close()
self.close()
2011-03-29 03:22:40 +00:00
(type, e, traceback) = sys.exc_info()
message = _(32900) # Unexpected error
if type is transmissionrpc.TransmissionError:
if e.original:
if e.original.code is 401:
message = _(32902) # Invalid auth
else:
message = _(32901) # Unable to connect
if xbmcgui.Dialog().yesno(_(32002), message, _(32003)):
__settings__.openSettings()
elif type is ValueError:
# In python 2.4, urllib2.HTTPDigestAuthHandler will barf up a lung
# if auth fails and the server wants non-digest authentication
message = _(32902) # Invalid auth
if xbmcgui.Dialog().yesno(_(32002), message, _(32003)):
__settings__.openSettings()
else:
message = _(32900) # Unexpected error
xbmcgui.Dialog().ok(_(32002), message)
return False
2010-05-28 02:38:06 +00:00
self.updateTorrents()
p.close()
self.timer = threading.Timer(UPDATE_INTERVAL, self.updateTorrents)
self.timer.start()
2010-05-28 02:38:06 +00:00
def updateTorrents(self):
list = self.getControl(120)
self.torrents = self.transmission.info()
for i, torrent in self.torrents.iteritems():
2010-05-28 02:38:06 +00:00
statusline = "[%(status)s] %(down)s down (%(pct).2f%%), %(up)s up (Ratio: %(ratio).2f)" % \
{'down': Bytes.format(torrent.downloadedEver), 'pct': torrent.progress, \
'up': Bytes.format(torrent.uploadedEver), 'ratio': torrent.ratio, \
'status': torrent.status}
if i not in self.list:
# Create a new list item
l = xbmcgui.ListItem(label=torrent.name, label2=statusline)
list.addItem(l)
self.list[i] = l
else:
# Update existing list item
l = self.list[i]
l.setLabel(torrent.name)
l.setLabel2(statusline)
l.setProperty('TorrentStatusIcon', STATUS_ICONS.get(torrent.status, 'pending.png'))
2010-05-28 02:38:06 +00:00
l.setProperty('TorrentID', str(i))
l.setProperty('TorrentProgress', "%3d%%" % torrent.progress)
2010-05-28 02:38:06 +00:00
removed = [id for id in self.list.keys() if id not in self.torrents.keys()]
2010-05-28 02:38:06 +00:00
if len(removed) > 0:
# Clear torrents from the list that have been removed
for id in removed:
del self.list[id]
list.reset()
for id, item in self.list.iteritems():
list.addItem(item)
list.setEnabled(bool(self.torrents))
2013-04-16 07:13:50 +00:00
# Update again, after an interval, but only if the timer has not been cancelled
if self.timer:
self.timer = threading.Timer(UPDATE_INTERVAL, self.updateTorrents)
self.timer.start()
2010-05-28 02:38:06 +00:00
def onClick(self, controlID):
list = self.getControl(120)
if (controlID == 111):
2010-05-28 02:38:06 +00:00
# Add torrent
2011-09-13 05:27:04 +00:00
engines = [
(_(32200), None),
(_(32202), search.TPB),
(_(32203), search.Mininova),
(_(32204), search.Kickass),
(_(32205), search.L337x),
(_(32206), search.YTS),
(_(32207), search.Lime),
2015-04-17 00:34:07 +00:00
(_(32208), search.EZTV),
2011-09-13 05:27:04 +00:00
]
selected = xbmcgui.Dialog().select(_(32000), [i[0] for i in engines])
2011-09-13 05:27:04 +00:00
if selected < 0:
return
engine = engines[selected][1]
if not engine:
filename = xbmcgui.Dialog().browse(1, _(32000), 'files', '.torrent')
2011-09-13 05:27:04 +00:00
try:
f = open(filename, 'r')
data = base64.b64encode(f.read())
self.transmission.add(data)
except:
pass
else:
kb = xbmc.Keyboard('', engines[selected][0])
kb.doModal()
if not kb.isConfirmed():
return
terms = kb.getText()
p = xbmcgui.DialogProgress()
p.create(_(32000), _(32290))
2011-09-13 05:27:04 +00:00
try:
results = engine().search(terms)
except:
p.close()
xbmcgui.Dialog().ok(_(32000), _(32292))
2011-09-13 05:27:04 +00:00
return
p.close()
if not results:
xbmcgui.Dialog().ok(_(32000), _(32291))
2011-09-13 05:27:04 +00:00
return
selected = xbmcgui.Dialog().select(_(32000), ['[S:%d L:%d] %s' % (t['seeds'], t['leechers'], t['name']) for t in results])
2011-09-13 05:27:04 +00:00
if selected < 0:
return
try:
2014-07-27 00:39:57 +00:00
self.transmission.add_torrent(results[selected]['url'])
2011-09-13 05:27:04 +00:00
except:
xbmcgui.Dialog().ok(_(32000), _(32293))
2011-09-13 05:27:04 +00:00
return
if (controlID == 112):
2010-05-28 02:38:06 +00:00
# Remove selected torrent
item = list.getSelectedItem()
if item and xbmcgui.Dialog().yesno(_(32000), 'Remove \'%s\'?' % self.torrents[int(item.getProperty('TorrentID'))].name):
remove_data = xbmcgui.Dialog().yesno(_(32000), 'Remove data as well?')
2010-05-28 02:38:06 +00:00
self.transmission.remove(int(item.getProperty('TorrentID')), remove_data)
if (controlID == 113):
2010-05-28 02:38:06 +00:00
# Stop selected torrent
item = list.getSelectedItem()
if item:
self.transmission.stop(int(item.getProperty('TorrentID')))
if (controlID == 114):
2010-05-28 02:38:06 +00:00
# Start selected torrent
item = list.getSelectedItem()
if item:
self.transmission.start(int(item.getProperty('TorrentID')))
if (controlID == 115):
2010-05-28 02:38:06 +00:00
# Stop all torrents
self.transmission.stop(self.torrents.keys())
if (controlID == 116):
2010-05-28 02:38:06 +00:00
# Start all torrents
self.transmission.start(self.torrents.keys())
if (controlID == 117):
2010-05-28 02:38:06 +00:00
# Exit button
2011-03-29 03:22:40 +00:00
self.close()
if (controlID == 118):
# Settings button
prev_settings = common.get_settings()
__settings__.openSettings()
p = xbmcgui.DialogProgress()
p.create(_(32000), _(32001)) # 'Transmission', 'Connecting to Transmission'
try:
self.transmission = common.get_rpc_client()
self.updateTorrents()
p.close()
except:
p.close()
xbmcgui.Dialog().ok(_(32002), _(32901))
# restore settings
self.set_settings(prev_settings)
try:
self.transmission = common.get_rpc_client()
except err:
xbmcgui.Dialog().ok(_(32002), _(32901))
self.close()
if (controlID == 120):
2010-05-28 02:38:06 +00:00
# A torrent was chosen, show details
item = list.getSelectedItem()
w = TorrentInfoGUI("script-Transmission-details.xml", __settings__.getAddonInfo('path') ,"Default")
2010-07-16 19:36:06 +00:00
w.setTorrent(self.transmission, int(item.getProperty('TorrentID')))
2010-05-28 02:38:06 +00:00
w.doModal()
del w
def onFocus(self, controlID):
pass
def onAction( self, action ):
if ( action.getButtonCode() in CANCEL_DIALOG ) or (action.getId() == KEY_MENU_ID):
2011-03-29 03:22:40 +00:00
self.close()
def close(self):
if self.timer:
self.timer.cancel()
self.timer.join()
2011-03-29 03:22:40 +00:00
super(TransmissionGUI, self).close()
2010-05-28 02:38:06 +00:00
class TorrentInfoGUI(xbmcgui.WindowXMLDialog):
def __init__(self, strXMLname, strFallbackPath, strDefaultName, bforeFallback=0):
2010-07-16 19:36:06 +00:00
self.transmission = None
2010-05-28 02:38:06 +00:00
self.torrent_id = None
2010-07-16 19:36:06 +00:00
self.list = {}
self.timer = None
2010-07-16 19:36:06 +00:00
def setTorrent(self, transmission, t_id):
self.transmission = transmission
2010-05-28 02:38:06 +00:00
self.torrent_id = t_id
self.timer = threading.Timer(UPDATE_INTERVAL, self.updateTorrent)
self.timer.start()
2010-07-16 19:36:06 +00:00
def updateTorrent(self):
pbar = self.getControl(219)
list = self.getControl(220)
labelName = self.getControl(1)
labelStatus = self.getControl(2)
labelProgress = self.getControl(11)
2010-07-16 19:36:06 +00:00
torrent = self.transmission.info()[self.torrent_id]
files = self.transmission.get_files(self.torrent_id)[self.torrent_id]
statusline = "[%(status)s] %(down)s down, %(up)s up (Ratio: %(ratio).2f)" % \
2010-07-16 19:36:06 +00:00
{'down': Bytes.format(torrent.downloadedEver), 'pct': torrent.progress, \
'up': Bytes.format(torrent.uploadedEver), 'ratio': torrent.ratio, \
'status': torrent.status}
if torrent.status is 'downloading':
statusline += " ETA: %(eta)s" % \
{'eta': torrent.eta}
2010-07-16 19:36:06 +00:00
labelName.setLabel(torrent.name)
labelStatus.setLabel(statusline)
labelProgress.setLabel('%3d%%' % (torrent.progress))
2010-07-16 19:36:06 +00:00
pbar.setPercent(torrent.progress)
2010-07-16 19:36:06 +00:00
for i, file in files.iteritems():
if i not in self.list:
# Create a new list item
l = xbmcgui.ListItem(label=file['name'])
list.addItem(l)
self.list[i] = l
else:
# Update existing list item
l = self.list[i]
l.setProperty('Progress', '[%3d%%]' % (file['completed'] * 100 / file['size']))
# Update again, after an interval
self.timer = threading.Timer(UPDATE_INTERVAL, self.updateTorrent)
self.timer.start()
2010-05-28 02:38:06 +00:00
def onInit(self):
2010-07-16 19:36:06 +00:00
self.updateTorrent()
def close(self):
if self.timer:
self.timer.cancel()
self.timer.join()
2010-07-16 19:36:06 +00:00
super(TorrentInfoGUI, self).close()
2010-05-28 02:38:06 +00:00
def onAction(self, action):
if (action.getButtonCode() in CANCEL_DIALOG) or (action.getId() == KEY_MENU_ID):
2010-05-28 02:38:06 +00:00
self.close()
2010-07-16 19:36:06 +00:00
pass
2010-05-28 02:38:06 +00:00
def onClick(self, controlID):
if controlID == 111:
self.close()
2010-05-28 02:38:06 +00:00
def onFocus(self, controlID):
pass