#!/usr/bin/env python # # Electrum - lightweight Bitcoin client # Copyright (C) 2011 thomasv@gitorious # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . import android from interface import WalletSynchronizer from wallet import Wallet from wallet import format_satoshis from decimal import Decimal import mnemonic import datetime, re def modal_dialog(title, msg = None): droid.dialogCreateAlert(title,msg) droid.dialogSetPositiveButtonText('OK') droid.dialogShow() droid.dialogGetResponse() droid.dialogDismiss() def modal_input(title, msg, value = None, etype=None): droid.dialogCreateInput(title, msg, value, etype) droid.dialogSetPositiveButtonText('OK') droid.dialogSetNegativeButtonText('Cancel') droid.dialogShow() response = droid.dialogGetResponse().result droid.dialogDismiss() if response.get('which') == 'positive': return response.get('value') def modal_question(q, msg, pos_text = 'OK', neg_text = 'Cancel'): droid.dialogCreateAlert(q, msg) droid.dialogSetPositiveButtonText(pos_text) droid.dialogSetNegativeButtonText(neg_text) droid.dialogShow() response = droid.dialogGetResponse().result droid.dialogDismiss() return response.get('which') == 'positive' def edit_label(addr): v = modal_input('Edit label',None,wallet.labels.get(addr)) if v is not None: if v: wallet.labels[addr] = v else: if addr in wallet.labels.keys(): wallet.labels.pop(addr) wallet.update_tx_history() wallet.save() droid.fullSetProperty("labelTextView", "text", v) def select_from_contacts(): title = 'Contacts:' droid.dialogCreateAlert(title) l = [] for i in range(len(wallet.addressbook)): addr = wallet.addressbook[i] label = wallet.labels.get(addr,addr) l.append( label ) droid.dialogSetItems(l) droid.dialogSetPositiveButtonText('New contact') droid.dialogShow() response = droid.dialogGetResponse().result droid.dialogDismiss() if response.get('which') == 'positive': return 'newcontact' result = response.get('item') print result if result is not None: addr = wallet.addressbook[result] return addr def select_from_addresses(): droid.dialogCreateAlert("Addresses:") l = [] for i in range(len(wallet.addresses)): addr = wallet.addresses[i] label = wallet.labels.get(addr,addr) l.append( label ) droid.dialogSetItems(l) droid.dialogShow() response = droid.dialogGetResponse() result = response.result.get('item') droid.dialogDismiss() if result is not None: addr = wallet.addresses[result] return addr def protocol_name(p): if p == 't': return 'TCP/stratum' if p == 'h': return 'HTTP/Stratum' if p == 'n': return 'TCP/native' def protocol_dialog(host, protocol, z): droid.dialogCreateAlert('Protocol',host) if z: protocols = z.keys() else: protocols = ['t','h','n'] l = [] current = protocols.index(protocol) for p in protocols: l.append(protocol_name(p)) droid.dialogSetSingleChoiceItems(l, current) droid.dialogSetPositiveButtonText('OK') droid.dialogSetNegativeButtonText('Cancel') droid.dialogShow() response = droid.dialogGetResponse().result if not response: return if response.get('which') == 'positive': response = droid.dialogGetSelectedItems().result[0] droid.dialogDismiss() p = protocols[response] port = z[p] return host + ':' + port + ':' + p def make_layout(s, scrollable = False): content = """ %s """%s if scrollable: content = """ %s """%content return """ %s """%content def main_layout(): return make_layout(""" %s """%get_history_layout(15),True) def qr_layout(addr): return make_layout(""" """%(addr,wallet.labels.get(addr,'')), True) payto_layout = make_layout(""" """,False) settings_layout = make_layout(""" """) def get_history_values(n): values = [] h = wallet.get_tx_history() length = min(n, len(h)) for i in range(length): line = h[-i-1] v = line['value'] try: dt = datetime.datetime.fromtimestamp( line['timestamp'] ) if dt.date() == dt.today().date(): time_str = str( dt.time() ) else: time_str = str( dt.date() ) conf = 'v' except: print line['timestamp'] time_str = 'pending' conf = 'o' tx_hash = line['tx_hash'] label = wallet.labels.get(tx_hash) is_default_label = (label == '') or (label is None) if is_default_label: label = line['default_label'] values.append((conf, ' ' + time_str, ' ' + format_satoshis(v,True), ' ' + label )) return values def get_history_layout(n): rows = "" i = 0 values = get_history_values(n) for v in values: a,b,c,d = v color = "#ff00ff00" if a == 'v' else "#ffff0000" rows += """ """%(i,a,color,i,b,i,c,i,d) i += 1 output = """ %s """% rows return output def set_history_layout(n): values = get_history_values(n) i = 0 for v in values: a,b,c,d = v droid.fullSetProperty("hl_%d_col1"%i,"text", a) if a == 'v': droid.fullSetProperty("hl_%d_col1"%i, "textColor","#ff00ff00") else: droid.fullSetProperty("hl_%d_col1"%i, "textColor","#ffff0000") droid.fullSetProperty("hl_%d_col2"%i,"text", b) droid.fullSetProperty("hl_%d_col3"%i,"text", c) droid.fullSetProperty("hl_%d_col4"%i,"text", d) i += 1 status_text = '' def update_layout(): global status_text if not wallet.interface.is_connected: text = "Not connected..." elif wallet.blocks == 0: text = "Server not ready" elif not wallet.up_to_date: text = "Synchronizing..." else: c, u = wallet.get_balance() text = "Balance:"+format_satoshis(c) if u : text += ' [' + format_satoshis(u,True).strip() + ']' # vibrate if status changed if text != status_text: if status_text and wallet.interface.is_connected and wallet.up_to_date: droid.vibrate() status_text = text droid.fullSetProperty("balanceTextView", "text", status_text) if wallet.up_to_date: set_history_layout(15) def pay_to(recipient, amount, fee, label): if wallet.use_encryption: password = droid.dialogGetPassword('Password').result if not password: return else: password = None droid.dialogCreateSpinnerProgress("Electrum", "signing transaction...") droid.dialogShow() try: tx = wallet.mktx( recipient, amount, label, password, fee) except BaseException, e: modal_dialog('error', e.message) droid.dialogDismiss() return droid.dialogDismiss() r, h = wallet.sendtx( tx ) if r: modal_dialog('Payment sent', h) return True else: modal_dialog('Error', h) def recover(): droid.dialogCreateAlert("Wallet not found","Do you want to create a new wallet, or restore an existing one?") droid.dialogSetPositiveButtonText('Create') droid.dialogSetNeutralButtonText('Restore') droid.dialogSetNegativeButtonText('Cancel') droid.dialogShow() response = droid.dialogGetResponse().result droid.dialogDismiss() if response.get('which') == 'negative': exit(1) is_recovery = response.get('which') == 'neutral' if not is_recovery: wallet.new_seed(None) else: if modal_question("Input method",None,'QR Code', 'mnemonic'): code = droid.scanBarcode() r = code.result if r: seed = r['extras']['SCAN_RESULT'] else: exit(1) else: m = modal_input('Mnemonic','please enter your code') try: seed = mnemonic.mn_decode(m.split(' ')) except: modal_dialog('error: could not decode this seed') exit(1) wallet.seed = str(seed) modal_dialog('Your seed is:', wallet.seed) modal_dialog('Mnemonic code:', ' '.join(mnemonic.mn_encode(wallet.seed)) ) msg = "recovering wallet..." if is_recovery else "creating wallet..." droid.dialogCreateSpinnerProgress("Electrum", msg) droid.dialogShow() wallet.init_mpk( wallet.seed ) WalletSynchronizer(wallet,True).start() wallet.update() droid.dialogDismiss() droid.vibrate() if is_recovery: if wallet.is_found(): wallet.update_tx_history() wallet.fill_addressbook() modal_dialog("recovery successful") else: if not modal_question("no transactions found for this seed","do you want to keep this wallet?"): exit(1) change_password_dialog() wallet.save() def make_new_contact(): code = droid.scanBarcode() r = code.result if r: address = r['extras']['SCAN_RESULT'] if address: if wallet.is_valid(address): if modal_question('Add to contacts?', address): wallet.addressbook.append(address) wallet.save() else: modal_dialog('Invalid address', address) do_refresh = False def update_callback(): global do_refresh print "gui callback", wallet.interface.is_connected, wallet.up_to_date do_refresh = True droid.eventPost("refresh",'z') def main_loop(): global do_refresh update_layout() out = None quitting = False while out is None: event = droid.eventWait(1000).result if event is None: if do_refresh: update_layout() do_refresh = False continue print "got event in main loop", repr(event) if event == 'OK': continue if event is None: continue #if event["name"]=="refresh": # request 2 taps before we exit if event["name"]=="key": if event["data"]["key"] == '4': if quitting: out = 'quit' else: quitting = True else: quitting = False if event["name"]=="click": id=event["data"]["id"] elif event["name"]=="settings": out = 'settings' elif event["name"] in menu_commands: out = event["name"] if out == 'contacts': global contact_addr contact_addr = select_from_contacts() if contact_addr == 'newcontact': make_new_contact() contact_addr = None if not contact_addr: out = None elif out == "receive": global receive_addr receive_addr = select_from_addresses() if receive_addr: amount = modal_input('Amount', 'Amount you want receive. ', '', "numberDecimal") if amount: receive_addr = 'bitcoin:%s?amount=%s'%(receive_addr, amount) if not receive_addr: out = None return out def payto_loop(): global recipient if recipient: droid.fullSetProperty("recipient","text",recipient) recipient = None out = None while out is None: event = droid.eventWait().result print "got event in payto loop", event if event["name"] == "click": id = event["data"]["id"] if id=="buttonPay": droid.fullQuery() recipient = droid.fullQueryDetail("recipient").result.get('text') label = droid.fullQueryDetail("label").result.get('text') amount = droid.fullQueryDetail('amount').result.get('text') if not wallet.is_valid(recipient): modal_dialog('Error','Invalid Bitcoin address') continue try: amount = int( 100000000 * Decimal(amount) ) except: modal_dialog('Error','Invalid amount') continue result = pay_to(recipient, amount, wallet.fee, label) if result: out = 'main' elif id=="buttonContacts": addr = select_from_contacts() droid.fullSetProperty("recipient","text",addr) elif id=="buttonQR": code = droid.scanBarcode() r = code.result if r: data = r['extras']['SCAN_RESULT'] if data: if re.match('^bitcoin:', data): payto, amount, label, _, _, _, _ = wallet.parse_url(data, None, None) droid.fullSetProperty("recipient", "text",payto) droid.fullSetProperty("amount", "text", amount) droid.fullSetProperty("label", "text", label) else: droid.fullSetProperty("recipient", "text", data) elif event["name"] in menu_commands: out = event["name"] elif event["name"]=="key": if event["data"]["key"] == '4': out = 'main' #elif event["name"]=="screen": # if event["data"]=="destroy": # out = 'main' return out receive_addr = '' contact_addr = '' recipient = '' def receive_loop(): out = None while out is None: event = droid.eventWait().result print "got event", event if event["name"]=="key": if event["data"]["key"] == '4': out = 'main' elif event["name"]=="clipboard": droid.setClipboard(receive_addr) modal_dialog('Address copied to clipboard',receive_addr) elif event["name"]=="edit": edit_label(receive_addr) return out def contacts_loop(): global recipient out = None while out is None: event = droid.eventWait().result print "got event", event if event["name"]=="key": if event["data"]["key"] == '4': out = 'main' elif event["name"]=="clipboard": droid.setClipboard(contact_addr) modal_dialog('Address copied to clipboard',contact_addr) elif event["name"]=="edit": edit_label(contact_addr) elif event["name"]=="paytocontact": recipient = contact_addr out = 'send' elif event["name"]=="deletecontact": if modal_question('delete contact', contact_addr): out = 'main' return out def server_dialog(plist): droid.dialogCreateAlert("Public servers") droid.dialogSetItems( plist.keys() ) droid.dialogSetPositiveButtonText('Private server') droid.dialogShow() response = droid.dialogGetResponse().result droid.dialogDismiss() if response.get('which') == 'positive': return modal_input('Private server', None) i = response.get('item') if i is not None: response = plist.keys()[i] return response def seed_dialog(): if wallet.use_encryption: password = droid.dialogGetPassword('Seed').result if not password: return else: password = None try: seed = wallet.pw_decode( wallet.seed, password) except: modal_dialog('error','incorrect password') return modal_dialog('Your seed is',seed) modal_dialog('Mnemonic code:', ' '.join(mnemonic.mn_encode(seed)) ) def change_password_dialog(): if wallet.use_encryption: password = droid.dialogGetPassword('Your wallet is encrypted').result if password is None: return else: password = None try: seed = wallet.pw_decode( wallet.seed, password) except: modal_dialog('error','incorrect password') return new_password = droid.dialogGetPassword('Choose a password').result if new_password == None: return if new_password != '': password2 = droid.dialogGetPassword('Confirm new password').result if new_password != password2: modal_dialog('error','passwords do not match') return wallet.update_password(seed, new_password) if new_password: modal_dialog('Password updated','your wallet is encrypted') else: modal_dialog('No password','your wallet is not encrypted') return True def settings_loop(): def set_listview(): server, port, p = wallet.server.split(':') fee = str( Decimal( wallet.fee)/100000000 ) is_encrypted = 'yes' if wallet.use_encryption else 'no' protocol = protocol_name(p) droid.fullShow(settings_layout) droid.fullSetList("myListView",['Server: ' + server, 'Protocol: '+ protocol, 'Port: '+port, 'Transaction fee: '+fee, 'Password: '+is_encrypted, 'Seed']) set_listview() out = None while out is None: event = droid.eventWait().result print "got event", event if event == 'OK': continue if not event: continue plist = {} for item in wallet.interface.servers: host, pp = item z = {} for item2 in pp: protocol, port = item2 z[protocol] = port plist[host] = z if event["name"] == "itemclick": pos = event["data"]["position"] host, port, protocol = wallet.server.split(':') if pos == "0": #server host = server_dialog(plist) if host: p = plist[host] port = p['t'] srv = host + ':' + port + ':t' try: wallet.set_server(srv) except: modal_dialog('error','invalid server') set_listview() elif pos == "1": #protocol if host in plist: srv = protocol_dialog(host, protocol, plist[host]) if srv: try: wallet.set_server(srv) except: modal_dialog('error','invalid server') set_listview() elif pos == "2": #port a_port = modal_input('Port number', 'If you use a public server, this field is set automatically when you set the protocol', port, "number") if a_port: if a_port != port: srv = host + ':' + a_port + ':'+ protocol try: wallet.set_server(srv) except: modal_dialog('error','invalid port number') set_listview() elif pos == "3": #fee fee = modal_input('Transaction fee', 'The fee will be this amount multiplied by the number of inputs in your transaction. ', str( Decimal( wallet.fee)/100000000 ), "numberDecimal") if fee: try: fee = int( 100000000 * Decimal(fee) ) except: modal_dialog('error','invalid fee value') if wallet.fee != fee: wallet.fee = fee wallet.save() set_listview() elif pos == "4": if change_password_dialog(): set_listview() elif pos == "5": seed_dialog() elif event["name"] in menu_commands: out = event["name"] elif event["name"] == 'cancel': out = 'main' elif event["name"] == "key": if event["data"]["key"] == '4': out = 'main' return out menu_commands = ["send", "receive", "settings", "contacts", "main"] droid = android.Android() wallet = Wallet(update_callback) wallet.set_path("/sdcard/electrum.dat") wallet.read() if not wallet.file_exists: recover() else: WalletSynchronizer(wallet,True).start() s = 'main' def add_menu(s): droid.clearOptionsMenu() if s == 'main': droid.addOptionsMenuItem("Send","send",None,"") droid.addOptionsMenuItem("Receive","receive",None,"") droid.addOptionsMenuItem("Contacts","contacts",None,"") droid.addOptionsMenuItem("Settings","settings",None,"") elif s == 'receive': droid.addOptionsMenuItem("Copy","clipboard",None,"") droid.addOptionsMenuItem("Label","edit",None,"") elif s == 'contacts': droid.addOptionsMenuItem("Copy","clipboard",None,"") droid.addOptionsMenuItem("Label","edit",None,"") droid.addOptionsMenuItem("Pay to","paytocontact",None,"") #droid.addOptionsMenuItem("Delete","deletecontact",None,"") def make_bitmap(addr): # fixme: this is highly inefficient droid.dialogCreateSpinnerProgress("please wait") droid.dialogShow() try: import pyqrnative, bmp qr = pyqrnative.QRCode(4, pyqrnative.QRErrorCorrectLevel.L) qr.addData(addr) qr.make() k = qr.getModuleCount() assert k == 33 bmp.save_qrcode(qr,"/sdcard/sl4a/qrcode.bmp") finally: droid.dialogDismiss() while True: add_menu(s) if s == 'main': droid.fullShow(main_layout()) s = main_loop() #droid.fullDismiss() elif s == 'send': droid.fullShow(payto_layout) s = payto_loop() #droid.fullDismiss() elif s == 'receive': make_bitmap(receive_addr) droid.fullShow(qr_layout(receive_addr)) s = receive_loop() elif s == 'contacts': make_bitmap(contact_addr) droid.fullShow(qr_layout(contact_addr)) s = contacts_loop() elif s == 'settings': #droid.fullShow(settings_layout) s = settings_loop() #droid.fullDismiss() else: break droid.makeToast("Bye!")