Browse Source

Clean up WaitingDialog

Prevent GC so callers don't have to.
283
Neil Booth 9 years ago
parent
commit
ffd081ac21
  1. 21
      gui/qt/main_window.py
  2. 79
      gui/qt/util.py
  3. 24
      plugins/audio_modem/qt.py
  4. 9
      plugins/trustedcoin/qt.py

21
gui/qt/main_window.py

@ -1280,16 +1280,14 @@ class ElectrumWindow(QMainWindow, MessageBoxMixin, PrintError):
def sign_thread(): def sign_thread():
if not self.wallet.is_watching_only(): if not self.wallet.is_watching_only():
self.wallet.sign_transaction(tx, password) self.wallet.sign_transaction(tx, password)
def on_sign_successful(ret): def on_signed(ret):
success[0] = True success[0] = True
def on_dialog_close(): def on_finished():
self.send_button.setDisabled(False) self.send_button.setDisabled(False)
callback(success[0]) callback(success[0])
# keep a reference to WaitingDialog or the gui might crash WaitingDialog(parent, _('Signing transaction...'), sign_thread,
self.waiting_dialog = WaitingDialog(parent, 'Signing transaction...', sign_thread, on_sign_successful, on_dialog_close) on_success=on_signed, on_finished=on_finished)
self.waiting_dialog.start()
def broadcast_transaction(self, tx, tx_desc, parent=None): def broadcast_transaction(self, tx, tx_desc, parent=None):
@ -1325,12 +1323,9 @@ class ElectrumWindow(QMainWindow, MessageBoxMixin, PrintError):
self.show_error(msg, parent=parent) self.show_error(msg, parent=parent)
self.send_button.setDisabled(False) self.send_button.setDisabled(False)
if parent == None: parent = parent or self
parent = self WaitingDialog(parent, _('Broadcasting transaction...'),
self.waiting_dialog = WaitingDialog(parent, 'Broadcasting transaction...', broadcast_thread, broadcast_done) broadcast_thread, broadcast_done)
self.waiting_dialog.start()
def prepare_for_payment_request(self): def prepare_for_payment_request(self):
self.tabs.setCurrentIndex(1) self.tabs.setCurrentIndex(1)
@ -2238,7 +2233,7 @@ class ElectrumWindow(QMainWindow, MessageBoxMixin, PrintError):
from electrum import qrscanner from electrum import qrscanner
try: try:
data = qrscanner.scan_qr(self.config) data = qrscanner.scan_qr(self.config)
except e: except BaseException as e:
self.show_error(str(e)) self.show_error(str(e))
return return
if not data: if not data:

79
gui/qt/util.py

@ -21,44 +21,7 @@ RED_FG = "QWidget {color:red;}"
BLUE_FG = "QWidget {color:blue;}" BLUE_FG = "QWidget {color:blue;}"
BLACK_FG = "QWidget {color:black;}" BLACK_FG = "QWidget {color:black;}"
dialogs = []
class WaitingDialog(QThread):
def __init__(self, parent, message, run_task, on_success=None, on_complete=None):
QThread.__init__(self)
self.parent = parent
self.d = QDialog(parent)
self.d.setWindowTitle('Please wait')
l = QLabel(message)
vbox = QVBoxLayout(self.d)
vbox.addWidget(l)
self.run_task = run_task
self.on_success = on_success
self.on_complete = on_complete
self.d.connect(self.d, SIGNAL('done'), self.close)
self.d.show()
def run(self):
self.error = None
try:
self.result = self.run_task()
except BaseException as e:
traceback.print_exc(file=sys.stdout)
self.error = str(e)
self.d.emit(SIGNAL('done'))
def close(self):
self.d.accept()
if self.error:
QMessageBox.warning(self.parent, _('Error'), self.error, _('OK'))
else:
if self.on_success:
if type(self.result) is not tuple:
self.result = (self.result,)
self.on_success(*self.result)
if self.on_complete:
self.on_complete()
class Timer(QThread): class Timer(QThread):
stopped = False stopped = False
@ -234,6 +197,46 @@ class WindowModalDialog(QDialog):
if title: if title:
self.setWindowTitle(title) self.setWindowTitle(title)
class WaitingDialog(QThread, MessageBoxMixin):
'''Shows a please wait dialog whilst runnning a task. It is not
necessary to maintain a reference to this dialog.'''
def __init__(self, parent, message, task, on_success=None,
on_finished=None):
global dialogs
dialogs.append(self) # Prevent GC
QThread.__init__(self)
self.task = task
self.on_success = on_success
self.on_finished = on_finished
self.dialog = WindowModalDialog(parent, _("Please wait"))
vbox = QVBoxLayout(self.dialog)
vbox.addWidget(QLabel(message))
self.dialog.show()
self.dialog.connect(self, SIGNAL("finished()"), self.finished)
self.start()
def run(self):
self.error = None
try:
self.result = self.task()
except BaseException as e:
traceback.print_exc(file=sys.stdout)
self.error = str(e)
def finished(self):
global dialogs
dialogs.remove(self)
if self.error:
self.show_error(self.error, parent=self.dialog.parent())
elif self.on_success:
result = self.result
if type(result) is not tuple:
result = (result,)
self.on_success(*result)
if self.on_finished:
self.on_finished()
self.dialog.accept()
def line_dialog(parent, title, label, ok_label, default=None): def line_dialog(parent, title, label, ok_label, default=None):
dialog = WindowModalDialog(parent, title) dialog = WindowModalDialog(parent, title)
dialog.setMinimumWidth(500) dialog.setMinimumWidth(500)

