improved help text
[electrum-nvc.git] / gui / qt / network_dialog.py
1 #!/usr/bin/env python
2 #
3 # Electrum - lightweight Bitcoin client
4 # Copyright (C) 2012 thomasv@gitorious
5 #
6 # This program is free software: you can redistribute it and/or modify
7 # it under the terms of the GNU General Public License as published by
8 # the Free Software Foundation, either version 3 of the License, or
9 # (at your option) any later version.
10 #
11 # This program is distributed in the hope that it will be useful,
12 # but WITHOUT ANY WARRANTY; without even the implied warranty of
13 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 # GNU General Public License for more details.
15 #
16 # You should have received a copy of the GNU General Public License
17 # along with this program. If not, see <http://www.gnu.org/licenses/>.
18
19 import sys, time, datetime, re, threading
20 from electrum.i18n import _
21 from electrum.util import print_error, print_msg
22 import os.path, json, ast, traceback
23
24 from PyQt4.QtGui import *
25 from PyQt4.QtCore import *
26 from electrum import DEFAULT_SERVERS, DEFAULT_PORTS
27
28 from util import *
29
30 protocol_names = ['TCP', 'HTTP', 'SSL', 'HTTPS']
31 protocol_letters = 'thsg'
32
33 class NetworkDialog(QDialog):
34     def __init__(self, network, config, parent):
35
36         QDialog.__init__(self,parent)
37         self.setModal(1)
38         self.setWindowTitle(_('Network'))
39         self.setMinimumSize(375, 20)
40
41         self.network = network
42         self.interface = interface = network.interface
43         self.config = config
44         self.protocol = None
45
46         if parent:
47             n = len(network.interfaces)
48             if n:
49                 status = _("Blockchain") + ": " + "%d "%(network.blockchain.height) + _("blocks") +  ".\n" + _("Getting block headers from %d nodes.")%n
50             else:
51                 status = _("Not connected")
52
53             if network.is_connected():
54                 status += "\n" + _("Server:") + " %s"%(interface.host) 
55             else:
56                 status += "\n" + _("Disconnected from server")
57                 
58         else:
59             import random
60             status = _("Please choose a server.") + "\n" + _("Select 'Cancel' if you are offline.")
61
62         server = network.default_server
63         self.servers = network.get_servers()
64
65
66         vbox = QVBoxLayout()
67         vbox.setSpacing(30)
68
69         hbox = QHBoxLayout()
70         l = QLabel()
71         l.setPixmap(QPixmap(":icons/network.png"))
72         hbox.addStretch(10)
73         hbox.addWidget(l)
74         hbox.addWidget(QLabel(status))
75         hbox.addStretch(50)
76         msg = _("Electrum sends your wallet addresses to a single server, in order to receive your transaction history.") + "\n\n" \
77             + _("In addition, Electrum connects to several nodes in order to download block headers and find out the longest blockchain.") + " " \
78             + _("This blockchain is used to verify your transactions.")
79         hbox.addWidget(HelpButton(msg))
80         vbox.addLayout(hbox)
81
82         # grid layout
83         grid = QGridLayout()
84         grid.setSpacing(8)
85         vbox.addLayout(grid)
86
87         # protocol
88         self.server_protocol = QComboBox()
89         self.server_host = QLineEdit()
90         self.server_host.setFixedWidth(200)
91         self.server_port = QLineEdit()
92         self.server_port.setFixedWidth(60)
93         self.server_protocol.addItems(protocol_names)
94         self.server_protocol.connect(self.server_protocol, SIGNAL('currentIndexChanged(int)'), self.change_protocol)
95
96         grid.addWidget(QLabel(_('Protocol') + ':'), 0, 0)
97         grid.addWidget(self.server_protocol, 0, 1)
98
99
100         # server
101         grid.addWidget(QLabel(_('Server') + ':'), 1, 0)
102
103         # auto connect
104         self.autocycle_cb = QCheckBox(_('Auto-connect'))
105         self.autocycle_cb.setChecked(self.config.get('auto_cycle', True))
106         grid.addWidget(self.autocycle_cb, 1, 1, 1, 2)
107         if not self.config.is_modifiable('auto_cycle'): self.autocycle_cb.setEnabled(False)
108         msg = _("If this option is enabled, Electrum will always use a server that is on the the longest blockchain.") + " " \
109             + _("If it is disabled, Electrum will warn you if your server is lagging.")
110         grid.addWidget(HelpButton(msg), 1, 3)
111
112         grid.addWidget(self.server_host, 2, 1, 1, 2)
113         grid.addWidget(self.server_port, 2, 3)
114
115
116         label = _('Active Servers') if network.irc_servers else _('Default Servers')
117         self.servers_list_widget = QTreeWidget(parent)
118         self.servers_list_widget.setHeaderLabels( [ label, _('Limit') ] )
119         self.servers_list_widget.setMaximumHeight(150)
120         self.servers_list_widget.setColumnWidth(0, 240)
121
122         if server:
123             host, port, protocol = server.split(':')
124             self.set_protocol(protocol)
125             self.change_server(host, protocol)
126         else:
127             self.set_protocol('s')
128
129         self.servers_list_widget.connect(self.servers_list_widget, 
130                                          SIGNAL('currentItemChanged(QTreeWidgetItem*,QTreeWidgetItem*)'), 
131                                          lambda x,y: self.server_changed(x))
132         grid.addWidget(self.servers_list_widget, 3, 1, 1, 3)
133
134         if not config.is_modifiable('server'):
135             for w in [self.server_host, self.server_port, self.server_protocol, self.servers_list_widget]: w.setEnabled(False)
136
137
138         
139         def enable_set_server():
140             enabled = not self.autocycle_cb.isChecked()
141             self.server_host.setEnabled(enabled)
142             self.server_port.setEnabled(enabled)
143             self.servers_list_widget.setEnabled(enabled)
144
145         self.autocycle_cb.clicked.connect(enable_set_server)
146         enable_set_server()
147
148         # proxy setting
149         self.proxy_mode = QComboBox()
150         self.proxy_host = QLineEdit()
151         self.proxy_host.setFixedWidth(200)
152         self.proxy_port = QLineEdit()
153         self.proxy_port.setFixedWidth(60)
154         self.proxy_mode.addItems(['NONE', 'SOCKS4', 'SOCKS5', 'HTTP'])
155
156         def check_for_disable(index = False):
157             if self.proxy_mode.currentText() != 'NONE':
158                 self.proxy_host.setEnabled(True)
159                 self.proxy_port.setEnabled(True)
160             else:
161                 self.proxy_host.setEnabled(False)
162                 self.proxy_port.setEnabled(False)
163
164         check_for_disable()
165         self.proxy_mode.connect(self.proxy_mode, SIGNAL('currentIndexChanged(int)'), check_for_disable)
166
167         if not self.config.is_modifiable('proxy'):
168             for w in [self.proxy_host, self.proxy_port, self.proxy_mode]: w.setEnabled(False)
169
170         proxy_config = network.proxy if network.proxy else { "mode":"none", "host":"localhost", "port":"8080"}
171         self.proxy_mode.setCurrentIndex(self.proxy_mode.findText(str(proxy_config.get("mode").upper())))
172         self.proxy_host.setText(proxy_config.get("host"))
173         self.proxy_port.setText(proxy_config.get("port"))
174
175         grid.addWidget(QLabel(_('Proxy') + ':'), 4, 0)
176         grid.addWidget(self.proxy_mode, 4, 1)
177         grid.addWidget(self.proxy_host, 4, 2)
178         grid.addWidget(self.proxy_port, 4, 3)
179
180         # buttons
181         vbox.addLayout(ok_cancel_buttons(self))
182         self.setLayout(vbox) 
183
184
185     def init_servers_list(self):
186         self.servers_list_widget.clear()
187         for _host, d in self.servers.items():
188             if d.get(self.protocol):
189                 pruning_level = d.get('pruning','')
190                 self.servers_list_widget.addTopLevelItem(QTreeWidgetItem( [ _host, pruning_level ] ))
191
192
193     def set_protocol(self, protocol):
194         if protocol != self.protocol:
195             self.protocol = protocol
196             self.init_servers_list()
197         
198     def change_protocol(self, index):
199         p = protocol_letters[index]
200         host = unicode(self.server_host.text())
201         pp = self.servers.get(host, DEFAULT_PORTS)
202         if p not in pp.keys():
203             p = pp.keys()[0]
204         port = pp[p]
205         self.server_host.setText( host )
206         self.server_port.setText( port )
207         self.set_protocol(p)
208
209     def server_changed(self, x):
210         if x: 
211             self.change_server(str(x.text(0)), self.protocol)
212
213     def change_server(self, host, protocol):
214
215         pp = self.servers.get(host, DEFAULT_PORTS)
216         if protocol:
217             port = pp.get(protocol)
218             if not port: protocol = None
219                     
220         if not protocol:
221             if 's' in pp.keys():
222                 protocol = 's'
223                 port = pp.get(protocol)
224             else:
225                 protocol = pp.keys()[0]
226                 port = pp.get(protocol)
227             
228         self.server_host.setText( host )
229         self.server_port.setText( port )
230         self.server_protocol.setCurrentIndex(protocol_letters.index(protocol))
231
232         if not self.servers: return
233         for p in protocol_letters:
234             i = protocol_letters.index(p)
235             j = self.server_protocol.model().index(i,0)
236             #if p not in pp.keys(): # and self.interface.is_connected:
237             #    self.server_protocol.model().setData(j, QVariant(0), Qt.UserRole-1)
238             #else:
239             #    self.server_protocol.model().setData(j, QVariant(33), Qt.UserRole-1)
240
241
242     def do_exec(self):
243
244         if not self.exec_():
245             return
246
247         host = str( self.server_host.text() )
248         port = str( self.server_port.text() )
249         protocol = protocol_letters[self.server_protocol.currentIndex()]
250
251         if self.proxy_mode.currentText() != 'NONE':
252             proxy = { 'mode':str(self.proxy_mode.currentText()).lower(), 
253                       'host':str(self.proxy_host.text()), 
254                       'port':str(self.proxy_port.text()) }
255         else:
256             proxy = None
257
258         auto_connect = self.autocycle_cb.isChecked()
259
260         self.network.set_parameters(host, port, protocol, proxy, auto_connect)
261         return True