diff --git a/gui/qt/qrtextedit.py b/gui/qt/qrtextedit.py index 8b7106e38..994733ab9 100644 --- a/gui/qt/qrtextedit.py +++ b/gui/qt/qrtextedit.py @@ -1,4 +1,5 @@ from electrum.i18n import _ +from electrum.plugins import run_hook from PyQt4.QtGui import * from PyQt4.QtCore import * @@ -25,6 +26,7 @@ class ShowQRTextEdit(QRTextEdit): super(ShowQRTextEdit, self).__init__(text) self.setReadOnly(1) self.button.clicked.connect(self.qr_show) + run_hook('show_text_edit', self) def qr_show(self): from qrcodewidget import QRDialog @@ -49,6 +51,7 @@ class ScanQRTextEdit(QRTextEdit): if win: assert hasattr(win,"config"), "You must pass a window with access to the config to ScanQRTextEdit constructor." self.button.clicked.connect(self.qr_input) + run_hook('scan_text_edit', self) def qr_input(self): diff --git a/icons.qrc b/icons.qrc index 94fd43e3c..e2197dd03 100644 --- a/icons.qrc +++ b/icons.qrc @@ -28,6 +28,8 @@ icons/network.png icons/dark_background.png icons/qrcode.png + icons/microphone.png + icons/speaker.png icons/trustedcoin.png diff --git a/icons/microphone.png b/icons/microphone.png new file mode 100644 index 000000000..f9a271c55 Binary files /dev/null and b/icons/microphone.png differ diff --git a/icons/speaker.png b/icons/speaker.png new file mode 100644 index 000000000..963db53f9 Binary files /dev/null and b/icons/speaker.png differ diff --git a/plugins/audio_modem.py b/plugins/audio_modem.py new file mode 100644 index 000000000..67b46aa5e --- /dev/null +++ b/plugins/audio_modem.py @@ -0,0 +1,171 @@ +from electrum.plugins import BasePlugin, hook +from electrum_gui.qt.util import WaitingDialog, EnterButton +from electrum.util import print_msg +from electrum.i18n import _ + +from PyQt4.QtGui import * +from PyQt4.QtCore import * + +import traceback +import zlib +import json +from io import BytesIO +import sys +import platform + +try: + import amodem.audio + import amodem.recv + import amodem.send + import amodem.config + print_msg('Audio MODEM is available.') + amodem.log.addHandler(amodem.logging.StreamHandler(sys.stderr)) + amodem.log.setLevel(amodem.logging.INFO) +except ImportError: + amodem = None + print_msg('Audio MODEM is not found.') + + +class Plugin(BasePlugin): + + def __init__(self, config, name): + BasePlugin.__init__(self, config, name) + if self.is_available(): + self.modem_config = amodem.config.slowest() + self.library_name = { + 'Linux': 'libportaudio.so' + }[platform.system()] + + def fullname(self): + return 'Audio MODEM' + + def description(self): + return ('Provides support for air-gapped transaction signing.\n\n' + 'Requires http://github.com/romanz/amodem/') + + def is_available(self): + return amodem is not None + + def requires_settings(self): + return True + + def settings_widget(self, window): + return EnterButton(_('Settings'), self.settings_dialog) + + def settings_dialog(self): + d = QDialog() + d.setWindowTitle("Settings") + + layout = QGridLayout(d) + layout.addWidget(QLabel(_('Bit rate [kbps]: ')), 0, 0) + + bitrates = list(sorted(amodem.config.bitrates.keys())) + + def _index_changed(index): + bitrate = bitrates[index] + self.modem_config = amodem.config.bitrates[bitrate] + + combo = QComboBox() + combo.addItems(map(str, bitrates)) + combo.currentIndexChanged.connect(_index_changed) + layout.addWidget(combo, 0, 1) + + ok_button = QPushButton(_("OK")) + ok_button.clicked.connect(d.accept) + layout.addWidget(ok_button, 1, 1) + + return bool(d.exec_()) + + @hook + def transaction_dialog(self, dialog): + b = QPushButton() + b.setIcon(QIcon(":icons/speaker.png")) + + def handler(): + blob = json.dumps(dialog.tx.as_dict()) + self.sender = self._send(parent=dialog, blob=blob) + self.sender.start() + b.clicked.connect(handler) + dialog.buttons.insertWidget(1, b) + + @hook + def scan_text_edit(self, parent): + def handler(): + self.receiver = self._recv(parent=parent) + self.receiver.start() + button = add_button(parent=parent, icon_name=':icons/microphone.png') + button.clicked.connect(handler) + + @hook + def show_text_edit(self, parent): + def handler(): + blob = str(parent.toPlainText()) + self.sender = self._send(parent=parent, blob=blob) + self.sender.start() + button = add_button(parent=parent, icon_name=':icons/speaker.png') + button.clicked.connect(handler) + + def _audio_interface(self): + return amodem.audio.Interface( + config=self.modem_config, + name=self.library_name + ) + + def _send(self, parent, blob): + def sender_thread(): + try: + with self._audio_interface() as interface: + src = BytesIO(blob) + dst = interface.player() + amodem.send.main(config=self.modem_config, src=src, dst=dst) + except Exception: + traceback.print_exc() + + print_msg('Sending:', repr(blob)) + blob = zlib.compress(blob) + + kbps = self.modem_config.modem_bps / 1e3 + msg = 'Sending to Audio MODEM ({0:.1f} kbps)...'.format(kbps) + return WaitingDialog(parent=parent, message=msg, run_task=sender_thread) + + def _recv(self, parent): + def receiver_thread(): + try: + with self._audio_interface() as interface: + src = interface.recorder() + dst = BytesIO() + amodem.recv.main(config=self.modem_config, src=src, dst=dst) + return dst.getvalue() + except Exception: + traceback.print_exc() + + def on_success(blob): + if blob: + blob = zlib.decompress(blob) + print_msg('Received:', repr(blob)) + parent.setText(blob) + + kbps = self.modem_config.modem_bps / 1e3 + msg = 'Receiving from Audio MODEM ({0:.1f} kbps)...'.format(kbps) + return WaitingDialog(parent=parent, message=msg, + run_task=receiver_thread, on_success=on_success) + + +def add_button(parent, icon_name): + audio_button = QToolButton(parent) + audio_button.setIcon(QIcon(icon_name)) + audio_button.setStyleSheet("QToolButton { border: none; padding: 0px; }") + audio_button.setVisible(True) + + parent_resizeEvent = parent.resizeEvent + + def resizeEvent(e): + result = parent_resizeEvent(e) + qr_button = parent.button + left = qr_button.geometry().left() - audio_button.sizeHint().width() + top = qr_button.geometry().top() + audio_button.move(left, top) + return result + + parent.resizeEvent = resizeEvent + return audio_button