3 from __future__ import with_statement
4 from __future__ import division
10 from PyQt4 import QtGui
11 from PyQt4 import QtCore
13 from util import qui_utils
14 from util import misc as misc_utils
17 _moduleLogger = logging.getLogger(__name__)
20 class CredentialsDialog(object):
22 def __init__(self, app):
23 self._usernameField = QtGui.QLineEdit()
24 self._passwordField = QtGui.QLineEdit()
25 self._passwordField.setEchoMode(QtGui.QLineEdit.PasswordEchoOnEdit)
27 self._credLayout = QtGui.QGridLayout()
28 self._credLayout.addWidget(QtGui.QLabel("Username"), 0, 0)
29 self._credLayout.addWidget(self._usernameField, 0, 1)
30 self._credLayout.addWidget(QtGui.QLabel("Password"), 1, 0)
31 self._credLayout.addWidget(self._passwordField, 1, 1)
33 self._loginButton = QtGui.QPushButton("&Login")
34 self._buttonLayout = QtGui.QDialogButtonBox(QtGui.QDialogButtonBox.Cancel)
35 self._buttonLayout.addButton(self._loginButton, QtGui.QDialogButtonBox.AcceptRole)
37 self._layout = QtGui.QVBoxLayout()
38 self._layout.addLayout(self._credLayout)
39 self._layout.addWidget(self._buttonLayout)
41 self._dialog = QtGui.QDialog()
42 self._dialog.setWindowTitle("Login")
43 self._dialog.setLayout(self._layout)
44 self._dialog.setAttribute(QtCore.Qt.WA_DeleteOnClose, False)
45 qui_utils.set_autorient(self._dialog, True)
46 self._buttonLayout.accepted.connect(self._dialog.accept)
47 self._buttonLayout.rejected.connect(self._dialog.reject)
49 self._closeWindowAction = QtGui.QAction(None)
50 self._closeWindowAction.setText("Close")
51 self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w"))
52 self._closeWindowAction.triggered.connect(self._on_close_window)
54 self._dialog.addAction(self._closeWindowAction)
55 self._dialog.addAction(app.quitAction)
56 self._dialog.addAction(app.fullscreenAction)
58 def run(self, defaultUsername, defaultPassword, parent=None):
59 self._dialog.setParent(parent, QtCore.Qt.Dialog)
61 self._usernameField.setText(defaultUsername)
62 self._passwordField.setText(defaultPassword)
64 response = self._dialog.exec_()
65 if response == QtGui.QDialog.Accepted:
66 return str(self._usernameField.text()), str(self._passwordField.text())
67 elif response == QtGui.QDialog.Rejected:
68 raise RuntimeError("Login Cancelled")
70 raise RuntimeError("Unknown Response")
72 self._dialog.setParent(None, QtCore.Qt.Dialog)
75 @QtCore.pyqtSlot(bool)
76 @misc_utils.log_exception(_moduleLogger)
77 def _on_close_window(self, checked = True):
81 class AccountDialog(object):
83 # @bug Can't configure callback number
85 def __init__(self, app):
88 self._accountNumberLabel = QtGui.QLabel("NUMBER NOT SET")
89 self._clearButton = QtGui.QPushButton("Clear Account")
90 self._clearButton.clicked.connect(self._on_clear)
92 self._credLayout = QtGui.QGridLayout()
93 self._credLayout.addWidget(QtGui.QLabel("Account"), 0, 0)
94 self._credLayout.addWidget(self._accountNumberLabel, 0, 1)
95 self._credLayout.addWidget(QtGui.QLabel("Callback"), 1, 0)
96 self._credLayout.addWidget(QtGui.QLabel(""), 2, 0)
97 self._credLayout.addWidget(self._clearButton, 2, 1)
98 self._credLayout.addWidget(QtGui.QLabel(""), 3, 0)
100 self._loginButton = QtGui.QPushButton("&Apply")
101 self._buttonLayout = QtGui.QDialogButtonBox(QtGui.QDialogButtonBox.Cancel)
102 self._buttonLayout.addButton(self._loginButton, QtGui.QDialogButtonBox.AcceptRole)
104 self._layout = QtGui.QVBoxLayout()
105 self._layout.addLayout(self._credLayout)
106 self._layout.addWidget(self._buttonLayout)
108 self._dialog = QtGui.QDialog()
109 self._dialog.setWindowTitle("Login")
110 self._dialog.setLayout(self._layout)
111 qui_utils.set_autorient(self._dialog, True)
112 self._buttonLayout.accepted.connect(self._dialog.accept)
113 self._buttonLayout.rejected.connect(self._dialog.reject)
115 self._closeWindowAction = QtGui.QAction(None)
116 self._closeWindowAction.setText("Close")
117 self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w"))
118 self._closeWindowAction.triggered.connect(self._on_close_window)
120 self._dialog.addAction(self._closeWindowAction)
121 self._dialog.addAction(app.quitAction)
122 self._dialog.addAction(app.fullscreenAction)
128 accountNumber = property(
129 lambda self: str(self._accountNumberLabel.text()),
130 lambda self, num: self._accountNumberLabel.setText(num),
133 def run(self, parent=None):
134 self._doClear = False
135 self._dialog.setParent(parent)
137 response = self._dialog.exec_()
141 @QtCore.pyqtSlot(bool)
142 def _on_clear(self, checked = False):
144 self._dialog.accept()
147 @QtCore.pyqtSlot(bool)
148 @misc_utils.log_exception(_moduleLogger)
149 def _on_close_window(self, checked = True):
150 self._dialog.reject()
153 class SMSEntryWindow(object):
157 def __init__(self, parent, app, session, errorLog):
158 self._session = session
159 self._session.draft.recipientsChanged.connect(self._on_recipients_changed)
160 self._session.draft.called.connect(self._on_op_finished)
161 self._session.draft.sentMessage.connect(self._on_op_finished)
162 self._session.draft.cancelled.connect(self._on_op_finished)
163 self._errorLog = errorLog
165 self._targetLayout = QtGui.QVBoxLayout()
166 self._targetList = QtGui.QWidget()
167 self._targetList.setLayout(self._targetLayout)
168 self._history = QtGui.QTextEdit()
169 self._history.setReadOnly(True)
170 self._smsEntry = QtGui.QTextEdit()
171 self._smsEntry.textChanged.connect(self._on_letter_count_changed)
173 self._entryLayout = QtGui.QVBoxLayout()
174 self._entryLayout.addWidget(self._targetList)
175 self._entryLayout.addWidget(self._history)
176 self._entryLayout.addWidget(self._smsEntry)
177 self._entryLayout.setContentsMargins(0, 0, 0, 0)
178 self._entryWidget = QtGui.QWidget()
179 self._entryWidget.setLayout(self._entryLayout)
180 self._entryWidget.setContentsMargins(0, 0, 0, 0)
181 self._scrollEntry = QtGui.QScrollArea()
182 self._scrollEntry.setWidget(self._entryWidget)
183 self._scrollEntry.setWidgetResizable(True)
184 self._scrollEntry.setAlignment(QtCore.Qt.AlignLeft | QtCore.Qt.AlignBottom)
185 self._scrollEntry.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
186 self._scrollEntry.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
188 self._characterCountLabel = QtGui.QLabel("Letters: %s" % 0)
189 self._singleNumberSelector = QtGui.QComboBox()
190 self._smsButton = QtGui.QPushButton("SMS")
191 self._smsButton.clicked.connect(self._on_sms_clicked)
192 self._dialButton = QtGui.QPushButton("Dial")
193 self._dialButton.clicked.connect(self._on_call_clicked)
195 self._buttonLayout = QtGui.QHBoxLayout()
196 self._buttonLayout.addWidget(self._characterCountLabel)
197 self._buttonLayout.addWidget(self._singleNumberSelector)
198 self._buttonLayout.addWidget(self._smsButton)
199 self._buttonLayout.addWidget(self._dialButton)
201 self._layout = QtGui.QVBoxLayout()
202 self._layout.addWidget(self._scrollEntry)
203 self._layout.addLayout(self._buttonLayout)
205 centralWidget = QtGui.QWidget()
206 centralWidget.setLayout(self._layout)
208 self._window = QtGui.QMainWindow(parent)
209 qui_utils.set_autorient(self._window, True)
210 qui_utils.set_stackable(self._window, True)
211 self._window.setWindowTitle("Contact")
212 self._window.setCentralWidget(centralWidget)
214 self._closeWindowAction = QtGui.QAction(None)
215 self._closeWindowAction.setText("Close")
216 self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w"))
217 self._closeWindowAction.triggered.connect(self._on_close_window)
219 fileMenu = self._window.menuBar().addMenu("&File")
220 fileMenu.addAction(self._closeWindowAction)
221 fileMenu.addAction(app.quitAction)
222 viewMenu = self._window.menuBar().addMenu("&View")
223 viewMenu.addAction(app.fullscreenAction)
226 self._update_recipients()
228 def _update_letter_count(self):
229 count = self._smsEntry.toPlainText().size()
230 numTexts, numCharInText = divmod(count, self.MAX_CHAR)
232 numCharsLeftInText = self.MAX_CHAR - numCharInText
233 self._characterCountLabel.setText("%d (%d)" % (numCharsLeftInText, numTexts))
235 def _update_button_state(self):
236 if self._session.draft.get_num_contacts() == 0:
237 self._dialButton.setEnabled(False)
238 self._smsButton.setEnabled(False)
239 elif self._session.draft.get_num_contacts() == 1:
240 count = self._smsEntry.toPlainText().size()
242 self._dialButton.setEnabled(True)
243 self._smsButton.setEnabled(False)
245 self._dialButton.setEnabled(False)
246 self._smsButton.setEnabled(True)
248 self._dialButton.setEnabled(False)
249 self._smsButton.setEnabled(True)
251 def _update_recipients(self):
252 draftContactsCount = self._session.draft.get_num_contacts()
253 if draftContactsCount == 0:
255 elif draftContactsCount == 1:
256 (cid, ) = self._session.draft.get_contacts()
257 title = self._session.draft.get_title(cid)
258 description = self._session.draft.get_description(cid)
259 numbers = self._session.draft.get_numbers(cid)
261 self._targetList.setVisible(False)
263 self._history.setHtml(description)
264 self._history.setVisible(True)
266 self._history.setHtml("")
267 self._history.setVisible(False)
268 self._populate_number_selector(self._singleNumberSelector, cid, numbers)
270 self._scroll_to_bottom()
271 self._window.setWindowTitle(title)
274 self._targetList.setVisible(True)
275 while self._targetLayout.count():
276 removedLayoutItem = self._targetLayout.takeAt(self._targetLayout.count()-1)
277 removedWidget = removedLayoutItem.widget()
278 removedWidget.close()
279 for cid in self._session.draft.get_contacts():
280 title = self._session.draft.get_title(cid)
281 description = self._session.draft.get_description(cid)
282 numbers = self._session.draft.get_numbers(cid)
284 titleLabel = QtGui.QLabel(title)
285 numberSelector = QtGui.QComboBox()
286 self._populate_number_selector(numberSelector, cid, numbers)
287 deleteButton = QtGui.QPushButton("Delete")
288 callback = functools.partial(
289 self._on_remove_contact,
292 callback.__name__ = "b"
293 deleteButton.clicked.connect(callback)
295 rowLayout = QtGui.QHBoxLayout()
296 rowLayout.addWidget(titleLabel)
297 rowLayout.addWidget(numberSelector)
298 rowLayout.addWidget(deleteButton)
299 rowWidget = QtGui.QWidget()
300 rowWidget.setLayout(rowLayout)
301 self._targetLayout.addWidget(rowWidget)
302 self._history.setHtml("")
303 self._history.setVisible(False)
304 self._singleNumberSelector.setVisible(False)
306 self._scroll_to_bottom()
307 self._window.setWindowTitle("Contacts")
310 def _populate_number_selector(self, selector, cid, numbers):
311 while 0 < selector.count():
312 selector.removeItem(0)
314 if len(numbers) == 1:
315 numbers, defaultIndex = _get_contact_numbers(self._session, cid, numbers[0])
319 for number, description in numbers:
321 label = "%s - %s" % (number, description)
324 selector.addItem(label)
325 selector.setVisible(True)
327 selector.setEnabled(True)
328 selector.setCurrentIndex(defaultIndex)
330 selector.setEnabled(False)
331 callback = functools.partial(
332 self._on_change_number,
335 callback.__name__ = "thanks partials for not having names and pyqt for requiring them"
336 selector.currentIndexChanged.connect(
337 QtCore.pyqtSlot(int)(callback)
340 def _scroll_to_bottom(self):
341 self._scrollEntry.ensureWidgetVisible(self._smsEntry)
343 @misc_utils.log_exception(_moduleLogger)
344 def _on_sms_clicked(self, arg):
345 message = str(self._smsEntry.toPlainText())
346 self._session.draft.send(message)
347 self._smsEntry.setPlainText("")
349 @misc_utils.log_exception(_moduleLogger)
350 def _on_call_clicked(self, arg):
351 self._session.draft.call()
352 self._smsEntry.setPlainText("")
354 @misc_utils.log_exception(_moduleLogger)
355 def _on_remove_contact(self, cid, toggled):
356 self._session.draft.remove_contact(cid)
358 @misc_utils.log_exception(_moduleLogger)
359 def _on_change_number(self, cid, index):
360 # Exception thrown when the first item is removed
361 numbers = self._session.draft.get_numbers(cid)
362 number = numbers[index][0]
363 self._session.draft.set_selected_number(cid, number)
366 @misc_utils.log_exception(_moduleLogger)
367 def _on_recipients_changed(self):
368 self._update_recipients()
371 @misc_utils.log_exception(_moduleLogger)
372 def _on_op_finished(self):
376 @misc_utils.log_exception(_moduleLogger)
377 def _on_letter_count_changed(self):
378 self._update_letter_count()
379 self._update_button_state()
382 @QtCore.pyqtSlot(bool)
383 @misc_utils.log_exception(_moduleLogger)
384 def _on_close_window(self, checked = True):
388 def _get_contact_numbers(session, contactId, numberDescription):
389 contactPhoneNumbers = []
390 if contactId and contactId != "0":
392 contactDetails = copy.deepcopy(session.get_contacts()[contactId])
393 contactPhoneNumbers = contactDetails["numbers"]
395 contactPhoneNumbers = []
396 contactPhoneNumbers = [
397 (contactPhoneNumber["phoneNumber"], contactPhoneNumber["phoneType"])
398 for contactPhoneNumber in contactPhoneNumbers
400 if contactPhoneNumbers:
401 uglyContactNumbers = (
402 misc_utils.make_ugly(contactNumber)
403 for (contactNumber, _) in contactPhoneNumbers
406 misc_utils.similar_ugly_numbers(numberDescription[0], contactNumber)
407 for contactNumber in uglyContactNumbers
410 defaultIndex = defaultMatches.index(True)
412 contactPhoneNumbers.append(numberDescription)
413 defaultIndex = len(contactPhoneNumbers)-1
415 "Could not find contact %r's number %s among %r" % (
416 contactId, numberDescription, contactPhoneNumbers
420 if not contactPhoneNumbers:
421 contactPhoneNumbers = [numberDescription]
424 return contactPhoneNumbers, defaultIndex