1 from PyQt4.QtGui import *
2 from PyQt4.QtCore import *
3 import PyQt4.QtCore as QtCore
6 from electrum import Wallet, mnemonic
8 from seed_dialog import SeedDialog
9 from network_dialog import NetworkDialog
11 from amountedit import AmountEdit
15 class InstallWizard(QDialog):
17 def __init__(self, config, network, storage):
18 QDialog.__init__(self)
20 self.network = network
21 self.storage = storage
24 def restore_or_create(self):
32 msg = _("Wallet file not found.")+"\n"+_("Do you want to create a new wallet, or to restore an existing one?")
34 label.setWordWrap(True)
35 grid.addWidget(label, 0, 0)
40 b1.setText(_("Create new wallet"))
44 b2.setText(_("Restore wallet from seed"))
47 b3.setText(_("Restore wallet from master public key"))
49 grid.addWidget(b1,1,0)
50 grid.addWidget(b2,2,0)
51 grid.addWidget(b3,3,0)
55 vbox.addLayout(ok_cancel_buttons(d, _('Next')))
70 def verify_seed(self, wallet):
71 r = self.seed_dialog(False)
76 QMessageBox.warning(None, _('Error'), 'incorrect seed', 'OK')
82 def seed_dialog(self, is_restore=True):
88 msg = _("Please enter your wallet seed (or your master public key if you want to create a watching-only wallet)." + ' ')
90 msg = _("Your seed is important! To make sure that you have properly saved your seed, please type it here." + ' ')
92 msg += _("Your seed can be entered as a sequence of words, or as a hexadecimal string."+ '\n')
95 label.setWordWrap(True)
99 seed_e.setMaximumHeight(100)
100 vbox.addWidget(seed_e)
105 gap_e = AmountEdit(None, True)
107 grid.addWidget(QLabel(_('Gap limit')), 2, 0)
108 grid.addWidget(gap_e, 2, 1)
109 grid.addWidget(HelpButton(_('Keep the default value unless you modified this parameter in your wallet.')), 2, 3)
112 vbox.addLayout(ok_cancel_buttons(d, _('Next')))
115 if not d.exec_(): return
118 seed = str(seed_e.toPlainText())
122 seed = mnemonic.mn_decode( seed.split() )
124 QMessageBox.warning(None, _('Error'), _('I cannot decode this'), _('OK'))
128 QMessageBox.warning(None, _('Error'), _('No seed'), _('OK'))
135 gap = int(unicode(gap_e.text()))
137 QMessageBox.warning(None, _('Error'), 'error', 'OK')
142 def mpk_dialog(self):
147 msg = _("Please enter your master public key.")
150 label.setWordWrap(True)
151 vbox.addWidget(label)
154 mpk_e.setMaximumHeight(100)
155 vbox.addWidget(mpk_e)
159 gap_e = AmountEdit(None, True)
161 grid.addWidget(QLabel(_('Gap limit')), 2, 0)
162 grid.addWidget(gap_e, 2, 1)
163 grid.addWidget(HelpButton(_('Keep the default value unless you modified this parameter in your wallet.')), 2, 3)
166 vbox.addLayout(ok_cancel_buttons(d, _('Next')))
169 if not d.exec_(): return
171 mpk = str(mpk_e.toPlainText())
174 gap = int(unicode(gap_e.text()))
176 QMessageBox.warning(None, _('Error'), 'error', 'OK')
182 def network_dialog(self):
190 label = QLabel(_("Network") + ":")
191 grid.addWidget(label, 0, 0)
195 b1 = QRadioButton(gb)
196 b1.setText(_("Auto connect"))
199 b2 = QRadioButton(gb)
200 b2.setText(_("Select server manually"))
202 b3 = QRadioButton(gb)
203 b3.setText(_("Stay offline"))
205 grid.addWidget(b1,1,0)
206 grid.addWidget(b2,2,0)
207 grid.addWidget(b3,3,0)
211 vbox.addLayout(ok_cancel_buttons(d, _('Next')))
218 return NetworkDialog(self.network, self.config, None).do_exec()
221 self.config.set_key('auto_cycle', True, True)
225 self.config.set_key("server", None, True)
226 self.config.set_key('auto_cycle', False, True)
231 def show_seed(self, wallet):
233 d.show_seed(wallet.seed, wallet.imported_keys)
236 def password_dialog(self, wallet):
237 from password_dialog import PasswordDialog
238 d = PasswordDialog(wallet)
242 def restore_wallet(self, wallet):
244 # wait until we are connected, because the user might have selected another server
245 if not wallet.interface.is_connected:
246 waiting = lambda: False if wallet.interface.is_connected else "%s \n" % (_("Connecting..."))
247 waiting_dialog(waiting)
249 waiting = lambda: False if wallet.is_up_to_date() else "%s\n%s %d\n%s %.1f"\
250 %(_("Please wait..."),_("Addresses generated:"),len(wallet.addresses(True)),_("Kilobytes received:"), wallet.interface.bytes_received/1024.)
252 # try to restore old account
253 wallet.create_old_account()
254 wallet.set_up_to_date(False)
255 wallet.interface.poke('synchronizer')
256 waiting_dialog(waiting)
258 if wallet.is_found():
259 wallet.seed_version = 4
260 wallet.storage.put('seed_version', wallet.seed_version, True)
262 wallet.accounts.pop(0)
263 wallet.create_accounts()
264 wallet.set_up_to_date(False)
265 wallet.interface.poke('synchronizer')
266 waiting_dialog(waiting)
268 if wallet.is_found():
269 QMessageBox.information(None, _('Information'), _("Recovery successful"), _('OK'))
271 QMessageBox.information(None, _('Information'), _("No transactions found for this seed"), _('OK'))
278 action = self.restore_or_create()
279 if not action: exit()
281 wallet = Wallet(self.storage)
283 if action == 'create':
284 wallet.init_seed(None)
285 self.show_seed(wallet)
286 if self.verify_seed(wallet):
288 wallet.create_accounts()
289 # generate first addresses offline
294 elif action == 'restore':
295 # ask for seed and gap.
296 sg = self.seed_dialog()
302 wallet.gap_limit = gap
303 wallet.init_seed(str(seed))
306 elif action == 'watching':
307 # ask for seed and gap.
308 sg = self.mpk_dialog()
314 wallet.gap_limit = gap
321 QMessageBox.warning(None, _('Error'), _('error'), _('OK'))
323 wallet.create_watching_only_wallet(c0,K0)
328 #if not self.config.get('server'):
329 self.network_dialog()
331 # start wallet threads
332 wallet.start_threads(self.network)
334 if action == 'restore':
336 keep_it = self.restore_wallet(wallet)
337 wallet.fill_addressbook()
340 traceback.print_exc(file=sys.stdout)
343 if not keep_it: return
345 self.password_dialog(wallet)