from PyQt4 . QtCore import *
from PyQt4 . QtGui import *
from decimal import Decimal as D
from util import appdata_dir , get_resource_path as rsrc
from i18n import _
import decimal
import exchange_rate
import os . path
import random
import re
import sys
import time
import wallet
import webbrowser
try :
import lib . gui_qt as gui_qt
except ImportError :
import electrum . gui_qt as gui_qt
bitcoin = lambda v : v * 100000000
def IconButton ( filename , parent = None ) :
pixmap = QPixmap ( filename )
icon = QIcon ( pixmap )
return QPushButton ( icon , " " , parent )
class Timer ( QThread ) :
def run ( self ) :
while True :
self . emit ( SIGNAL ( ' timersignal ' ) )
time . sleep ( 0.5 )
def resize_line_edit_width ( line_edit , text_input ) :
metrics = QFontMetrics ( qApp . font ( ) )
# Create an extra character to add some space on the end
text_input + = " A "
line_edit . setMinimumWidth ( metrics . width ( text_input ) )
def cd_data_dir ( ) :
assert sys . argv
prefix_path = os . path . dirname ( sys . argv [ 0 ] )
local_data = os . path . join ( prefix_path , " data " )
if os . path . exists ( os . path . join ( local_data , " style.css " ) ) :
data_dir = local_data
else :
data_dir = appdata_dir ( )
QDir . setCurrent ( data_dir )
class ElectrumGui :
def __init__ ( self , wallet ) :
self . wallet = wallet
self . app = QApplication ( sys . argv )
# Should probably not modify the current path but instead
# change the behaviour of rsrc(...)
self . old_path = QDir . currentPath ( )
cd_data_dir ( )
with open ( rsrc ( " style.css " ) ) as style_file :
self . app . setStyleSheet ( style_file . read ( ) )
def main ( self , url ) :
actuator = MiniActuator ( self . wallet )
self . mini = MiniWindow ( actuator , self . expand )
driver = MiniDriver ( self . wallet , self . mini )
# Reset path back to original value now that loading the GUI
# is completed.
QDir . setCurrent ( self . old_path )
if url :
self . set_url ( url )
timer = Timer ( )
timer . start ( )
self . expert = gui_qt . ElectrumWindow ( self . wallet )
self . expert . app = self . app
self . expert . connect_slots ( timer )
self . expert . update_wallet ( )
self . app . exec_ ( )
def expand ( self ) :
self . mini . hide ( )
self . expert . show ( )
def set_url ( self , url ) :
payto , amount , label , message , signature , identity , url = \
self . wallet . parse_url ( url , self . show_message , self . show_question )
self . mini . set_payment_fields ( payto , amount )
def show_message ( self , message ) :
QMessageBox . information ( self . mini , _ ( " Message " ) , message , _ ( " OK " ) )
def show_question ( self , message ) :
choice = QMessageBox . question ( self . mini , _ ( " Message " ) , message ,
QMessageBox . Yes | QMessageBox . No ,
QMessageBox . No )
return choice == QMessageBox . Yes
def restore_or_create ( self ) :
qt_gui_object = gui_qt . ElectrumGui ( self . wallet , self . app )
return qt_gui_object . restore_or_create ( )
class MiniWindow ( QDialog ) :
def __init__ ( self , actuator , expand_callback ) :
super ( MiniWindow , self ) . __init__ ( )
self . actuator = actuator
self . btc_balance = None
self . quote_currencies = [ " EUR " , " USD " , " GBP " ]
self . actuator . set_configured_currency ( self . set_quote_currency )
self . exchanger = exchange_rate . Exchanger ( self )
# Needed because price discovery is done in a different thread
# which needs to be sent back to this main one to update the GUI
self . connect ( self , SIGNAL ( " refresh_balance() " ) , self . refresh_balance )
self . balance_label = BalanceLabel ( self . change_quote_currency )
self . balance_label . setObjectName ( " balance_label " )
self . receive_button = QPushButton ( _ ( " &Receive " ) )
self . receive_button . setObjectName ( " receive_button " )
self . receive_button . setDefault ( True )
self . connect ( self . receive_button , SIGNAL ( " clicked() " ) , self . copy_address )
# Bitcoin address code
self . address_input = QLineEdit ( )
self . address_input . setPlaceholderText ( _ ( " Enter a Bitcoin address... " ) )
self . address_input . setObjectName ( " address_input " )
self . connect ( self . address_input , SIGNAL ( " textEdited(QString) " ) , self . address_field_changed )
resize_line_edit_width ( self . address_input , " 1BtaFUr3qVvAmwrsuDuu5zk6e4s2rxd2Gy " )
self . address_completions = QStringListModel ( )
address_completer = QCompleter ( self . address_input )
address_completer . setCaseSensitivity ( False )
address_completer . setModel ( self . address_completions )
self . address_input . setCompleter ( address_completer )
address_layout = QHBoxLayout ( )
address_layout . addWidget ( self . address_input )
self . amount_input = QLineEdit ( )
self . amount_input . setPlaceholderText ( _ ( " ... and amount " ) )
self . amount_input . setObjectName ( " amount_input " )
# This is changed according to the user's displayed balance
self . amount_validator = QDoubleValidator ( self . amount_input )
self . amount_validator . setNotation ( QDoubleValidator . StandardNotation )
self . amount_validator . setDecimals ( 8 )
self . amount_input . setValidator ( self . amount_validator )
# This removes the very ugly OSX highlighting, please leave this in :D
self . address_input . setAttribute ( Qt . WA_MacShowFocusRect , 0 )
self . amount_input . setAttribute ( Qt . WA_MacShowFocusRect , 0 )
self . connect ( self . amount_input , SIGNAL ( " textChanged(QString) " ) ,
self . amount_input_changed )
amount_layout = QHBoxLayout ( )
amount_layout . addWidget ( self . amount_input )
amount_layout . addStretch ( )
self . send_button = QPushButton ( _ ( " &Send " ) )
self . send_button . setObjectName ( " send_button " )
self . send_button . setDisabled ( True ) ;
self . connect ( self . send_button , SIGNAL ( " clicked() " ) , self . send )
main_layout = QGridLayout ( self )
main_layout . addWidget ( self . balance_label , 0 , 0 )
main_layout . addWidget ( self . receive_button , 0 , 1 )
main_layout . addWidget ( self . address_input , 1 , 0 , 1 , - 1 )
main_layout . addLayout ( amount_layout , 2 , 0 )
main_layout . addWidget ( self . send_button , 2 , 1 )
menubar = QMenuBar ( )
file_menu = menubar . addMenu ( _ ( " &File " ) )
file_menu . addAction ( _ ( " Open " ) )
view_menu = menubar . addMenu ( _ ( " &View " ) )
view_menu . addMenu ( _ ( " &Themes " ) )
view_menu . addAction ( _ ( " Show History " ) )
settings_menu = menubar . addMenu ( _ ( " &Settings " ) )
expert_gui = settings_menu . addAction ( _ ( " &Switch to expert GUI " ) )
self . connect ( expert_gui , SIGNAL ( " triggered() " ) , expand_callback )
menubar . addMenu ( _ ( " &Help " ) )
main_layout . setMenuBar ( menubar )
quit_shortcut = QShortcut ( QKeySequence ( " Ctrl+Q " ) , self )
self . connect ( quit_shortcut , SIGNAL ( " activated() " ) , self . close )
close_shortcut = QShortcut ( QKeySequence ( " Ctrl+W " ) , self )
self . connect ( close_shortcut , SIGNAL ( " activated() " ) , self . close )
self . setWindowIcon ( QIcon ( " :electrum.png " ) )
self . setWindowTitle ( " Electrum " )
self . setWindowFlags ( Qt . Window | Qt . MSWindowsFixedSizeDialogHint )
self . layout ( ) . setSizeConstraint ( QLayout . SetFixedSize )
self . setObjectName ( " main_window " )
self . show ( )
def recompute_style ( self ) :
qApp . style ( ) . unpolish ( self )
qApp . style ( ) . polish ( self )
def closeEvent ( self , event ) :
super ( MiniWindow , self ) . closeEvent ( event )
qApp . quit ( )
def set_payment_fields ( self , dest_address , amount ) :
self . address_input . setText ( dest_address )
self . address_field_changed ( dest_address )
self . amount_input . setText ( amount )
def activate ( self ) :
pass
def deactivate ( self ) :
pass
def set_quote_currency ( self , currency ) :
assert currency in self . quote_currencies
self . quote_currencies . remove ( currency )
self . quote_currencies = [ currency ] + self . quote_currencies
self . refresh_balance ( )
def change_quote_currency ( self ) :
self . quote_currencies = \
self . quote_currencies [ 1 : ] + self . quote_currencies [ 0 : 1 ]
self . actuator . set_config_currency ( self . quote_currencies [ 0 ] )
self . refresh_balance ( )
def refresh_balance ( self ) :
if self . btc_balance is None :
# Price has been discovered before wallet has been loaded
# and server connect... so bail.
return
self . set_balances ( self . btc_balance )
self . amount_input_changed ( self . amount_input . text ( ) )
def set_balances ( self , btc_balance ) :
self . btc_balance = btc_balance
quote_text = self . create_quote_text ( btc_balance )
if quote_text :
quote_text = " ( %s ) " % quote_text
btc_balance = " %.2f " % ( btc_balance / bitcoin ( 1 ) )
self . balance_label . set_balance_text ( btc_balance , quote_text )
self . setWindowTitle ( " Electrum - %s BTC " % btc_balance )
def amount_input_changed ( self , amount_text ) :
self . check_button_status ( )
try :
amount = D ( str ( amount_text ) )
except decimal . InvalidOperation :
self . balance_label . show_balance ( )
else :
quote_text = self . create_quote_text ( amount * bitcoin ( 1 ) )
if quote_text :
self . balance_label . set_amount_text ( quote_text )
self . balance_label . show_amount ( )
else :
self . balance_label . show_balance ( )
def create_quote_text ( self , btc_balance ) :
quote_currency = self . quote_currencies [ 0 ]
quote_balance = self . exchanger . exchange ( btc_balance , quote_currency )
if quote_balance is None :
quote_text = " "
else :
quote_text = " %.2f %s " % ( ( quote_balance / bitcoin ( 1 ) ) ,
quote_currency )
return quote_text
def send ( self ) :
if self . actuator . send ( self . address_input . text ( ) , self . amount_input . text ( ) , self ) :
self . address_input . setText ( " " )
self . amount_input . setText ( " " )
def check_button_status ( self ) :
if self . address_input . property ( " isValid " ) == True and len ( self . amount_input . text ( ) ) != 0 :
self . send_button . setDisabled ( False )
else :
self . send_button . setDisabled ( True )
def address_field_changed ( self , address ) :
if self . actuator . is_valid ( address ) :
self . check_button_status ( )
self . address_input . setProperty ( " isValid " , True )
self . recompute_style ( self . address_input )
else :
self . send_button . setDisabled ( True )
self . address_input . setProperty ( " isValid " , False )
self . recompute_style ( self . address_input )
if len ( address ) == 0 :
self . address_input . setProperty ( " isValid " , None )
self . recompute_style ( self . address_input )
def recompute_style ( self , element ) :
self . style ( ) . unpolish ( element )
self . style ( ) . polish ( element )
def copy_address ( self ) :
receive_popup = ReceivePopup ( self . receive_button )
self . actuator . copy_address ( receive_popup )
def update_completions ( self , completions ) :
self . address_completions . setStringList ( completions )
def acceptbit ( self ) :
self . actuator . acceptbit ( self . quote_currencies [ 0 ] )
def show_about ( self ) :
QMessageBox . about ( self , " Electrum " ,
_ ( " Electrum ' s focus is speed, with low resource usage and simplifying Bitcoin. You do not need to perform regular backups, because your wallet can be recovered from a secret phrase that you can memorize or write on paper. Startup times are instant because it operates in conjuction with high-performance servers that handle the most complicated parts of the Bitcoin system. " ) )
def show_report_bug ( self ) :
QMessageBox . information ( self , " Electrum - " + _ ( " Reporting Bugs " ) ,
_ ( " Email bug reports to %s " ) % " genjix " + " @ " + " riseup.net " )
class BalanceLabel ( QLabel ) :
SHOW_CONNECTING = 1
SHOW_BALANCE = 2
SHOW_AMOUNT = 3
def __init__ ( self , change_quote_currency , parent = None ) :
super ( QLabel , self ) . __init__ ( _ ( " Connecting... " ) , parent )
self . change_quote_currency = change_quote_currency
self . state = self . SHOW_CONNECTING
self . balance_text = " "
self . amount_text = " "
def mousePressEvent ( self , event ) :
if self . state != self . SHOW_CONNECTING :
self . change_quote_currency ( )
def set_balance_text ( self , btc_balance , quote_text ) :
if self . state == self . SHOW_CONNECTING :
self . state = self . SHOW_BALANCE
self . balance_text = " <span style= ' font-size: 18pt ' > %s </span> <span style= ' font-size: 10pt ' >BTC</span> <span style= ' font-size: 10pt ' > %s </span> " % ( btc_balance , quote_text )
if self . state == self . SHOW_BALANCE :
self . setText ( self . balance_text )
def set_amount_text ( self , quote_text ) :
self . amount_text = " <span style= ' font-size: 10pt ' > %s </span> " % quote_text
if self . state == self . SHOW_AMOUNT :
self . setText ( self . amount_text )
def show_balance ( self ) :
if self . state == self . SHOW_AMOUNT :
self . state = self . SHOW_BALANCE
self . setText ( self . balance_text )
def show_amount ( self ) :
if self . state == self . SHOW_BALANCE :
self . state = self . SHOW_AMOUNT
self . setText ( self . amount_text )
def ok_cancel_buttons ( dialog ) :
row_layout = QHBoxLayout ( )
row_layout . addStretch ( 1 )
ok_button = QPushButton ( _ ( " OK " ) )
row_layout . addWidget ( ok_button )
ok_button . clicked . connect ( dialog . accept )
cancel_button = QPushButton ( _ ( " Cancel " ) )
row_layout . addWidget ( cancel_button )
cancel_button . clicked . connect ( dialog . reject )
return row_layout
class PasswordDialog ( QDialog ) :
def __init__ ( self , parent ) :
super ( QDialog , self ) . __init__ ( parent )
self . setModal ( True )
self . password_input = QLineEdit ( )
self . password_input . setEchoMode ( QLineEdit . Password )
main_layout = QVBoxLayout ( self )
message = _ ( ' Please enter your password ' )
main_layout . addWidget ( QLabel ( message ) )
grid = QGridLayout ( )
grid . setSpacing ( 8 )
grid . addWidget ( QLabel ( _ ( ' Password ' ) ) , 1 , 0 )
grid . addWidget ( self . password_input , 1 , 1 )
main_layout . addLayout ( grid )
main_layout . addLayout ( ok_cancel_buttons ( self ) )
self . setLayout ( main_layout )
def run ( self ) :
if not self . exec_ ( ) :
return
return unicode ( self . password_input . text ( ) )
class ReceivePopup ( QDialog ) :
def leaveEvent ( self , event ) :
self . close ( )
def setup ( self , address ) :
label = QLabel ( _ ( " Copied your Bitcoin address to the clipboard! " ) )
address_display = QLineEdit ( address )
address_display . setReadOnly ( True )
resize_line_edit_width ( address_display , address )
main_layout = QVBoxLayout ( self )
main_layout . addWidget ( label )
main_layout . addWidget ( address_display )
self . setMouseTracking ( True )
self . setWindowTitle ( " Electrum - " + _ ( " Receive Bitcoin payment " ) )
self . setWindowFlags ( Qt . Window | Qt . FramelessWindowHint | Qt . MSWindowsFixedSizeDialogHint )
self . layout ( ) . setSizeConstraint ( QLayout . SetFixedSize )
#self.setFrameStyle(QFrame.WinPanel|QFrame.Raised)
#self.setAlignment(Qt.AlignCenter)
def popup ( self ) :
parent = self . parent ( )
top_left_pos = parent . mapToGlobal ( parent . rect ( ) . bottomLeft ( ) )
self . move ( top_left_pos )
center_mouse_pos = self . mapToGlobal ( self . rect ( ) . center ( ) )
QCursor . setPos ( center_mouse_pos )
self . show ( )
class MiniActuator :
def __init__ ( self , wallet ) :
self . wallet = wallet
def set_configured_currency ( self , set_quote_currency ) :
currency = self . wallet . conversion_currency
# currency can be none when Electrum is used for the first
# time and no setting has been created yet.
if currency is not None :
set_quote_currency ( currency )
def set_config_currency ( self , conversion_currency ) :
self . wallet . conversion_currency = conversion_currency
def copy_address ( self , receive_popup ) :
addrs = [ addr for addr in self . wallet . all_addresses ( )
if not self . wallet . is_change ( addr ) ]
# Select most recent addresses from gap limit
addrs = addrs [ - self . wallet . gap_limit : ]
copied_address = random . choice ( addrs )
qApp . clipboard ( ) . setText ( copied_address )
receive_popup . setup ( copied_address )
receive_popup . popup ( )
def send ( self , address , amount , parent_window ) :
dest_address = self . fetch_destination ( address )
if dest_address is None or not self . wallet . is_valid ( dest_address ) :
QMessageBox . warning ( parent_window , _ ( ' Error ' ) ,
_ ( ' Invalid Bitcoin Address ' ) + ' : \n ' + address , _ ( ' OK ' ) )
return False
convert_amount = lambda amount : \
int ( D ( unicode ( amount ) ) * bitcoin ( 1 ) )
amount = convert_amount ( amount )
if self . wallet . use_encryption :
password_dialog = PasswordDialog ( parent_window )
password = password_dialog . run ( )
if not password :
return
else :
password = None
fee = 0
# 0.1 BTC = 10000000
if amount < bitcoin ( 1 ) / 10 :
# 0.001 BTC
fee = bitcoin ( 1 ) / 1000
try :
tx = self . wallet . mktx ( dest_address , amount , " " , password , fee )
except BaseException as error :
QMessageBox . warning ( parent_window , _ ( ' Error ' ) , str ( error ) , _ ( ' OK ' ) )
return False
status , message = self . wallet . sendtx ( tx )
if not status :
QMessageBox . warning ( parent_window , _ ( ' Error ' ) , message , _ ( ' OK ' ) )
return False
QMessageBox . information ( parent_window , ' ' ,
_ ( ' Payment sent. ' ) + ' \n ' + message , _ ( ' OK ' ) )
return True
def fetch_destination ( self , address ) :
recipient = unicode ( address ) . strip ( )
# alias
match1 = re . match ( " ^(|([ \ w \ - \ .]+)@)(( \ w[ \ w \ -]+ \ .)+[ \ w \ -]+)$ " ,
recipient )
# label or alias, with address in brackets
match2 = re . match ( " (.*?) \ s* \ <([1-9A-HJ-NP-Za-km-z] { 26,}) \ > " ,
recipient )
if match1 :
dest_address = \
self . wallet . get_alias ( recipient , True ,
self . show_message , self . question )
return dest_address
elif match2 :
return match2 . group ( 2 )
else :
return recipient
def is_valid ( self , address ) :
return self . wallet . is_valid ( address )
def acceptbit ( self , currency ) :
master_pubkey = self . wallet . master_public_key . encode ( " hex " )
url = " http://acceptbit.com/mpk/ %s / %s " % ( master_pubkey , currency )
webbrowser . open ( url )
class MiniDriver ( QObject ) :
INITIALIZING = 0
CONNECTING = 1
SYNCHRONIZING = 2
READY = 3
def __init__ ( self , wallet , window ) :
super ( QObject , self ) . __init__ ( )
self . wallet = wallet
self . window = window
self . wallet . register_callback ( self . update_callback )
self . state = None
self . initializing ( )
self . connect ( self , SIGNAL ( " updatesignal() " ) , self . update )
self . update_callback ( )
# This is a hack to workaround that Qt does not like changing the
# window properties from this other thread before the runloop has
# been called from.
def update_callback ( self ) :
self . emit ( SIGNAL ( " updatesignal() " ) )
def update ( self ) :
if not self . wallet . interface :
self . initializing ( )
elif not self . wallet . interface . is_connected :
self . connecting ( )
elif not self . wallet . blocks == - 1 :
self . connecting ( )
elif not self . wallet . is_up_to_date :
self . synchronizing ( )
else :
self . ready ( )
if self . wallet . up_to_date :
self . update_balance ( )
self . update_completions ( )
def initializing ( self ) :
if self . state == self . INITIALIZING :
return
self . state = self . INITIALIZING
self . window . deactivate ( )
def connecting ( self ) :
if self . state == self . CONNECTING :
return
self . state = self . CONNECTING
self . window . deactivate ( )
def synchronizing ( self ) :
if self . state == self . SYNCHRONIZING :
return
self . state = self . SYNCHRONIZING
self . window . deactivate ( )
def ready ( self ) :
if self . state == self . READY :
return
self . state = self . READY
self . window . activate ( )
def update_balance ( self ) :
conf_balance , unconf_balance = self . wallet . get_balance ( )
balance = D ( conf_balance + unconf_balance )
self . window . set_balances ( balance )
def update_completions ( self ) :
completions = [ ]
for addr , label in self . wallet . labels . items ( ) :
if addr in self . wallet . addressbook :
completions . append ( " %s < %s > " % ( label , addr ) )
completions = completions + self . wallet . aliases . keys ( )
self . window . update_completions ( completions )
if __name__ == " __main__ " :
app = QApplication ( sys . argv )
with open ( rsrc ( " style.css " ) ) as style_file :
app . setStyleSheet ( style_file . read ( ) )
mini = MiniWindow ( )
sys . exit ( app . exec_ ( ) )