Merge branch 'amodem-plugin' of https://github.com/romanz/electrum into romanz-amodem-plugin
This commit is contained in:
commit
ebdca0b940
|
@ -1,4 +1,5 @@
|
||||||
from electrum.i18n import _
|
from electrum.i18n import _
|
||||||
|
from electrum.plugins import run_hook
|
||||||
from PyQt4.QtGui import *
|
from PyQt4.QtGui import *
|
||||||
from PyQt4.QtCore import *
|
from PyQt4.QtCore import *
|
||||||
|
|
||||||
|
@ -25,6 +26,7 @@ class ShowQRTextEdit(QRTextEdit):
|
||||||
super(ShowQRTextEdit, self).__init__(text)
|
super(ShowQRTextEdit, self).__init__(text)
|
||||||
self.setReadOnly(1)
|
self.setReadOnly(1)
|
||||||
self.button.clicked.connect(self.qr_show)
|
self.button.clicked.connect(self.qr_show)
|
||||||
|
run_hook('show_text_edit', self)
|
||||||
|
|
||||||
def qr_show(self):
|
def qr_show(self):
|
||||||
from qrcodewidget import QRDialog
|
from qrcodewidget import QRDialog
|
||||||
|
@ -49,6 +51,7 @@ class ScanQRTextEdit(QRTextEdit):
|
||||||
if win:
|
if win:
|
||||||
assert hasattr(win,"config"), "You must pass a window with access to the config to ScanQRTextEdit constructor."
|
assert hasattr(win,"config"), "You must pass a window with access to the config to ScanQRTextEdit constructor."
|
||||||
self.button.clicked.connect(self.qr_input)
|
self.button.clicked.connect(self.qr_input)
|
||||||
|
run_hook('scan_text_edit', self)
|
||||||
|
|
||||||
|
|
||||||
def qr_input(self):
|
def qr_input(self):
|
||||||
|
|
|
@ -28,6 +28,8 @@
|
||||||
<file>icons/network.png</file>
|
<file>icons/network.png</file>
|
||||||
<file>icons/dark_background.png</file>
|
<file>icons/dark_background.png</file>
|
||||||
<file>icons/qrcode.png</file>
|
<file>icons/qrcode.png</file>
|
||||||
|
<file>icons/microphone.png</file>
|
||||||
|
<file>icons/speaker.png</file>
|
||||||
<file>icons/trustedcoin.png</file>
|
<file>icons/trustedcoin.png</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
</RCC>
|
</RCC>
|
||||||
|
|
Binary file not shown.
After Width: | Height: | Size: 199 B |
Binary file not shown.
After Width: | Height: | Size: 392 B |
|
@ -0,0 +1,173 @@
|
||||||
|
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 enabled.')
|
||||||
|
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
|
||||||
|
|
||||||
|
is_enabled = is_available
|
||||||
|
|
||||||
|
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
|
Loading…
Reference in New Issue