24
plugins/audio_modem/qt.py

@ -1,5 +1,4 @@
from functools import partial from functools import partial
import traceback
import zlib import zlib
import json import json
from io import BytesIO from io import BytesIO
@ -75,24 +74,20 @@ class Plugin(BasePlugin):
def handler(): def handler():
blob = json.dumps(dialog.tx.as_dict()) blob = json.dumps(dialog.tx.as_dict())
self.sender = self._send(parent=dialog, blob=blob) self._send(parent=dialog, blob=blob)
self.sender.start()
b.clicked.connect(handler) b.clicked.connect(handler)
dialog.sharing_buttons.insert(-1, b) dialog.sharing_buttons.insert(-1, b)
@hook @hook
def scan_text_edit(self, parent): def scan_text_edit(self, parent):
def handler(): parent.addButton(':icons/microphone.png', partial(self._recv, parent),
self.receiver = self._recv(parent=parent) _("Read from microphone"))
self.receiver.start()
parent.addButton(':icons/microphone.png', handler, _("Read from microphone"))
@hook @hook
def show_text_edit(self, parent): def show_text_edit(self, parent):
def handler(): def handler():
blob = str(parent.toPlainText()) blob = str(parent.toPlainText())
self.sender = self._send(parent=parent, blob=blob) self._send(parent=parent, blob=blob)
self.sender.start()
parent.addButton(':icons/speaker.png', handler, _("Send to speaker")) parent.addButton(':icons/speaker.png', handler, _("Send to speaker"))
def _audio_interface(self): def _audio_interface(self):
@ -101,31 +96,25 @@ class Plugin(BasePlugin):
def _send(self, parent, blob): def _send(self, parent, blob):
def sender_thread(): def sender_thread():
try:
with self._audio_interface() as interface: with self._audio_interface() as interface:
src = BytesIO(blob) src = BytesIO(blob)
dst = interface.player() dst = interface.player()
amodem.main.send(config=self.modem_config, src=src, dst=dst) amodem.main.send(config=self.modem_config, src=src, dst=dst)
except Exception:
traceback.print_exc()
print_msg('Sending:', repr(blob)) print_msg('Sending:', repr(blob))
blob = zlib.compress(blob) blob = zlib.compress(blob)
kbps = self.modem_config.modem_bps / 1e3 kbps = self.modem_config.modem_bps / 1e3
msg = 'Sending to Audio MODEM ({0:.1f} kbps)...'.format(kbps) msg = 'Sending to Audio MODEM ({0:.1f} kbps)...'.format(kbps)
return WaitingDialog(parent=parent, message=msg, run_task=sender_thread) WaitingDialog(parent, msg, sender_thread)
def _recv(self, parent): def _recv(self, parent):
def receiver_thread(): def receiver_thread():
try:
with self._audio_interface() as interface: with self._audio_interface() as interface:
src = interface.recorder() src = interface.recorder()
dst = BytesIO() dst = BytesIO()
amodem.main.recv(config=self.modem_config, src=src, dst=dst) amodem.main.recv(config=self.modem_config, src=src, dst=dst)
return dst.getvalue() return dst.getvalue()
except Exception:
traceback.print_exc()
def on_success(blob): def on_success(blob):
if blob: if blob:
@ -135,5 +124,4 @@ class Plugin(BasePlugin):
kbps = self.modem_config.modem_bps / 1e3 kbps = self.modem_config.modem_bps / 1e3
msg = 'Receiving from Audio MODEM ({0:.1f} kbps)...'.format(kbps) msg = 'Receiving from Audio MODEM ({0:.1f} kbps)...'.format(kbps)
return WaitingDialog(parent=parent, message=msg, WaitingDialog(parent, msg, receiver_thread, on_success=on_success)
run_task=receiver_thread, on_success=on_success)

9
plugins/trustedcoin/qt.py

@ -92,9 +92,8 @@ class Plugin(TrustedCoinPlugin):
if wallet.billing_info is None: if wallet.billing_info is None:
# request billing info before forming the transaction # request billing info before forming the transaction
task = partial(self.request_billing_info, wallet) task = partial(self.request_billing_info, wallet)
waiting_dialog = WaitingDialog(window, 'please wait...', task) dialog = WaitingDialog(window, 'please wait...', task)
waiting_dialog.start() dialog.wait()
waiting_dialog.wait()
if wallet.billing_info is None: if wallet.billing_info is None:
window.show_message('Could not contact server') window.show_message('Could not contact server')
return True return True
@ -103,8 +102,8 @@ class Plugin(TrustedCoinPlugin):
def settings_dialog(self, window): def settings_dialog(self, window):
task = partial(self.request_billing_info, window.wallet) task = partial(self.request_billing_info, window.wallet)
self.waiting_dialog = WaitingDialog(window, 'please wait...', task, partial(self.show_settings_dialog, window)) WaitingDialog(window, 'please wait...', task,
self.waiting_dialog.start() on_success=partial(self.show_settings_dialog, window))
def show_settings_dialog(self, window, success): def show_settings_dialog(self, window, success):
if not success: if not success:

Loading…
Cancel
Save