Note: the option to load a backup from SD card when the device already has a seed has been removed. The device always returns an error when attempting this.
118 lines
3.8 KiB
Python
118 lines
3.8 KiB
Python
import threading
|
|
from functools import partial
|
|
from typing import TYPE_CHECKING
|
|
|
|
from PyQt5.QtCore import pyqtSignal
|
|
|
|
from electrum.i18n import _
|
|
from electrum.plugin import hook
|
|
from electrum.wallet import Standard_Wallet, Abstract_Wallet
|
|
from electrum.util import UserCancelled, UserFacingException
|
|
|
|
from electrum.plugins.hw_wallet.qt import QtHandlerBase, QtPluginBase
|
|
from electrum.plugins.hw_wallet.plugin import only_hook_if_libraries_available, OperationCancelled
|
|
|
|
from electrum.gui.qt.wizard.wallet import WCScriptAndDerivation, WCHWXPub, WCHWUnlock
|
|
|
|
from .digitalbitbox import DigitalBitboxPlugin, DeviceErased
|
|
|
|
if TYPE_CHECKING:
|
|
from electrum.gui.qt.wizard.wallet import QENewWalletWizard
|
|
|
|
|
|
class Plugin(DigitalBitboxPlugin, QtPluginBase):
|
|
icon_unpaired = "digitalbitbox_unpaired.png"
|
|
icon_paired = "digitalbitbox.png"
|
|
|
|
def create_handler(self, window):
|
|
return DigitalBitbox_Handler(window)
|
|
|
|
@only_hook_if_libraries_available
|
|
@hook
|
|
def receive_menu(self, menu, addrs, wallet: Abstract_Wallet):
|
|
if type(wallet) is not Standard_Wallet:
|
|
return
|
|
|
|
keystore = wallet.get_keystore()
|
|
if type(keystore) is not self.keystore_class:
|
|
return
|
|
|
|
if not self.is_mobile_paired():
|
|
return
|
|
|
|
if len(addrs) == 1:
|
|
addr = addrs[0]
|
|
if wallet.get_txin_type(addr) != 'p2pkh':
|
|
return
|
|
|
|
def show_address():
|
|
keystore.thread.add(partial(self.show_address, wallet, addr, keystore))
|
|
|
|
menu.addAction(_("Show on {}").format(self.device), show_address)
|
|
|
|
@hook
|
|
def init_wallet_wizard(self, wizard: 'QENewWalletWizard'):
|
|
self.extend_wizard(wizard)
|
|
|
|
# insert digitalbitbox pages in new wallet wizard
|
|
def extend_wizard(self, wizard: 'QENewWalletWizard'):
|
|
super().extend_wizard(wizard)
|
|
views = {
|
|
'dbitbox_start': {'gui': WCDigitalBitboxScriptAndDerivation},
|
|
'dbitbox_xpub': {'gui': WCHWXPub},
|
|
'dbitbox_unlock': {'gui': WCHWUnlock}
|
|
}
|
|
wizard.navmap_merge(views)
|
|
|
|
|
|
class DigitalBitbox_Handler(QtHandlerBase):
|
|
def __init__(self, win):
|
|
super(DigitalBitbox_Handler, self).__init__(win, 'Digital Bitbox')
|
|
|
|
|
|
class WCDigitalBitboxScriptAndDerivation(WCScriptAndDerivation):
|
|
requestRecheck = pyqtSignal()
|
|
|
|
def __init__(self, parent, wizard):
|
|
WCScriptAndDerivation.__init__(self, parent, wizard)
|
|
self._busy = True
|
|
self.title = ''
|
|
self.client = None
|
|
|
|
self.requestRecheck.connect(self.check_device)
|
|
|
|
def on_ready(self):
|
|
super().on_ready()
|
|
_name, _info = self.wizard_data['hardware_device']
|
|
plugin = self.wizard.plugins.get_plugin(_info.plugin_name)
|
|
|
|
device_id = _info.device.id_
|
|
self.client = self.wizard.plugins.device_manager.client_by_id(device_id, scan_now=False)
|
|
if not self.client.handler:
|
|
self.client.handler = plugin.create_handler(self.wizard)
|
|
self.client.setupRunning = True
|
|
self.check_device()
|
|
|
|
def check_device(self):
|
|
self.error = None
|
|
self.busy = True
|
|
|
|
def check_task():
|
|
try:
|
|
self.client.check_device_dialog()
|
|
self.title = _('Script type and Derivation path')
|
|
self.valid = True
|
|
except (UserCancelled, OperationCancelled):
|
|
self.error = _('Cancelled')
|
|
self.wizard.requestPrev.emit()
|
|
except DeviceErased:
|
|
self.error = _('Device erased')
|
|
self.requestRecheck.emit()
|
|
except UserFacingException as e:
|
|
self.error = str(e)
|
|
finally:
|
|
self.busy = False
|
|
|
|
t = threading.Thread(target=check_task, daemon=True)
|
|
t.start()
|