3 from __future__ import with_statement
4 from __future__ import division
10 import util.qt_compat as qt_compat
11 QtCore = qt_compat.QtCore
12 QtGui = qt_compat.import_module("QtGui")
15 from util import qwrappers
16 from util import qui_utils
17 from util import misc as misc_utils
20 _moduleLogger = logging.getLogger(__name__)
23 class CredentialsDialog(object):
25 def __init__(self, app):
27 self._usernameField = QtGui.QLineEdit()
28 self._passwordField = QtGui.QLineEdit()
29 self._passwordField.setEchoMode(QtGui.QLineEdit.Password)
31 self._credLayout = QtGui.QGridLayout()
32 self._credLayout.addWidget(QtGui.QLabel("Username"), 0, 0)
33 self._credLayout.addWidget(self._usernameField, 0, 1)
34 self._credLayout.addWidget(QtGui.QLabel("Password"), 1, 0)
35 self._credLayout.addWidget(self._passwordField, 1, 1)
37 self._loginButton = QtGui.QPushButton("&Login")
38 self._buttonLayout = QtGui.QDialogButtonBox(QtGui.QDialogButtonBox.Cancel)
39 self._buttonLayout.addButton(self._loginButton, QtGui.QDialogButtonBox.AcceptRole)
41 self._layout = QtGui.QVBoxLayout()
42 self._layout.addLayout(self._credLayout)
43 self._layout.addWidget(self._buttonLayout)
45 self._dialog = QtGui.QDialog()
46 self._dialog.setWindowTitle("Login")
47 self._dialog.setLayout(self._layout)
48 self._dialog.setAttribute(QtCore.Qt.WA_DeleteOnClose, False)
49 self._buttonLayout.accepted.connect(self._dialog.accept)
50 self._buttonLayout.rejected.connect(self._dialog.reject)
52 self._closeWindowAction = QtGui.QAction(None)
53 self._closeWindowAction.setText("Close")
54 self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w"))
55 self._closeWindowAction.triggered.connect(self._on_close_window)
57 self._dialog.addAction(self._closeWindowAction)
58 self._dialog.addAction(app.quitAction)
59 self._dialog.addAction(app.fullscreenAction)
61 def run(self, defaultUsername, defaultPassword, parent=None):
62 self._dialog.setParent(parent, QtCore.Qt.Dialog)
64 self._usernameField.setText(defaultUsername)
65 self._passwordField.setText(defaultPassword)
67 response = self._dialog.exec_()
68 if response == QtGui.QDialog.Accepted:
69 return str(self._usernameField.text()), str(self._passwordField.text())
70 elif response == QtGui.QDialog.Rejected:
73 _moduleLogger.error("Unknown response")
76 self._dialog.setParent(None, QtCore.Qt.Dialog)
82 _moduleLogger.exception("Oh well")
86 @misc_utils.log_exception(_moduleLogger)
87 def _on_close_window(self, checked = True):
88 with qui_utils.notify_error(self._app.errorLog):
92 class AboutDialog(object):
94 def __init__(self, app):
96 self._title = QtGui.QLabel(
97 "<h1>%s</h1><h3>Version: %s</h3>" % (
98 constants.__pretty_app_name__, constants.__version__
101 self._title.setTextFormat(QtCore.Qt.RichText)
102 self._title.setAlignment(QtCore.Qt.AlignCenter)
103 self._copyright = QtGui.QLabel("<h6>Developed by Ed Page<h6><h6>Icons: See website</h6>")
104 self._copyright.setTextFormat(QtCore.Qt.RichText)
105 self._copyright.setAlignment(QtCore.Qt.AlignCenter)
106 self._link = QtGui.QLabel('<a href="http://gc-dialer.garage.maemo.org">DialCentral Website</a>')
107 self._link.setTextFormat(QtCore.Qt.RichText)
108 self._link.setAlignment(QtCore.Qt.AlignCenter)
109 self._link.setOpenExternalLinks(True)
111 self._buttonLayout = QtGui.QDialogButtonBox(QtGui.QDialogButtonBox.Cancel)
113 self._layout = QtGui.QVBoxLayout()
114 self._layout.addWidget(self._title)
115 self._layout.addWidget(self._copyright)
116 self._layout.addWidget(self._link)
117 self._layout.addWidget(self._buttonLayout)
119 self._dialog = QtGui.QDialog()
120 self._dialog.setWindowTitle("About")
121 self._dialog.setLayout(self._layout)
122 self._buttonLayout.rejected.connect(self._dialog.reject)
124 self._closeWindowAction = QtGui.QAction(None)
125 self._closeWindowAction.setText("Close")
126 self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w"))
127 self._closeWindowAction.triggered.connect(self._on_close_window)
129 self._dialog.addAction(self._closeWindowAction)
130 self._dialog.addAction(app.quitAction)
131 self._dialog.addAction(app.fullscreenAction)
133 def run(self, parent=None):
134 self._dialog.setParent(parent, QtCore.Qt.Dialog)
136 response = self._dialog.exec_()
141 self._dialog.reject()
143 _moduleLogger.exception("Oh well")
146 @qt_compat.Slot(bool)
147 @misc_utils.log_exception(_moduleLogger)
148 def _on_close_window(self, checked = True):
149 with qui_utils.notify_error(self._app.errorLog):
150 self._dialog.reject()
153 class AccountDialog(QtCore.QObject, qwrappers.WindowWrapper):
155 # @bug Can't enter custom callback numbers
157 _RECURRENCE_CHOICES = [
173 ALARM_NONE = "No Alert"
174 ALARM_BACKGROUND = "Background Alert"
175 ALARM_APPLICATION = "Application Alert"
177 VOICEMAIL_CHECK_NOT_SUPPORTED = "Not Supported"
178 VOICEMAIL_CHECK_DISABLED = "Disabled"
179 VOICEMAIL_CHECK_ENABLED = "Enabled"
181 settingsApproved = qt_compat.Signal()
183 def __init__(self, parent, app, errorLog):
184 QtCore.QObject.__init__(self)
185 qwrappers.WindowWrapper.__init__(self, parent, app)
187 self._doClear = False
189 self._accountNumberLabel = QtGui.QLabel("NUMBER NOT SET")
190 self._notificationSelecter = QtGui.QComboBox()
191 self._notificationSelecter.currentIndexChanged.connect(self._on_notification_change)
192 self._notificationTimeSelector = QtGui.QComboBox()
193 #self._notificationTimeSelector.setEditable(True)
194 self._notificationTimeSelector.setInsertPolicy(QtGui.QComboBox.InsertAtTop)
195 for _, label in self._RECURRENCE_CHOICES:
196 self._notificationTimeSelector.addItem(label)
197 self._missedCallsNotificationButton = QtGui.QCheckBox("Missed Calls")
198 self._voicemailNotificationButton = QtGui.QCheckBox("Voicemail")
199 self._smsNotificationButton = QtGui.QCheckBox("SMS")
200 self._voicemailOnMissedButton = QtGui.QCheckBox("Voicemail Update on Missed Calls")
201 self._clearButton = QtGui.QPushButton("Clear Account")
202 self._clearButton.clicked.connect(self._on_clear)
203 self._callbackSelector = QtGui.QComboBox()
204 #self._callbackSelector.setEditable(True)
205 self._callbackSelector.setInsertPolicy(QtGui.QComboBox.InsertAtTop)
206 self._orientationSelector = QtGui.QComboBox()
207 for orientationMode in [
208 self._app.DEFAULT_ORIENTATION,
209 self._app.AUTO_ORIENTATION,
210 self._app.LANDSCAPE_ORIENTATION,
211 self._app.PORTRAIT_ORIENTATION,
213 self._orientationSelector.addItem(orientationMode)
215 self._update_notification_state()
217 self._credLayout = QtGui.QGridLayout()
218 self._credLayout.addWidget(QtGui.QLabel("Account"), 0, 0)
219 self._credLayout.addWidget(self._accountNumberLabel, 0, 1)
220 self._credLayout.addWidget(QtGui.QLabel("Callback"), 1, 0)
221 self._credLayout.addWidget(self._callbackSelector, 1, 1)
222 self._credLayout.addWidget(self._notificationSelecter, 2, 0)
223 self._credLayout.addWidget(self._notificationTimeSelector, 2, 1)
224 self._credLayout.addWidget(QtGui.QLabel(""), 3, 0)
225 self._credLayout.addWidget(self._missedCallsNotificationButton, 3, 1)
226 self._credLayout.addWidget(QtGui.QLabel(""), 4, 0)
227 self._credLayout.addWidget(self._voicemailNotificationButton, 4, 1)
228 self._credLayout.addWidget(QtGui.QLabel(""), 5, 0)
229 self._credLayout.addWidget(self._smsNotificationButton, 5, 1)
230 self._credLayout.addWidget(QtGui.QLabel("Other"), 6, 0)
231 self._credLayout.addWidget(self._voicemailOnMissedButton, 6, 1)
232 self._credLayout.addWidget(QtGui.QLabel("Orientation"), 7, 0)
233 self._credLayout.addWidget(self._orientationSelector, 7, 1)
234 self._credLayout.addWidget(QtGui.QLabel(""), 8, 0)
235 self._credLayout.addWidget(QtGui.QLabel(""), 9, 0)
236 self._credLayout.addWidget(self._clearButton, 9, 1)
238 self._credWidget = QtGui.QWidget()
239 self._credWidget.setLayout(self._credLayout)
240 self._credWidget.setContentsMargins(0, 0, 0, 0)
241 self._scrollSettings = QtGui.QScrollArea()
242 self._scrollSettings.setWidget(self._credWidget)
243 self._scrollSettings.setWidgetResizable(True)
244 self._scrollSettings.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
245 self._scrollSettings.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
247 self._applyButton = QtGui.QPushButton("&Apply")
248 self._applyButton.clicked.connect(self._on_settings_apply)
249 self._cancelButton = QtGui.QPushButton("&Cancel")
250 self._cancelButton.clicked.connect(self._on_settings_cancel)
251 self._buttonLayout = QtGui.QHBoxLayout()
252 self._buttonLayout.addStretch()
253 self._buttonLayout.addWidget(self._cancelButton)
254 self._buttonLayout.addStretch()
255 self._buttonLayout.addWidget(self._applyButton)
256 self._buttonLayout.addStretch()
258 self._layout.addWidget(self._scrollSettings)
259 self._layout.addLayout(self._buttonLayout)
260 self._layout.setDirection(QtGui.QBoxLayout.TopToBottom)
262 self._window.setWindowTitle("Account")
263 self._window.setAttribute(QtCore.Qt.WA_DeleteOnClose, False)
269 def setIfNotificationsSupported(self, isSupported):
271 self._notificationSelecter.clear()
272 self._notificationSelecter.addItems([self.ALARM_NONE, self.ALARM_APPLICATION, self.ALARM_BACKGROUND])
273 self._notificationTimeSelector.setEnabled(False)
274 self._missedCallsNotificationButton.setEnabled(False)
275 self._voicemailNotificationButton.setEnabled(False)
276 self._smsNotificationButton.setEnabled(False)
278 self._notificationSelecter.clear()
279 self._notificationSelecter.addItems([self.ALARM_NONE, self.ALARM_APPLICATION])
280 self._notificationTimeSelector.setEnabled(False)
281 self._missedCallsNotificationButton.setEnabled(False)
282 self._voicemailNotificationButton.setEnabled(False)
283 self._smsNotificationButton.setEnabled(False)
285 def set_account_number(self, num):
286 self._accountNumberLabel.setText(num)
288 orientation = property(
289 lambda self: str(self._orientationSelector.currentText()),
290 lambda self, mode: qui_utils.set_current_index(self._orientationSelector, mode),
293 def _set_voicemail_on_missed(self, status):
294 if status == self.VOICEMAIL_CHECK_NOT_SUPPORTED:
295 self._voicemailOnMissedButton.setChecked(False)
296 self._voicemailOnMissedButton.hide()
297 elif status == self.VOICEMAIL_CHECK_DISABLED:
298 self._voicemailOnMissedButton.setChecked(False)
299 self._voicemailOnMissedButton.show()
300 elif status == self.VOICEMAIL_CHECK_ENABLED:
301 self._voicemailOnMissedButton.setChecked(True)
302 self._voicemailOnMissedButton.show()
304 raise RuntimeError("Unsupported option for updating voicemail on missed calls %r" % status)
306 def _get_voicemail_on_missed(self):
307 if not self._voicemailOnMissedButton.isVisible():
308 return self.VOICEMAIL_CHECK_NOT_SUPPORTED
309 elif self._voicemailOnMissedButton.isChecked():
310 return self.VOICEMAIL_CHECK_ENABLED
312 return self.VOICEMAIL_CHECK_DISABLED
314 updateVMOnMissedCall = property(_get_voicemail_on_missed, _set_voicemail_on_missed)
316 notifications = property(
317 lambda self: str(self._notificationSelecter.currentText()),
318 lambda self, enabled: qui_utils.set_current_index(self._notificationSelecter, enabled),
321 notifyOnMissed = property(
322 lambda self: self._missedCallsNotificationButton.isChecked(),
323 lambda self, enabled: self._missedCallsNotificationButton.setChecked(enabled),
326 notifyOnVoicemail = property(
327 lambda self: self._voicemailNotificationButton.isChecked(),
328 lambda self, enabled: self._voicemailNotificationButton.setChecked(enabled),
331 notifyOnSms = property(
332 lambda self: self._smsNotificationButton.isChecked(),
333 lambda self, enabled: self._smsNotificationButton.setChecked(enabled),
336 def _get_notification_time(self):
337 index = self._notificationTimeSelector.currentIndex()
338 minutes = self._RECURRENCE_CHOICES[index][0]
341 def _set_notification_time(self, minutes):
342 for i, (time, _) in enumerate(self._RECURRENCE_CHOICES):
344 self._notificationTimeSelector.setCurrentIndex(i)
347 self._notificationTimeSelector.setCurrentIndex(0)
349 notificationTime = property(_get_notification_time, _set_notification_time)
352 def selectedCallback(self):
353 index = self._callbackSelector.currentIndex()
354 data = str(self._callbackSelector.itemData(index))
357 def set_callbacks(self, choices, default):
358 self._callbackSelector.clear()
360 self._callbackSelector.addItem("Not Set", "")
362 uglyDefault = misc_utils.make_ugly(default)
364 uglyDefault = default
365 for number, description in choices.iteritems():
366 prettyNumber = misc_utils.make_pretty(number)
367 uglyNumber = misc_utils.make_ugly(number)
369 prettyNumber = number
372 self._callbackSelector.addItem("%s - %s" % (prettyNumber, description), uglyNumber)
373 if uglyNumber == uglyDefault:
374 self._callbackSelector.setCurrentIndex(self._callbackSelector.count() - 1)
377 self._doClear = False
384 _moduleLogger.exception("Oh well")
386 def _update_notification_state(self):
387 currentText = str(self._notificationSelecter.currentText())
388 if currentText == self.ALARM_BACKGROUND:
389 self._notificationTimeSelector.setEnabled(True)
391 self._missedCallsNotificationButton.setEnabled(True)
392 self._voicemailNotificationButton.setEnabled(True)
393 self._smsNotificationButton.setEnabled(True)
394 elif currentText == self.ALARM_APPLICATION:
395 self._notificationTimeSelector.setEnabled(True)
397 self._missedCallsNotificationButton.setEnabled(False)
398 self._voicemailNotificationButton.setEnabled(True)
399 self._smsNotificationButton.setEnabled(True)
401 self._missedCallsNotificationButton.setChecked(False)
403 self._notificationTimeSelector.setEnabled(False)
405 self._missedCallsNotificationButton.setEnabled(False)
406 self._voicemailNotificationButton.setEnabled(False)
407 self._smsNotificationButton.setEnabled(False)
409 self._missedCallsNotificationButton.setChecked(False)
410 self._voicemailNotificationButton.setChecked(False)
411 self._smsNotificationButton.setChecked(False)
414 @misc_utils.log_exception(_moduleLogger)
415 def _on_notification_change(self, index):
416 with qui_utils.notify_error(self._app.errorLog):
417 self._update_notification_state()
420 @qt_compat.Slot(bool)
421 @misc_utils.log_exception(_moduleLogger)
422 def _on_settings_cancel(self, checked = False):
423 with qui_utils.notify_error(self._app.errorLog):
427 @qt_compat.Slot(bool)
428 def _on_settings_apply(self, checked = False):
429 self.__on_settings_apply(checked)
431 @misc_utils.log_exception(_moduleLogger)
432 def __on_settings_apply(self, checked = False):
433 with qui_utils.notify_error(self._app.errorLog):
434 self.settingsApproved.emit()
438 @qt_compat.Slot(bool)
439 @misc_utils.log_exception(_moduleLogger)
440 def _on_clear(self, checked = False):
441 with qui_utils.notify_error(self._app.errorLog):
443 self.settingsApproved.emit()
447 class ContactList(object):
449 _SENTINEL_ICON = QtGui.QIcon()
451 def __init__(self, app, session):
453 self._session = session
454 self._targetLayout = QtGui.QVBoxLayout()
455 self._targetList = QtGui.QWidget()
456 self._targetList.setLayout(self._targetLayout)
458 self._closeIcon = qui_utils.get_theme_icon(("window-close", "general_close", "gtk-close"), self._SENTINEL_ICON)
462 return self._targetList
464 def setVisible(self, isVisible):
465 self._targetList.setVisible(isVisible)
468 cids = list(self._session.draft.get_contacts())
469 amountCommon = min(len(cids), len(self._uiItems))
471 # Run through everything in common
472 for i in xrange(0, amountCommon):
474 uiItem = self._uiItems[i]
475 title = self._session.draft.get_title(cid)
476 description = self._session.draft.get_description(cid)
477 numbers = self._session.draft.get_numbers(cid)
479 uiItem["title"] = title
480 uiItem["description"] = description
481 uiItem["numbers"] = numbers
482 uiItem["label"].setText(title)
483 self._populate_number_selector(uiItem["selector"], cid, i, numbers)
484 uiItem["rowWidget"].setVisible(True)
486 # More contacts than ui items
487 for i in xrange(amountCommon, len(cids)):
489 title = self._session.draft.get_title(cid)
490 description = self._session.draft.get_description(cid)
491 numbers = self._session.draft.get_numbers(cid)
493 titleLabel = QtGui.QLabel(title)
494 titleLabel.setWordWrap(True)
495 numberSelector = QtGui.QComboBox()
496 self._populate_number_selector(numberSelector, cid, i, numbers)
498 callback = functools.partial(
499 self._on_change_number,
502 callback.__name__ = "thanks partials for not having names and pyqt for requiring them"
503 numberSelector.activated.connect(
504 qt_compat.Slot(int)(callback)
507 if self._closeIcon is self._SENTINEL_ICON:
508 deleteButton = QtGui.QPushButton("Delete")
510 deleteButton = QtGui.QPushButton(self._closeIcon, "")
511 deleteButton.setSizePolicy(QtGui.QSizePolicy(
512 QtGui.QSizePolicy.Minimum,
513 QtGui.QSizePolicy.Minimum,
514 QtGui.QSizePolicy.PushButton,
516 callback = functools.partial(
517 self._on_remove_contact,
520 callback.__name__ = "thanks partials for not having names and pyqt for requiring them"
521 deleteButton.clicked.connect(callback)
523 rowLayout = QtGui.QHBoxLayout()
524 rowLayout.addWidget(titleLabel, 1000)
525 rowLayout.addWidget(numberSelector, 0)
526 rowLayout.addWidget(deleteButton, 0)
527 rowWidget = QtGui.QWidget()
528 rowWidget.setLayout(rowLayout)
529 self._targetLayout.addWidget(rowWidget)
533 uiItem["title"] = title
534 uiItem["description"] = description
535 uiItem["numbers"] = numbers
536 uiItem["label"] = titleLabel
537 uiItem["selector"] = numberSelector
538 uiItem["rowWidget"] = rowWidget
539 self._uiItems.append(uiItem)
542 # More UI items than contacts
543 for i in xrange(amountCommon, len(self._uiItems)):
544 uiItem = self._uiItems[i]
545 uiItem["rowWidget"].setVisible(False)
548 def _populate_number_selector(self, selector, cid, cidIndex, numbers):
551 selectedNumber = self._session.draft.get_selected_number(cid)
552 if len(numbers) == 1:
553 # If no alt numbers available, check the address book
554 numbers, defaultIndex = _get_contact_numbers(self._session, cid, selectedNumber, numbers[0][1])
556 defaultIndex = _index_number(numbers, selectedNumber)
558 for number, description in numbers:
560 label = "%s - %s" % (number, description)
563 selector.addItem(label)
564 selector.setVisible(True)
566 selector.setEnabled(True)
567 selector.setCurrentIndex(defaultIndex)
569 selector.setEnabled(False)
571 @misc_utils.log_exception(_moduleLogger)
572 def _on_change_number(self, cidIndex, index):
573 with qui_utils.notify_error(self._app.errorLog):
574 # Exception thrown when the first item is removed
576 cid = self._uiItems[cidIndex]["cid"]
577 numbers = self._session.draft.get_numbers(cid)
579 _moduleLogger.error("Contact no longer available (or bizarre error): %r (%r)" % (cid, index))
582 _moduleLogger.error("Contact no longer available (or bizarre error): %r (%r)" % (cid, index))
584 number = numbers[index][0]
585 self._session.draft.set_selected_number(cid, number)
587 @misc_utils.log_exception(_moduleLogger)
588 def _on_remove_contact(self, index, toggled):
589 with qui_utils.notify_error(self._app.errorLog):
590 self._session.draft.remove_contact(self._uiItems[index]["cid"])
593 class VoicemailPlayer(object):
595 def __init__(self, app, session, errorLog):
597 self._session = session
598 self._errorLog = errorLog
600 self._session.voicemailAvailable.connect(self._on_voicemail_downloaded)
601 self._session.draft.recipientsChanged.connect(self._on_recipients_changed)
603 self._playButton = QtGui.QPushButton("Play")
604 self._playButton.clicked.connect(self._on_voicemail_play)
605 self._pauseButton = QtGui.QPushButton("Pause")
606 self._pauseButton.clicked.connect(self._on_voicemail_pause)
607 self._pauseButton.hide()
608 self._resumeButton = QtGui.QPushButton("Resume")
609 self._resumeButton.clicked.connect(self._on_voicemail_resume)
610 self._resumeButton.hide()
611 self._stopButton = QtGui.QPushButton("Stop")
612 self._stopButton.clicked.connect(self._on_voicemail_stop)
613 self._stopButton.hide()
615 self._downloadButton = QtGui.QPushButton("Download Voicemail")
616 self._downloadButton.clicked.connect(self._on_voicemail_download)
617 self._downloadLayout = QtGui.QHBoxLayout()
618 self._downloadLayout.addWidget(self._downloadButton)
619 self._downloadWidget = QtGui.QWidget()
620 self._downloadWidget.setLayout(self._downloadLayout)
622 self._playLabel = QtGui.QLabel("Voicemail")
623 self._saveButton = QtGui.QPushButton("Save")
624 self._saveButton.clicked.connect(self._on_voicemail_save)
625 self._playerLayout = QtGui.QHBoxLayout()
626 self._playerLayout.addWidget(self._playLabel)
627 self._playerLayout.addWidget(self._playButton)
628 self._playerLayout.addWidget(self._pauseButton)
629 self._playerLayout.addWidget(self._resumeButton)
630 self._playerLayout.addWidget(self._stopButton)
631 self._playerLayout.addWidget(self._saveButton)
632 self._playerWidget = QtGui.QWidget()
633 self._playerWidget.setLayout(self._playerLayout)
635 self._visibleWidget = None
636 self._layout = QtGui.QHBoxLayout()
637 self._layout.setContentsMargins(0, 0, 0, 0)
638 self._widget = QtGui.QWidget()
639 self._widget.setLayout(self._layout)
647 self._session.voicemailAvailable.disconnect(self._on_voicemail_downloaded)
648 self._session.draft.recipientsChanged.disconnect(self._on_recipients_changed)
649 self._invalidate_token()
651 def _invalidate_token(self):
652 if self._token is not None:
653 self._token.invalidate()
654 self._token.error.disconnect(self._on_play_error)
655 self._token.stateChange.connect(self._on_play_state)
656 self._token.invalidated.connect(self._on_play_invalidated)
658 def _show_download(self, messageId):
659 if self._visibleWidget is self._downloadWidget:
662 self._layout.addWidget(self._downloadWidget)
663 self._visibleWidget = self._downloadWidget
664 self._visibleWidget.show()
666 def _show_player(self, messageId):
667 if self._visibleWidget is self._playerWidget:
670 self._layout.addWidget(self._playerWidget)
671 self._visibleWidget = self._playerWidget
672 self._visibleWidget.show()
675 if self._visibleWidget is None:
677 self._visibleWidget.hide()
678 self._layout.removeWidget(self._visibleWidget)
679 self._visibleWidget = None
681 def _update_play_state(self):
682 if self._token is not None and self._token.isValid:
683 self._playButton.setText("Stop")
685 self._playButton.setText("Play")
687 def _update_state(self):
688 if self._session.draft.get_num_contacts() != 1:
692 (cid, ) = self._session.draft.get_contacts()
693 messageId = self._session.draft.get_message_id(cid)
694 if messageId is None:
698 if self._session.is_available(messageId):
699 self._show_player(messageId)
701 self._show_download(messageId)
702 if self._token is not None:
703 self._token.invalidate()
705 @misc_utils.log_exception(_moduleLogger)
706 def _on_voicemail_save(self, arg):
707 with qui_utils.notify_error(self._app.errorLog):
708 targetPath = QtGui.QFileDialog.getSaveFileName(None, caption="Save Voicemail", filter="Audio File (*.mp3)")
709 targetPath = unicode(targetPath)
713 (cid, ) = self._session.draft.get_contacts()
714 messageId = self._session.draft.get_message_id(cid)
715 sourcePath = self._session.voicemail_path(messageId)
717 shutil.copy2(sourcePath, targetPath)
719 @misc_utils.log_exception(_moduleLogger)
720 def _on_play_error(self, error):
721 with qui_utils.notify_error(self._app.errorLog):
722 self._app.errorLog.push_error(error)
724 @misc_utils.log_exception(_moduleLogger)
725 def _on_play_invalidated(self):
726 with qui_utils.notify_error(self._app.errorLog):
727 self._playButton.show()
728 self._pauseButton.hide()
729 self._resumeButton.hide()
730 self._stopButton.hide()
731 self._invalidate_token()
733 @misc_utils.log_exception(_moduleLogger)
734 def _on_play_state(self, state):
735 with qui_utils.notify_error(self._app.errorLog):
736 if state == self._token.STATE_PLAY:
737 self._playButton.hide()
738 self._pauseButton.show()
739 self._resumeButton.hide()
740 self._stopButton.show()
741 elif state == self._token.STATE_PAUSE:
742 self._playButton.hide()
743 self._pauseButton.hide()
744 self._resumeButton.show()
745 self._stopButton.show()
746 elif state == self._token.STATE_STOP:
747 self._playButton.show()
748 self._pauseButton.hide()
749 self._resumeButton.hide()
750 self._stopButton.hide()
752 @misc_utils.log_exception(_moduleLogger)
753 def _on_voicemail_play(self, arg):
754 with qui_utils.notify_error(self._app.errorLog):
755 (cid, ) = self._session.draft.get_contacts()
756 messageId = self._session.draft.get_message_id(cid)
757 sourcePath = self._session.voicemail_path(messageId)
759 self._invalidate_token()
760 uri = "file://%s" % sourcePath
761 self._token = self._app.streamHandler.set_file(uri)
762 self._token.stateChange.connect(self._on_play_state)
763 self._token.invalidated.connect(self._on_play_invalidated)
764 self._token.error.connect(self._on_play_error)
767 @misc_utils.log_exception(_moduleLogger)
768 def _on_voicemail_pause(self, arg):
769 with qui_utils.notify_error(self._app.errorLog):
772 @misc_utils.log_exception(_moduleLogger)
773 def _on_voicemail_resume(self, arg):
774 with qui_utils.notify_error(self._app.errorLog):
777 @misc_utils.log_exception(_moduleLogger)
778 def _on_voicemail_stop(self, arg):
779 with qui_utils.notify_error(self._app.errorLog):
782 @misc_utils.log_exception(_moduleLogger)
783 def _on_voicemail_download(self, arg):
784 with qui_utils.notify_error(self._app.errorLog):
785 (cid, ) = self._session.draft.get_contacts()
786 messageId = self._session.draft.get_message_id(cid)
787 self._session.download_voicemail(messageId)
791 @misc_utils.log_exception(_moduleLogger)
792 def _on_recipients_changed(self):
793 with qui_utils.notify_error(self._app.errorLog):
796 @qt_compat.Slot(str, str)
797 @misc_utils.log_exception(_moduleLogger)
798 def _on_voicemail_downloaded(self, messageId, filepath):
799 with qui_utils.notify_error(self._app.errorLog):
803 class SMSEntryWindow(qwrappers.WindowWrapper):
806 # @bug Somehow a window is being destroyed on object creation which causes glitches on Maemo 5
808 def __init__(self, parent, app, session, errorLog):
809 qwrappers.WindowWrapper.__init__(self, parent, app)
810 self._session = session
811 self._session.messagesUpdated.connect(self._on_refresh_history)
812 self._session.historyUpdated.connect(self._on_refresh_history)
813 self._session.draft.recipientsChanged.connect(self._on_recipients_changed)
815 self._session.draft.sendingMessage.connect(self._on_op_started)
816 self._session.draft.calling.connect(self._on_op_started)
817 self._session.draft.calling.connect(self._on_calling_started)
818 self._session.draft.cancelling.connect(self._on_op_started)
820 self._session.draft.sentMessage.connect(self._on_op_finished)
821 self._session.draft.called.connect(self._on_op_finished)
822 self._session.draft.cancelled.connect(self._on_op_finished)
823 self._session.draft.error.connect(self._on_op_error)
825 self._errorLog = errorLog
827 self._targetList = ContactList(self._app, self._session)
828 self._history = QtGui.QLabel()
829 self._history.setTextFormat(QtCore.Qt.RichText)
830 self._history.setWordWrap(True)
831 self._voicemailPlayer = VoicemailPlayer(self._app, self._session, self._errorLog)
832 self._smsEntry = QtGui.QTextEdit()
833 self._smsEntry.textChanged.connect(self._on_letter_count_changed)
835 self._entryLayout = QtGui.QVBoxLayout()
836 self._entryLayout.addWidget(self._targetList.toplevel)
837 self._entryLayout.addWidget(self._history)
838 self._entryLayout.addWidget(self._voicemailPlayer.toplevel, 0)
839 self._entryLayout.addWidget(self._smsEntry)
840 self._entryLayout.setContentsMargins(0, 0, 0, 0)
841 self._entryWidget = QtGui.QWidget()
842 self._entryWidget.setLayout(self._entryLayout)
843 self._entryWidget.setContentsMargins(0, 0, 0, 0)
844 self._scrollEntry = QtGui.QScrollArea()
845 self._scrollEntry.setWidget(self._entryWidget)
846 self._scrollEntry.setWidgetResizable(True)
847 self._scrollEntry.setAlignment(QtCore.Qt.AlignLeft | QtCore.Qt.AlignBottom)
848 self._scrollEntry.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded)
849 self._scrollEntry.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
851 self._characterCountLabel = QtGui.QLabel("")
852 self._singleNumberSelector = QtGui.QComboBox()
854 self._singleNumberSelector.activated.connect(self._on_single_change_number)
855 self._smsButton = QtGui.QPushButton("SMS")
856 self._smsButton.clicked.connect(self._on_sms_clicked)
857 self._smsButton.setEnabled(False)
858 self._dialButton = QtGui.QPushButton("Dial")
859 self._dialButton.clicked.connect(self._on_call_clicked)
860 self._cancelButton = QtGui.QPushButton("Cancel Call")
861 self._cancelButton.clicked.connect(self._on_cancel_clicked)
862 self._cancelButton.setVisible(False)
864 self._buttonLayout = QtGui.QHBoxLayout()
865 self._buttonLayout.addWidget(self._characterCountLabel)
866 self._buttonLayout.addStretch()
867 self._buttonLayout.addWidget(self._singleNumberSelector)
868 self._buttonLayout.addStretch()
869 self._buttonLayout.addWidget(self._smsButton)
870 self._buttonLayout.addWidget(self._dialButton)
871 self._buttonLayout.addWidget(self._cancelButton)
873 self._layout.addWidget(self._errorDisplay.toplevel)
874 self._layout.addWidget(self._scrollEntry)
875 self._layout.addLayout(self._buttonLayout)
876 self._layout.setDirection(QtGui.QBoxLayout.TopToBottom)
878 self._window.setWindowTitle("Contact")
879 self._window.closed.connect(self._on_close_window)
880 self._window.hidden.connect(self._on_close_window)
881 self._window.resized.connect(self._on_window_resized)
883 self._scrollTimer = QtCore.QTimer()
884 self._scrollTimer.setInterval(100)
885 self._scrollTimer.setSingleShot(True)
886 self._scrollTimer.timeout.connect(self._on_delayed_scroll_to_bottom)
888 self._smsEntry.setPlainText(self._session.draft.message)
889 self._update_letter_count()
890 self._update_target_fields()
891 self.set_fullscreen(self._app.fullscreenAction.isChecked())
892 self.update_orientation(self._app.orientation)
895 if self._window is None:
898 window = self._window
900 message = unicode(self._smsEntry.toPlainText())
901 self._session.draft.message = message
903 except AttributeError:
904 _moduleLogger.exception("Oh well")
906 _moduleLogger.exception("Oh well")
909 self._session.messagesUpdated.disconnect(self._on_refresh_history)
910 self._session.historyUpdated.disconnect(self._on_refresh_history)
911 self._session.draft.recipientsChanged.disconnect(self._on_recipients_changed)
912 self._session.draft.sendingMessage.disconnect(self._on_op_started)
913 self._session.draft.calling.disconnect(self._on_op_started)
914 self._session.draft.calling.disconnect(self._on_calling_started)
915 self._session.draft.cancelling.disconnect(self._on_op_started)
916 self._session.draft.sentMessage.disconnect(self._on_op_finished)
917 self._session.draft.called.disconnect(self._on_op_finished)
918 self._session.draft.cancelled.disconnect(self._on_op_finished)
919 self._session.draft.error.disconnect(self._on_op_error)
920 self._voicemailPlayer.destroy()
921 window = self._window
926 except AttributeError:
927 _moduleLogger.exception("Oh well")
929 _moduleLogger.exception("Oh well")
931 def update_orientation(self, orientation):
932 qwrappers.WindowWrapper.update_orientation(self, orientation)
933 self._scroll_to_bottom()
935 def _update_letter_count(self):
936 count = len(self._smsEntry.toPlainText())
937 numTexts, numCharInText = divmod(count, self.MAX_CHAR)
939 numCharsLeftInText = self.MAX_CHAR - numCharInText
940 self._characterCountLabel.setText("%d (%d)" % (numCharsLeftInText, numTexts))
942 def _update_button_state(self):
943 self._cancelButton.setEnabled(True)
944 if self._session.draft.get_num_contacts() == 0:
945 self._dialButton.setEnabled(False)
946 self._smsButton.setEnabled(False)
947 elif self._session.draft.get_num_contacts() == 1:
948 count = len(self._smsEntry.toPlainText())
950 self._dialButton.setEnabled(True)
951 self._smsButton.setEnabled(False)
953 self._dialButton.setEnabled(False)
954 self._smsButton.setEnabled(True)
956 self._dialButton.setEnabled(False)
957 count = len(self._smsEntry.toPlainText())
959 self._smsButton.setEnabled(False)
961 self._smsButton.setEnabled(True)
963 def _update_history(self, cid):
964 draftContactsCount = self._session.draft.get_num_contacts()
965 if draftContactsCount != 1:
966 self._history.setVisible(False)
968 description = self._session.draft.get_description(cid)
970 self._targetList.setVisible(False)
972 self._history.setText(description)
973 self._history.setVisible(True)
975 self._history.setText("")
976 self._history.setVisible(False)
978 def _update_target_fields(self):
979 draftContactsCount = self._session.draft.get_num_contacts()
980 if draftContactsCount == 0:
983 elif draftContactsCount == 1:
984 (cid, ) = self._session.draft.get_contacts()
985 title = self._session.draft.get_title(cid)
986 numbers = self._session.draft.get_numbers(cid)
988 self._targetList.setVisible(False)
989 self._update_history(cid)
990 self._populate_number_selector(self._singleNumberSelector, cid, 0, numbers)
993 self._scroll_to_bottom()
994 self._window.setWindowTitle(title)
995 self._smsEntry.setFocus(QtCore.Qt.OtherFocusReason)
997 self._window.raise_()
999 self._targetList.setVisible(True)
1000 self._targetList.update()
1001 self._history.setText("")
1002 self._history.setVisible(False)
1003 self._singleNumberSelector.setVisible(False)
1005 self._scroll_to_bottom()
1006 self._window.setWindowTitle("Contacts")
1007 self._smsEntry.setFocus(QtCore.Qt.OtherFocusReason)
1009 self._window.raise_()
1011 def _populate_number_selector(self, selector, cid, cidIndex, numbers):
1014 selectedNumber = self._session.draft.get_selected_number(cid)
1015 if len(numbers) == 1:
1016 # If no alt numbers available, check the address book
1017 numbers, defaultIndex = _get_contact_numbers(self._session, cid, selectedNumber, numbers[0][1])
1019 defaultIndex = _index_number(numbers, selectedNumber)
1021 for number, description in numbers:
1023 label = "%s - %s" % (number, description)
1026 selector.addItem(label)
1027 selector.setVisible(True)
1028 if 1 < len(numbers):
1029 selector.setEnabled(True)
1030 selector.setCurrentIndex(defaultIndex)
1032 selector.setEnabled(False)
1034 def _scroll_to_bottom(self):
1035 self._scrollTimer.start()
1037 @misc_utils.log_exception(_moduleLogger)
1038 def _on_delayed_scroll_to_bottom(self):
1039 with qui_utils.notify_error(self._app.errorLog):
1040 self._scrollEntry.ensureWidgetVisible(self._smsEntry)
1042 @misc_utils.log_exception(_moduleLogger)
1043 def _on_sms_clicked(self, arg):
1044 with qui_utils.notify_error(self._app.errorLog):
1045 message = unicode(self._smsEntry.toPlainText())
1046 self._session.draft.message = message
1047 self._session.draft.send()
1049 @misc_utils.log_exception(_moduleLogger)
1050 def _on_call_clicked(self, arg):
1051 with qui_utils.notify_error(self._app.errorLog):
1052 message = unicode(self._smsEntry.toPlainText())
1053 self._session.draft.message = message
1054 self._session.draft.call()
1057 @misc_utils.log_exception(_moduleLogger)
1058 def _on_cancel_clicked(self, message):
1059 with qui_utils.notify_error(self._app.errorLog):
1060 self._session.draft.cancel()
1062 @misc_utils.log_exception(_moduleLogger)
1063 def _on_single_change_number(self, index):
1064 with qui_utils.notify_error(self._app.errorLog):
1065 # Exception thrown when the first item is removed
1068 numbers = self._session.draft.get_numbers(cid)
1070 _moduleLogger.error("Contact no longer available (or bizarre error): %r (%r)" % (cid, index))
1072 number = numbers[index][0]
1073 self._session.draft.set_selected_number(cid, number)
1076 @misc_utils.log_exception(_moduleLogger)
1077 def _on_refresh_history(self):
1078 with qui_utils.notify_error(self._app.errorLog):
1079 draftContactsCount = self._session.draft.get_num_contacts()
1080 if draftContactsCount != 1:
1081 # Changing contact count will automatically refresh it
1083 (cid, ) = self._session.draft.get_contacts()
1084 self._update_history(cid)
1087 @misc_utils.log_exception(_moduleLogger)
1088 def _on_recipients_changed(self):
1089 with qui_utils.notify_error(self._app.errorLog):
1090 self._update_target_fields()
1091 self._update_button_state()
1094 @misc_utils.log_exception(_moduleLogger)
1095 def _on_op_started(self):
1096 with qui_utils.notify_error(self._app.errorLog):
1097 self._smsEntry.setReadOnly(True)
1098 self._smsButton.setVisible(False)
1099 self._dialButton.setVisible(False)
1103 @misc_utils.log_exception(_moduleLogger)
1104 def _on_calling_started(self):
1105 with qui_utils.notify_error(self._app.errorLog):
1106 self._cancelButton.setVisible(True)
1109 @misc_utils.log_exception(_moduleLogger)
1110 def _on_op_finished(self):
1111 with qui_utils.notify_error(self._app.errorLog):
1112 self._smsEntry.setPlainText("")
1113 self._smsEntry.setReadOnly(False)
1114 self._cancelButton.setVisible(False)
1115 self._smsButton.setVisible(True)
1116 self._dialButton.setVisible(True)
1121 @misc_utils.log_exception(_moduleLogger)
1122 def _on_op_error(self, message):
1123 with qui_utils.notify_error(self._app.errorLog):
1124 self._smsEntry.setReadOnly(False)
1125 self._cancelButton.setVisible(False)
1126 self._smsButton.setVisible(True)
1127 self._dialButton.setVisible(True)
1129 self._errorLog.push_error(message)
1132 @misc_utils.log_exception(_moduleLogger)
1133 def _on_letter_count_changed(self):
1134 with qui_utils.notify_error(self._app.errorLog):
1135 self._update_letter_count()
1136 self._update_button_state()
1139 @misc_utils.log_exception(_moduleLogger)
1140 def _on_window_resized(self):
1141 with qui_utils.notify_error(self._app.errorLog):
1142 self._scroll_to_bottom()
1145 @qt_compat.Slot(bool)
1146 @misc_utils.log_exception(_moduleLogger)
1147 def _on_close_window(self, checked = True):
1148 with qui_utils.notify_error(self._app.errorLog):
1152 def _index_number(numbers, default):
1153 uglyDefault = misc_utils.make_ugly(default)
1154 uglyContactNumbers = list(
1155 misc_utils.make_ugly(contactNumber)
1156 for (contactNumber, _) in numbers
1159 misc_utils.similar_ugly_numbers(uglyDefault, contactNumber)
1160 for contactNumber in uglyContactNumbers
1163 defaultIndex = defaultMatches.index(True)
1167 "Could not find contact number %s among %r" % (
1174 def _get_contact_numbers(session, contactId, number, description):
1175 contactPhoneNumbers = []
1176 if contactId and contactId != "0":
1178 contactDetails = copy.deepcopy(session.get_contacts()[contactId])
1179 contactPhoneNumbers = contactDetails["numbers"]
1181 contactPhoneNumbers = []
1182 contactPhoneNumbers = [
1183 (contactPhoneNumber["phoneNumber"], contactPhoneNumber.get("phoneType", "Unknown"))
1184 for contactPhoneNumber in contactPhoneNumbers
1186 defaultIndex = _index_number(contactPhoneNumbers, number)
1188 if not contactPhoneNumbers or defaultIndex == -1:
1189 contactPhoneNumbers += [(number, description)]
1192 return contactPhoneNumbers, defaultIndex