Clean up WaitingDialog
Prevent GC so callers don't have to.
This commit is contained in:
@@ -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:
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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)
|
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
Reference in New Issue
Block a user