3 from decimal import Decimal
5 from PyQt4.QtGui import *
6 from PyQt4.QtCore import *
7 import PyQt4.QtCore as QtCore
8 import PyQt4.QtGui as QtGui
10 from electrum_gui.qt.qrcodewidget import QRCodeWidget
12 from electrum import bmp, pyqrnative, BasePlugin
13 from electrum.i18n import _
16 if platform.system() == 'Windows':
17 MONOSPACE_FONT = 'Lucida Console'
18 elif platform.system() == 'Darwin':
19 MONOSPACE_FONT = 'Monaco'
21 MONOSPACE_FONT = 'monospace'
25 class QR_Window(QWidget):
27 def __init__(self, exchanger):
28 QWidget.__init__(self)
29 self.exchanger = exchanger
30 self.setWindowTitle('Electrum - '+_('Invoice'))
31 self.setMinimumSize(800, 250)
35 self.setFocusPolicy(QtCore.Qt.NoFocus)
37 main_box = QHBoxLayout()
39 self.qrw = QRCodeWidget()
40 main_box.addWidget(self.qrw, 1)
43 main_box.addLayout(vbox)
45 self.address_label = QLabel("")
46 #self.address_label.setFont(QFont(MONOSPACE_FONT))
47 vbox.addWidget(self.address_label)
49 self.label_label = QLabel("")
50 vbox.addWidget(self.label_label)
52 self.amount_label = QLabel("")
53 vbox.addWidget(self.amount_label)
56 self.setLayout(main_box)
59 def set_content(self, addr, label, amount, currency):
61 address_text = "<span style='font-size: 18pt'>%s</span>" % addr if addr else ""
62 self.address_label.setText(address_text)
64 if currency == 'BTC': currency = None
68 self.amount = Decimal(amount) / self.exchanger.exchange(1, currency) if currency else amount
70 self.amount = Decimal(amount)
71 self.amount = self.amount.quantize(Decimal('1.0000'))
74 amount_text += "<span style='font-size: 18pt'>%s %s</span><br/>" % (amount, currency)
75 amount_text += "<span style='font-size: 21pt'>%s</span> <span style='font-size: 16pt'>BTC</span> " % str(self.amount)
76 self.amount_label.setText(amount_text)
79 label_text = "<span style='font-size: 21pt'>%s</span>" % label if label else ""
80 self.label_label.setText(label_text)
82 msg = 'bitcoin:'+self.address
83 if self.amount is not None:
84 msg += '?amount=%s'%(str( self.amount))
85 if self.label is not None:
86 msg += '&label=%s'%(self.label)
87 elif self.label is not None:
88 msg += '?label=%s'%(self.label)
90 self.qrw.set_addr( msg )
95 class Plugin(BasePlugin):
98 return 'Point of Sale'
100 def description(self):
101 return _('Show QR code window and amounts requested for each address. Add menu item to request amount.')+_(' Note: This requires the exchange rate plugin to be installed.')
104 self.window = self.gui.main_window
105 self.wallet = self.window.wallet
107 self.qr_window = None
108 self.merchant_name = self.config.get('merchant_name', 'Invoice')
110 self.window.expert_mode = True
111 self.window.receive_list.setColumnCount(5)
112 self.window.receive_list.setHeaderLabels([ _('Address'), _('Label'), _('Balance'), _('Tx'), _('Request')])
113 self.requested_amounts = {}
114 self.toggle_QR_window(True)
117 if not self.config.get('use_exchange_rate'):
118 self.gui.main_window.show_message("Please enable exchange rates first!")
121 return BasePlugin.enable(self)
124 def load_wallet(self, wallet):
126 self.requested_amounts = self.wallet.storage.get('requested_amounts',{})
129 self.window.receive_list.setHeaderLabels([ _('Address'), _('Label'), _('Balance'), _('Tx')])
130 self.window.receive_list.setColumnCount(4)
131 for i,width in enumerate(self.window.column_widths['receive']):
132 self.window.receive_list.setColumnWidth(i, width)
133 self.toggle_QR_window(False)
136 def close_main_window(self):
138 self.qr_window.close()
139 self.qr_window = None
142 def timer_actions(self):
144 self.qr_window.qrw.update_qr()
147 def toggle_QR_window(self, show):
148 if show and not self.qr_window:
149 self.qr_window = QR_Window(self.gui.exchanger)
150 self.qr_window.setVisible(True)
151 self.qr_window_geometry = self.qr_window.geometry()
152 item = self.window.receive_list.currentItem()
154 address = str(item.text(1))
155 label = self.wallet.labels.get(address)
156 amount, currency = self.requested_amounts.get(address, (None, None))
157 self.qr_window.set_content( address, label, amount, currency )
159 elif show and self.qr_window and not self.qr_window.isVisible():
160 self.qr_window.setVisible(True)
161 self.qr_window.setGeometry(self.qr_window_geometry)
163 elif not show and self.qr_window and self.qr_window.isVisible():
164 self.qr_window_geometry = self.qr_window.geometry()
165 self.qr_window.setVisible(False)
169 def update_receive_item(self, address, item):
171 amount, currency = self.requested_amounts.get(address, (None, None))
173 print "cannot get requested amount", address, self.requested_amounts.get(address)
174 amount, currency = None, None
175 self.requested_amounts.pop(address)
177 amount_str = amount + (' ' + currency if currency else '') if amount is not None else ''
178 item.setData(column_index,0,amount_str)
182 def current_item_changed(self, a):
185 if a is not None and self.qr_window and self.qr_window.isVisible():
186 address = str(a.text(0))
187 label = self.wallet.labels.get(address)
189 amount, currency = self.requested_amounts.get(address, (None, None))
191 amount, currency = None, None
192 self.qr_window.set_content( address, label, amount, currency )
196 def item_changed(self, item, column):
197 if column != column_index:
199 address = str( item.text(0) )
200 text = str( item.text(column) )
202 seq = self.wallet.get_address_index(address)
205 print "cannot get index"
208 text = text.strip().upper()
210 m = re.match('^(\d*(|\.\d*))\s*(|BTC|EUR|USD|GBP|CNY|JPY|RUB|BRL)$', text)
211 if m and m.group(1) and m.group(1)!='.':
213 currency = m.group(3)
217 currency = currency.upper()
219 self.requested_amounts[address] = (amount, currency)
220 self.wallet.storage.put('requested_amounts', self.requested_amounts, True)
222 label = self.wallet.labels.get(address)
224 label = self.merchant_name + ' - %04d'%(index+1)
225 self.wallet.labels[address] = label
228 self.qr_window.set_content( address, label, amount, currency )
231 item.setText(column,'')
232 if address in self.requested_amounts:
233 self.requested_amounts.pop(address)
235 self.window.update_receive_item(self.window.receive_list.currentItem())
240 def edit_amount(self):
241 l = self.window.receive_list
242 item = l.currentItem()
243 item.setFlags(Qt.ItemIsEditable|Qt.ItemIsSelectable | Qt.ItemIsUserCheckable | Qt.ItemIsEnabled | Qt.ItemIsDragEnabled)
244 l.editItem( item, column_index )
245 item.setFlags(Qt.ItemIsSelectable | Qt.ItemIsUserCheckable | Qt.ItemIsEnabled | Qt.ItemIsDragEnabled)
248 def receive_menu(self, menu):
249 menu.addAction(_("Request amount"), self.edit_amount)
250 menu.addAction(_("Show Invoice"), lambda: self.toggle_QR_window(True))