|
|
|
from kivy.app import App
|
|
|
|
from kivy.factory import Factory
|
|
|
|
from kivy.properties import ObjectProperty
|
|
|
|
from kivy.lang import Builder
|
|
|
|
from kivy.clock import Clock
|
|
|
|
from kivy.uix.label import Label
|
|
|
|
|
|
|
|
from electrum_gui.kivy.i18n import _
|
|
|
|
from datetime import datetime
|
|
|
|
|
|
|
|
Builder.load_string('''
|
|
|
|
<TxDialog@Popup>
|
|
|
|
id: popup
|
|
|
|
title: _('Transaction')
|
|
|
|
is_mine: True
|
|
|
|
can_sign: False
|
|
|
|
can_broadcast: False
|
|
|
|
fee_str: ''
|
|
|
|
date_str: ''
|
|
|
|
amount_str: ''
|
|
|
|
tx_hash: ''
|
|
|
|
status_str: ''
|
|
|
|
description: ''
|
|
|
|
outputs_str: ''
|
|
|
|
ScrollView:
|
|
|
|
BoxLayout:
|
|
|
|
orientation: 'vertical'
|
|
|
|
spacing: '10dp'
|
|
|
|
padding: '10dp'
|
|
|
|
GridLayout:
|
|
|
|
size_hint: 1, None
|
|
|
|
height: self.minimum_height
|
|
|
|
cols: 2
|
|
|
|
spacing: '10dp'
|
|
|
|
TopLabel:
|
|
|
|
text: _('Status')
|
|
|
|
TopLabel:
|
|
|
|
text: root.status_str
|
|
|
|
TopLabel:
|
|
|
|
text: _('Description') if root.description else ''
|
|
|
|
TopLabel:
|
|
|
|
text: root.description
|
|
|
|
TopLabel:
|
|
|
|
text: _('Date') if root.date_str else ''
|
|
|
|
TopLabel:
|
|
|
|
text: root.date_str
|
|
|
|
TopLabel:
|
|
|
|
text: _('Amount sent') if root.is_mine else _('Amount received')
|
|
|
|
TopLabel:
|
|
|
|
text: root.amount_str
|
|
|
|
TopLabel:
|
|
|
|
text: _('Transaction fee') if root.fee_str else ''
|
|
|
|
TopLabel:
|
|
|
|
text: root.fee_str
|
|
|
|
TopLabel:
|
|
|
|
text: _('Outputs') + ':'
|
|
|
|
OutputList:
|
|
|
|
id: outputs
|
|
|
|
TopLabel:
|
|
|
|
text: _('Transaction ID') + ':' if root.tx_hash else ''
|
|
|
|
TopLabel:
|
|
|
|
font_size: '6pt'
|
|
|
|
text: '[ref=x]%s[/ref]' %' '.join(map(''.join, zip(*[iter(root.tx_hash)]*4))) if root.tx_hash else ''
|
|
|
|
padding: '10dp', '10dp'
|
|
|
|
on_ref_press:
|
|
|
|
app._clipboard.copy(self.text)
|
|
|
|
app.show_info(_('Transaction ID copied to clipboard'))
|
|
|
|
canvas.before:
|
|
|
|
Color:
|
|
|
|
rgb: .3, .3, .3
|
|
|
|
Rectangle:
|
|
|
|
size: self.size
|
|
|
|
pos: self.pos
|
|
|
|
Widget:
|
|
|
|
size_hint: 1, 0.2
|
|
|
|
|
|
|
|
BoxLayout:
|
|
|
|
size_hint: 1, None
|
|
|
|
height: '48dp'
|
|
|
|
Button:
|
|
|
|
size_hint: 0.5, None
|
|
|
|
height: '48dp'
|
|
|
|
text: _('Sign') if root.can_sign else _('Broadcast') if root.can_broadcast else ''
|
|
|
|
opacity: 1 if root.can_sign or root.can_broadcast else 0
|
|
|
|
disabled: not( root.can_sign or root.can_broadcast )
|
|
|
|
on_release:
|
|
|
|
if root.can_sign: root.do_sign()
|
|
|
|
if root.can_broadcast: root.do_broadcast()
|
|
|
|
IconButton:
|
|
|
|
size_hint: 0.5, None
|
|
|
|
height: '48dp'
|
|
|
|
icon: 'atlas://gui/kivy/theming/light/qrcode'
|
|
|
|
on_release: root.show_qr()
|
|
|
|
Button:
|
|
|
|
size_hint: 0.5, None
|
|
|
|
height: '48dp'
|
|
|
|
text: _('Close')
|
|
|
|
on_release: popup.dismiss()
|
|
|
|
''')
|
|
|
|
|
|
|
|
|
|
|
|
class TxDialog(Factory.Popup):
|
|
|
|
|
|
|
|
def __init__(self, app, tx):
|
|
|
|
Factory.Popup.__init__(self)
|
|
|
|
self.app = app
|
|
|
|
self.wallet = self.app.wallet
|
|
|
|
self.tx = tx
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
self.can_broadcast = False
|
|
|
|
if self.tx.is_complete():
|
|
|
|
self.tx_hash = self.tx.hash()
|
|
|
|
self.description = self.wallet.get_label(self.tx_hash)
|
|
|
|
if self.tx_hash in self.wallet.transactions.keys():
|
|
|
|
conf, timestamp = self.wallet.get_confirmations(self.tx_hash)
|
|
|
|
self.status_str = _("%d confirmations")%conf if conf else _('Pending')
|
|
|
|
if timestamp:
|
|
|
|
self.date_str = datetime.fromtimestamp(timestamp).isoformat(' ')[:-3]
|
|
|
|
else:
|
|
|
|
self.can_broadcast = self.app.network is not None
|
|
|
|
self.status_str = _('Signed')
|
|
|
|
else:
|
|
|
|
s, r = self.tx.signature_count()
|
|
|
|
self.status_str = _("Unsigned") if s == 0 else _('Partially signed') + ' (%d/%d)'%(s,r)
|
|
|
|
|
|
|
|
is_relevant, is_mine, v, fee = self.wallet.get_wallet_delta(self.tx)
|
|
|
|
self.is_mine = is_mine
|
|
|
|
if is_relevant:
|
|
|
|
if is_mine:
|
|
|
|
if fee is not None:
|
|
|
|
self.amount_str = self.app.format_amount_and_units(-v+fee)
|
|
|
|
self.fee_str = self.app.format_amount_and_units(-fee)
|
|
|
|
else:
|
|
|
|
self.amount_str = self.app.format_amount_and_units(-v)
|
|
|
|
self.fee_str = _("unknown")
|
|
|
|
else:
|
|
|
|
self.amount_str = self.app.format_amount_and_units(v)
|
|
|
|
self.fee_str = ''
|
|
|
|
else:
|
|
|
|
self.amount_str = _("Transaction unrelated to your wallet")
|
|
|
|
self.fee_str = ''
|
|
|
|
self.can_sign = self.wallet.can_sign(self.tx)
|
|
|
|
|
|
|
|
self.ids.outputs.clear_widgets()
|
|
|
|
for (type, address, amount) in self.tx.outputs():
|
|
|
|
self.ids.outputs.add_output(address, amount)
|
|
|
|
|
|
|
|
def do_sign(self):
|
|
|
|
self.app.protected(_("Enter your PIN code in order to sign this transaction"), self._do_sign, ())
|
|
|
|
|
|
|
|
def _do_sign(self, password):
|
|
|
|
self.status_str = _('Signing') + '...'
|
|
|
|
Clock.schedule_once(lambda dt: self.__do_sign(password), 0.1)
|
|
|
|
|
|
|
|
def __do_sign(self, password):
|
|
|
|
self.app.wallet.sign_transaction(self.tx, password)
|
|
|
|
self.update()
|
|
|
|
|
|
|
|
def do_broadcast(self):
|
|
|
|
self.app.show_info(_('Broadcasting'))
|
|
|
|
ok, txid = self.app.wallet.sendtx(self.tx)
|
|
|
|
self.app.show_info(txid)
|
|
|
|
|
|
|
|
def show_qr(self):
|
|
|
|
from electrum.bitcoin import base_encode
|
|
|
|
text = str(self.tx).decode('hex')
|
|
|
|
text = base_encode(text, base=43)
|
|
|
|
self.app.qr_dialog(_("Raw Transaction"), text)
|