from PyQt4.QtGui import * from PyQt4.QtCore import * import PyQt4.QtCore as QtCore from electrum.i18n import _ from electrum import Wallet, mnemonic from seed_dialog import SeedDialog from network_dialog import NetworkDialog from qt_util import * from amountedit import AmountEdit import sys import threading class InstallWizard(QDialog): def __init__(self, config, network, storage): QDialog.__init__(self) self.config = config self.network = network self.storage = storage self.setMinimumSize(575, 400) self.setWindowTitle('Electrum') self.connect(self, QtCore.SIGNAL('accept'), self.accept) def restore_or_create(self): grid = QGridLayout() grid.setSpacing(5) msg = _("Wallet file not found.")+"\n"+_("Do you want to create a new wallet, or to restore an existing one?") label = QLabel(msg) label.setWordWrap(True) grid.addWidget(label, 0, 0) gb = QGroupBox() b1 = QRadioButton(gb) b1.setText(_("Create new wallet")) b1.setChecked(True) b2 = QRadioButton(gb) b2.setText(_("Restore wallet from seed")) b3 = QRadioButton(gb) b3.setText(_("Restore wallet from master public key")) grid.addWidget(b1,1,0) grid.addWidget(b2,2,0) grid.addWidget(b3,3,0) vbox = QVBoxLayout(self) vbox.addLayout(grid) vbox.addStretch(1) vbox.addLayout(ok_cancel_buttons(self, _('Next'))) if not self.exec_(): return if b1.isChecked(): answer = 'create' elif b2.isChecked(): answer = 'restore' else: answer = 'watching' return answer def verify_seed(self, wallet): r = self.seed_dialog(False) if not r: return if r != wallet.seed: QMessageBox.warning(None, _('Error'), 'incorrect seed', 'OK') return False else: return True def seed_dialog(self, is_restore=True): if self.layout(): QWidget().setLayout(self.layout()) vbox = QVBoxLayout(self) if is_restore: msg = _("Please enter your wallet seed." + ' ') msg += _("Your seed can be entered as a sequence of words, or as a hexadecimal string."+ '\n') else: msg = _("Your seed is important!") \ + "\n" + _("To make sure that you have properly saved your seed, please retype it here." + ' ') logo = QLabel() logo.setPixmap(QPixmap(":icons/seed.png").scaledToWidth(56)) logo.setMaximumWidth(60) label = QLabel(msg) label.setWordWrap(True) seed_e = QTextEdit() seed_e.setMaximumHeight(100) vbox.addWidget(label) grid = QGridLayout() grid.addWidget(logo, 0, 0) grid.addWidget(seed_e, 0, 1) vbox.addLayout(grid) vbox.addStretch(1) vbox.addLayout(ok_cancel_buttons(self, _('Next'))) if not self.exec_(): return try: seed = str(seed_e.toPlainText()) seed.decode('hex') except: try: seed = mnemonic.mn_decode( seed.split() ) except: QMessageBox.warning(None, _('Error'), _('I cannot decode this'), _('OK')) return if not seed: QMessageBox.warning(None, _('Error'), _('No seed'), _('OK')) return return seed def waiting_dialog(self, task, msg= _("Please wait...")): def target(): task() self.emit(QtCore.SIGNAL('accept')) if self.layout(): QWidget().setLayout(self.layout()) vbox = QVBoxLayout(self) self.waiting_label = QLabel(msg) vbox.addWidget(self.waiting_label) self.show() t = threading.Thread(target = target) t.start() self.exec_() def mpk_dialog(self): if self.layout(): QWidget().setLayout(self.layout()) vbox = QVBoxLayout(self) msg = _("Please enter your master public key.") label=QLabel(msg) label.setWordWrap(True) vbox.addWidget(label) mpk_e = QTextEdit() mpk_e.setMaximumHeight(100) vbox.addWidget(mpk_e) grid = QGridLayout() grid.setSpacing(8) vbox.addLayout(grid) vbox.addStretch(1) vbox.addLayout(ok_cancel_buttons(self, _('Next'))) if not self.exec_(): return mpk = str(mpk_e.toPlainText()) return mpk def network_dialog(self): if self.layout(): QWidget().setLayout(self.layout()) grid = QGridLayout() grid.setSpacing(5) label = QLabel(_("Network") + ":") grid.addWidget(label, 0, 0) gb = QGroupBox() b1 = QRadioButton(gb) b1.setText(_("Auto connect")) b1.setChecked(True) b2 = QRadioButton(gb) b2.setText(_("Select server manually")) b3 = QRadioButton(gb) b3.setText(_("Stay offline")) grid.addWidget(b1,1,0) grid.addWidget(b2,2,0) grid.addWidget(b3,3,0) vbox = QVBoxLayout(self) vbox.addLayout(grid) vbox.addStretch(1) vbox.addLayout(ok_cancel_buttons(self, _('Next'))) if not self.exec_(): return if b2.isChecked(): return NetworkDialog(self.network, self.config, None).do_exec() elif b1.isChecked(): self.config.set_key('auto_cycle', True, True) return else: self.config.set_key("server", None, True) self.config.set_key('auto_cycle', False, True) return def show_seed(self, wallet): from seed_dialog import make_seed_dialog if self.layout(): QWidget().setLayout(self.layout()) make_seed_dialog(self, wallet.seed, wallet.imported_keys) self.exec_() def password_dialog(self, wallet): msg = _("Please choose a password to encrypt your wallet keys.")+'\n'\ +_("Leave these fields empty if you want to disable encryption.") from password_dialog import make_password_dialog, run_password_dialog if self.layout(): QWidget().setLayout(self.layout()) make_password_dialog(self, wallet, msg) run_password_dialog(self, wallet, self) def run(self): action = self.restore_or_create() if not action: exit() wallet = Wallet(self.storage) gap = self.config.get('gap_limit', 5) if gap != 5: wallet.gap_limit = gap wallet.storage.put('gap_limit', gap, True) if action == 'create': wallet.init_seed(None) self.show_seed(wallet) if self.verify_seed(wallet): def create(): wallet.save_seed() wallet.create_accounts() wallet.synchronize() # generate first addresses offline self.waiting_dialog(create) else: return elif action == 'restore': # ask for seed and gap. seed = self.seed_dialog() if not seed: return wallet.init_seed(str(seed)) wallet.save_seed() elif action == 'watching': # ask for seed and gap. mpk = self.mpk_dialog() if not mpk: return wallet.seed = '' print eval(mpk) try: c0, K0 = eval(mpk) except: QMessageBox.warning(None, _('Error'), _('error'), _('OK')) return wallet.create_watching_only_wallet(c0,K0) else: raise #if not self.config.get('server'): self.network_dialog() # start wallet threads wallet.start_threads(self.network) if action == 'restore': def wait_for_wallet(): wallet.set_up_to_date(False) while not wallet.is_up_to_date(): msg = "%s\n%s %d\n%s %.1f"%(_("Please wait..."),_("Addresses generated:"),len(wallet.addresses(True)),_("Kilobytes received:"), self.network.interface.bytes_received/1024.) self.waiting_label.setText(msg) time.sleep(0.1) def wait_for_network(): while not self.network.interface.is_connected: msg = "%s \n" % (_("Connecting...")) self.waiting_label.setText(msg) time.sleep(0.1) def restore(): # wait until we are connected, because the user might have selected another server wait_for_network() # try to restore old account wallet.create_old_account() wait_for_wallet() if wallet.is_found(): wallet.seed_version = 4 wallet.storage.put('seed_version', wallet.seed_version, True) else: wallet.accounts.pop(0) wallet.create_accounts() wait_for_wallet() self.waiting_dialog(restore) if wallet.is_found(): QMessageBox.information(None, _('Information'), _("Recovery successful"), _('OK')) else: QMessageBox.information(None, _('Information'), _("No transactions found for this seed"), _('OK')) wallet.fill_addressbook() self.password_dialog(wallet) return wallet