X-Git-Url: http://git.maemo.org/git/?p=gc-dialer;a=blobdiff_plain;f=src%2Fdialogs.py;h=0e1a92e122423bdd14378b9e6c206e9f4e65d9da;hp=5f8bb01fc691d9f130211ee8355069b46dc61f89;hb=d328b32038487b5056c0f3b4e19c7f1e9159958d;hpb=2226f829043ccf106da22c31d77d56a737a1482c diff --git a/src/dialogs.py b/src/dialogs.py index 5f8bb01..0e1a92e 100644 --- a/src/dialogs.py +++ b/src/dialogs.py @@ -7,9 +7,12 @@ import functools import copy import logging -from PyQt4 import QtGui -from PyQt4 import QtCore +import util.qt_compat as qt_compat +QtCore = qt_compat.QtCore +QtGui = qt_compat.import_module("QtGui") +import constants +from util import qwrappers from util import qui_utils from util import misc as misc_utils @@ -20,9 +23,10 @@ _moduleLogger = logging.getLogger(__name__) class CredentialsDialog(object): def __init__(self, app): + self._app = app self._usernameField = QtGui.QLineEdit() self._passwordField = QtGui.QLineEdit() - self._passwordField.setEchoMode(QtGui.QLineEdit.PasswordEchoOnEdit) + self._passwordField.setEchoMode(QtGui.QLineEdit.Password) self._credLayout = QtGui.QGridLayout() self._credLayout.addWidget(QtGui.QLabel("Username"), 0, 0) @@ -42,7 +46,6 @@ class CredentialsDialog(object): self._dialog.setWindowTitle("Login") self._dialog.setLayout(self._layout) self._dialog.setAttribute(QtCore.Qt.WA_DeleteOnClose, False) - qui_utils.set_autorient(self._dialog, True) self._buttonLayout.accepted.connect(self._dialog.accept) self._buttonLayout.rejected.connect(self._dialog.reject) @@ -65,50 +68,178 @@ class CredentialsDialog(object): if response == QtGui.QDialog.Accepted: return str(self._usernameField.text()), str(self._passwordField.text()) elif response == QtGui.QDialog.Rejected: - raise RuntimeError("Login Cancelled") + return None else: - raise RuntimeError("Unknown Response") + _moduleLogger.error("Unknown response") + return None finally: self._dialog.setParent(None, QtCore.Qt.Dialog) - @QtCore.pyqtSlot() - @QtCore.pyqtSlot(bool) + def close(self): + try: + self._dialog.reject() + except RuntimeError: + _moduleLogger.exception("Oh well") + + @qt_compat.Slot() + @qt_compat.Slot(bool) + @misc_utils.log_exception(_moduleLogger) + def _on_close_window(self, checked = True): + with qui_utils.notify_error(self._app.errorLog): + self._dialog.reject() + + +class AboutDialog(object): + + def __init__(self, app): + self._app = app + self._title = QtGui.QLabel( + "

%s

Version: %s

" % ( + constants.__pretty_app_name__, constants.__version__ + ) + ) + self._title.setTextFormat(QtCore.Qt.RichText) + self._title.setAlignment(QtCore.Qt.AlignCenter) + self._copyright = QtGui.QLabel("
Developed by Ed Page
Icons: See website
") + self._copyright.setTextFormat(QtCore.Qt.RichText) + self._copyright.setAlignment(QtCore.Qt.AlignCenter) + self._link = QtGui.QLabel('DialCentral Website') + self._link.setTextFormat(QtCore.Qt.RichText) + self._link.setAlignment(QtCore.Qt.AlignCenter) + self._link.setOpenExternalLinks(True) + + self._buttonLayout = QtGui.QDialogButtonBox(QtGui.QDialogButtonBox.Cancel) + + self._layout = QtGui.QVBoxLayout() + self._layout.addWidget(self._title) + self._layout.addWidget(self._copyright) + self._layout.addWidget(self._link) + self._layout.addWidget(self._buttonLayout) + + self._dialog = QtGui.QDialog() + self._dialog.setWindowTitle("About") + self._dialog.setLayout(self._layout) + self._buttonLayout.rejected.connect(self._dialog.reject) + + self._closeWindowAction = QtGui.QAction(None) + self._closeWindowAction.setText("Close") + self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w")) + self._closeWindowAction.triggered.connect(self._on_close_window) + + self._dialog.addAction(self._closeWindowAction) + self._dialog.addAction(app.quitAction) + self._dialog.addAction(app.fullscreenAction) + + def run(self, parent=None): + self._dialog.setParent(parent, QtCore.Qt.Dialog) + + response = self._dialog.exec_() + return response + + def close(self): + try: + self._dialog.reject() + except RuntimeError: + _moduleLogger.exception("Oh well") + + @qt_compat.Slot() + @qt_compat.Slot(bool) @misc_utils.log_exception(_moduleLogger) def _on_close_window(self, checked = True): - self._dialog.reject() + with qui_utils.notify_error(self._app.errorLog): + self._dialog.reject() class AccountDialog(object): - # @bug Can't configure callback number + # @bug Can't enter custom callback numbers + + _RECURRENCE_CHOICES = [ + (1, "1 minute"), + (2, "2 minutes"), + (3, "3 minutes"), + (5, "5 minutes"), + (8, "8 minutes"), + (10, "10 minutes"), + (15, "15 minutes"), + (30, "30 minutes"), + (45, "45 minutes"), + (60, "1 hour"), + (3*60, "3 hours"), + (6*60, "6 hours"), + (12*60, "12 hours"), + ] + + ALARM_NONE = "No Alert" + ALARM_BACKGROUND = "Background Alert" + ALARM_APPLICATION = "Application Alert" + + VOICEMAIL_CHECK_NOT_SUPPORTED = "Not Supported" + VOICEMAIL_CHECK_DISABLED = "Disabled" + VOICEMAIL_CHECK_ENABLED = "Enabled" def __init__(self, app): + self._app = app self._doClear = False self._accountNumberLabel = QtGui.QLabel("NUMBER NOT SET") + self._notificationSelecter = QtGui.QComboBox() + self._notificationSelecter.currentIndexChanged.connect(self._on_notification_change) + self._notificationTimeSelector = QtGui.QComboBox() + #self._notificationTimeSelector.setEditable(True) + self._notificationTimeSelector.setInsertPolicy(QtGui.QComboBox.InsertAtTop) + for _, label in self._RECURRENCE_CHOICES: + self._notificationTimeSelector.addItem(label) + self._missedCallsNotificationButton = QtGui.QCheckBox("Missed Calls") + self._voicemailNotificationButton = QtGui.QCheckBox("Voicemail") + self._smsNotificationButton = QtGui.QCheckBox("SMS") + self._voicemailOnMissedButton = QtGui.QCheckBox("Voicemail Update on Missed Calls") self._clearButton = QtGui.QPushButton("Clear Account") self._clearButton.clicked.connect(self._on_clear) + self._callbackSelector = QtGui.QComboBox() + #self._callbackSelector.setEditable(True) + self._callbackSelector.setInsertPolicy(QtGui.QComboBox.InsertAtTop) + + self._update_notification_state() self._credLayout = QtGui.QGridLayout() self._credLayout.addWidget(QtGui.QLabel("Account"), 0, 0) self._credLayout.addWidget(self._accountNumberLabel, 0, 1) self._credLayout.addWidget(QtGui.QLabel("Callback"), 1, 0) - self._credLayout.addWidget(QtGui.QLabel(""), 2, 0) - self._credLayout.addWidget(self._clearButton, 2, 1) + self._credLayout.addWidget(self._callbackSelector, 1, 1) + self._credLayout.addWidget(self._notificationSelecter, 2, 0) + self._credLayout.addWidget(self._notificationTimeSelector, 2, 1) self._credLayout.addWidget(QtGui.QLabel(""), 3, 0) + self._credLayout.addWidget(self._missedCallsNotificationButton, 3, 1) + self._credLayout.addWidget(QtGui.QLabel(""), 4, 0) + self._credLayout.addWidget(self._voicemailNotificationButton, 4, 1) + self._credLayout.addWidget(QtGui.QLabel(""), 5, 0) + self._credLayout.addWidget(self._smsNotificationButton, 5, 1) + self._credLayout.addWidget(QtGui.QLabel("Other"), 6, 0) + self._credLayout.addWidget(self._voicemailOnMissedButton, 6, 1) + + self._credLayout.addWidget(QtGui.QLabel(""), 7, 0) + self._credLayout.addWidget(self._clearButton, 7, 1) + self._credWidget = QtGui.QWidget() + self._credWidget.setLayout(self._credLayout) + self._credWidget.setContentsMargins(0, 0, 0, 0) + self._scrollSettings = QtGui.QScrollArea() + self._scrollSettings.setWidget(self._credWidget) + self._scrollSettings.setWidgetResizable(True) + self._scrollSettings.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded) + self._scrollSettings.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff) self._loginButton = QtGui.QPushButton("&Apply") self._buttonLayout = QtGui.QDialogButtonBox(QtGui.QDialogButtonBox.Cancel) self._buttonLayout.addButton(self._loginButton, QtGui.QDialogButtonBox.AcceptRole) self._layout = QtGui.QVBoxLayout() - self._layout.addLayout(self._credLayout) + self._layout.addWidget(self._scrollSettings) self._layout.addWidget(self._buttonLayout) self._dialog = QtGui.QDialog() - self._dialog.setWindowTitle("Login") + self._dialog.setWindowTitle("Account") self._dialog.setLayout(self._layout) - qui_utils.set_autorient(self._dialog, True) self._buttonLayout.accepted.connect(self._dialog.accept) self._buttonLayout.rejected.connect(self._dialog.reject) @@ -125,59 +256,568 @@ class AccountDialog(object): def doClear(self): return self._doClear - accountNumber = property( - lambda self: str(self._accountNumberLabel.text()), - lambda self, num: self._accountNumberLabel.setText(num), + def setIfNotificationsSupported(self, isSupported): + if isSupported: + self._notificationSelecter.clear() + self._notificationSelecter.addItems([self.ALARM_NONE, self.ALARM_APPLICATION, self.ALARM_BACKGROUND]) + self._notificationTimeSelector.setEnabled(False) + self._missedCallsNotificationButton.setEnabled(False) + self._voicemailNotificationButton.setEnabled(False) + self._smsNotificationButton.setEnabled(False) + else: + self._notificationSelecter.clear() + self._notificationSelecter.addItems([self.ALARM_NONE, self.ALARM_APPLICATION]) + self._notificationTimeSelector.setEnabled(False) + self._missedCallsNotificationButton.setEnabled(False) + self._voicemailNotificationButton.setEnabled(False) + self._smsNotificationButton.setEnabled(False) + + def set_account_number(self, num): + self._accountNumberLabel.setText(num) + + def _set_voicemail_on_missed(self, status): + if status == self.VOICEMAIL_CHECK_NOT_SUPPORTED: + self._voicemailOnMissedButton.setChecked(False) + self._voicemailOnMissedButton.hide() + elif status == self.VOICEMAIL_CHECK_DISABLED: + self._voicemailOnMissedButton.setChecked(False) + self._voicemailOnMissedButton.show() + elif status == self.VOICEMAIL_CHECK_ENABLED: + self._voicemailOnMissedButton.setChecked(True) + self._voicemailOnMissedButton.show() + else: + raise RuntimeError("Unsupported option for updating voicemail on missed calls %r" % status) + + def _get_voicemail_on_missed(self): + if not self._voicemailOnMissedButton.isVisible(): + return self.VOICEMAIL_CHECK_NOT_SUPPORTED + elif self._voicemailOnMissedButton.isChecked(): + return self.VOICEMAIL_CHECK_ENABLED + else: + return self.VOICEMAIL_CHECK_DISABLED + + updateVMOnMissedCall = property(_get_voicemail_on_missed, _set_voicemail_on_missed) + + def _set_notifications(self, enabled): + for i in xrange(self._notificationSelecter.count()): + if self._notificationSelecter.itemText(i) == enabled: + self._notificationSelecter.setCurrentIndex(i) + break + else: + self._notificationSelecter.setCurrentIndex(0) + + notifications = property( + lambda self: str(self._notificationSelecter.currentText()), + _set_notifications, + ) + + notifyOnMissed = property( + lambda self: self._missedCallsNotificationButton.isChecked(), + lambda self, enabled: self._missedCallsNotificationButton.setChecked(enabled), + ) + + notifyOnVoicemail = property( + lambda self: self._voicemailNotificationButton.isChecked(), + lambda self, enabled: self._voicemailNotificationButton.setChecked(enabled), + ) + + notifyOnSms = property( + lambda self: self._smsNotificationButton.isChecked(), + lambda self, enabled: self._smsNotificationButton.setChecked(enabled), ) + def _get_notification_time(self): + index = self._notificationTimeSelector.currentIndex() + minutes = self._RECURRENCE_CHOICES[index][0] + return minutes + + def _set_notification_time(self, minutes): + for i, (time, _) in enumerate(self._RECURRENCE_CHOICES): + if time == minutes: + self._notificationTimeSelector.setCurrentIndex(i) + break + else: + self._notificationTimeSelector.setCurrentIndex(0) + + notificationTime = property(_get_notification_time, _set_notification_time) + + @property + def selectedCallback(self): + index = self._callbackSelector.currentIndex() + data = str(self._callbackSelector.itemData(index)) + return data + + def set_callbacks(self, choices, default): + self._callbackSelector.clear() + + self._callbackSelector.addItem("Not Set", "") + + uglyDefault = misc_utils.make_ugly(default) + for number, description in choices.iteritems(): + prettyNumber = misc_utils.make_pretty(number) + uglyNumber = misc_utils.make_ugly(number) + if not uglyNumber: + continue + + self._callbackSelector.addItem("%s - %s" % (prettyNumber, description), uglyNumber) + if uglyNumber == uglyDefault: + self._callbackSelector.setCurrentIndex(self._callbackSelector.count() - 1) + def run(self, parent=None): self._doClear = False - self._dialog.setParent(parent) + self._dialog.setParent(parent, QtCore.Qt.Dialog) response = self._dialog.exec_() return response - @QtCore.pyqtSlot() - @QtCore.pyqtSlot(bool) + def close(self): + try: + self._dialog.reject() + except RuntimeError: + _moduleLogger.exception("Oh well") + + def _update_notification_state(self): + currentText = str(self._notificationSelecter.currentText()) + if currentText == self.ALARM_BACKGROUND: + self._notificationTimeSelector.setEnabled(True) + + self._missedCallsNotificationButton.setEnabled(True) + self._voicemailNotificationButton.setEnabled(True) + self._smsNotificationButton.setEnabled(True) + elif currentText == self.ALARM_APPLICATION: + self._notificationTimeSelector.setEnabled(True) + + self._missedCallsNotificationButton.setEnabled(False) + self._voicemailNotificationButton.setEnabled(True) + self._smsNotificationButton.setEnabled(True) + + self._missedCallsNotificationButton.setChecked(False) + else: + self._notificationTimeSelector.setEnabled(False) + + self._missedCallsNotificationButton.setEnabled(False) + self._voicemailNotificationButton.setEnabled(False) + self._smsNotificationButton.setEnabled(False) + + self._missedCallsNotificationButton.setChecked(False) + self._voicemailNotificationButton.setChecked(False) + self._smsNotificationButton.setChecked(False) + + @qt_compat.Slot(int) + @misc_utils.log_exception(_moduleLogger) + def _on_notification_change(self, index): + with qui_utils.notify_error(self._app.errorLog): + self._update_notification_state() + + @qt_compat.Slot() + @qt_compat.Slot(bool) + @misc_utils.log_exception(_moduleLogger) def _on_clear(self, checked = False): - self._doClear = True - self._dialog.accept() + with qui_utils.notify_error(self._app.errorLog): + self._doClear = True + self._dialog.accept() - @QtCore.pyqtSlot() - @QtCore.pyqtSlot(bool) + @qt_compat.Slot() + @qt_compat.Slot(bool) @misc_utils.log_exception(_moduleLogger) def _on_close_window(self, checked = True): - self._dialog.reject() + with qui_utils.notify_error(self._app.errorLog): + self._dialog.reject() -class SMSEntryWindow(object): +class ContactList(object): - MAX_CHAR = 160 + _SENTINEL_ICON = QtGui.QIcon() + + def __init__(self, app, session): + self._app = app + self._session = session + self._targetLayout = QtGui.QVBoxLayout() + self._targetList = QtGui.QWidget() + self._targetList.setLayout(self._targetLayout) + self._uiItems = [] + self._closeIcon = qui_utils.get_theme_icon(("window-close", "general_close", "gtk-close"), self._SENTINEL_ICON) + + @property + def toplevel(self): + return self._targetList + + def setVisible(self, isVisible): + self._targetList.setVisible(isVisible) + + def update(self): + cids = list(self._session.draft.get_contacts()) + amountCommon = min(len(cids), len(self._uiItems)) + + # Run through everything in common + for i in xrange(0, amountCommon): + cid = cids[i] + uiItem = self._uiItems[i] + title = self._session.draft.get_title(cid) + description = self._session.draft.get_description(cid) + numbers = self._session.draft.get_numbers(cid) + uiItem["cid"] = cid + uiItem["title"] = title + uiItem["description"] = description + uiItem["numbers"] = numbers + uiItem["label"].setText(title) + self._populate_number_selector(uiItem["selector"], cid, i, numbers) + uiItem["rowWidget"].setVisible(True) + + # More contacts than ui items + for i in xrange(amountCommon, len(cids)): + cid = cids[i] + title = self._session.draft.get_title(cid) + description = self._session.draft.get_description(cid) + numbers = self._session.draft.get_numbers(cid) + + titleLabel = QtGui.QLabel(title) + titleLabel.setWordWrap(True) + numberSelector = QtGui.QComboBox() + self._populate_number_selector(numberSelector, cid, i, numbers) + + callback = functools.partial( + self._on_change_number, + i + ) + callback.__name__ = "thanks partials for not having names and pyqt for requiring them" + numberSelector.activated.connect( + qt_compat.Slot(int)(callback) + ) + + if self._closeIcon is self._SENTINEL_ICON: + deleteButton = QtGui.QPushButton("Delete") + else: + deleteButton = QtGui.QPushButton(self._closeIcon, "") + deleteButton.setSizePolicy(QtGui.QSizePolicy( + QtGui.QSizePolicy.Minimum, + QtGui.QSizePolicy.Minimum, + QtGui.QSizePolicy.PushButton, + )) + callback = functools.partial( + self._on_remove_contact, + i + ) + callback.__name__ = "thanks partials for not having names and pyqt for requiring them" + deleteButton.clicked.connect(callback) + + rowLayout = QtGui.QHBoxLayout() + rowLayout.addWidget(titleLabel, 1000) + rowLayout.addWidget(numberSelector, 0) + rowLayout.addWidget(deleteButton, 0) + rowWidget = QtGui.QWidget() + rowWidget.setLayout(rowLayout) + self._targetLayout.addWidget(rowWidget) + + uiItem = {} + uiItem["cid"] = cid + uiItem["title"] = title + uiItem["description"] = description + uiItem["numbers"] = numbers + uiItem["label"] = titleLabel + uiItem["selector"] = numberSelector + uiItem["rowWidget"] = rowWidget + self._uiItems.append(uiItem) + amountCommon = i+1 + + # More UI items than contacts + for i in xrange(amountCommon, len(self._uiItems)): + uiItem = self._uiItems[i] + uiItem["rowWidget"].setVisible(False) + amountCommon = i+1 + + def _populate_number_selector(self, selector, cid, cidIndex, numbers): + selector.clear() + + selectedNumber = self._session.draft.get_selected_number(cid) + if len(numbers) == 1: + # If no alt numbers available, check the address book + numbers, defaultIndex = _get_contact_numbers(self._session, cid, selectedNumber, numbers[0][1]) + else: + defaultIndex = _index_number(numbers, selectedNumber) + + for number, description in numbers: + if description: + label = "%s - %s" % (number, description) + else: + label = number + selector.addItem(label) + selector.setVisible(True) + if 1 < len(numbers): + selector.setEnabled(True) + selector.setCurrentIndex(defaultIndex) + else: + selector.setEnabled(False) + + @misc_utils.log_exception(_moduleLogger) + def _on_change_number(self, cidIndex, index): + with qui_utils.notify_error(self._app.errorLog): + # Exception thrown when the first item is removed + try: + cid = self._uiItems[cidIndex]["cid"] + numbers = self._session.draft.get_numbers(cid) + except IndexError: + _moduleLogger.error("Contact no longer available (or bizarre error): %r (%r)" % (cid, index)) + return + except KeyError: + _moduleLogger.error("Contact no longer available (or bizarre error): %r (%r)" % (cid, index)) + return + number = numbers[index][0] + self._session.draft.set_selected_number(cid, number) + + @misc_utils.log_exception(_moduleLogger) + def _on_remove_contact(self, index, toggled): + with qui_utils.notify_error(self._app.errorLog): + self._session.draft.remove_contact(self._uiItems[index]["cid"]) + + +class VoicemailPlayer(object): + + def __init__(self, app, session, errorLog): + self._app = app + self._session = session + self._errorLog = errorLog + self._token = None + self._session.voicemailAvailable.connect(self._on_voicemail_downloaded) + self._session.draft.recipientsChanged.connect(self._on_recipients_changed) + + self._playButton = QtGui.QPushButton("Play") + self._playButton.clicked.connect(self._on_voicemail_play) + self._pauseButton = QtGui.QPushButton("Pause") + self._pauseButton.clicked.connect(self._on_voicemail_pause) + self._pauseButton.hide() + self._resumeButton = QtGui.QPushButton("Resume") + self._resumeButton.clicked.connect(self._on_voicemail_resume) + self._resumeButton.hide() + self._stopButton = QtGui.QPushButton("Stop") + self._stopButton.clicked.connect(self._on_voicemail_stop) + self._stopButton.hide() + + self._downloadButton = QtGui.QPushButton("Download Voicemail") + self._downloadButton.clicked.connect(self._on_voicemail_download) + self._downloadLayout = QtGui.QHBoxLayout() + self._downloadLayout.addWidget(self._downloadButton) + self._downloadWidget = QtGui.QWidget() + self._downloadWidget.setLayout(self._downloadLayout) + + self._playLabel = QtGui.QLabel("Voicemail") + self._saveButton = QtGui.QPushButton("Save") + self._saveButton.clicked.connect(self._on_voicemail_save) + self._playerLayout = QtGui.QHBoxLayout() + self._playerLayout.addWidget(self._playLabel) + self._playerLayout.addWidget(self._playButton) + self._playerLayout.addWidget(self._pauseButton) + self._playerLayout.addWidget(self._resumeButton) + self._playerLayout.addWidget(self._stopButton) + self._playerLayout.addWidget(self._saveButton) + self._playerWidget = QtGui.QWidget() + self._playerWidget.setLayout(self._playerLayout) + + self._visibleWidget = None + self._layout = QtGui.QHBoxLayout() + self._layout.setContentsMargins(0, 0, 0, 0) + self._widget = QtGui.QWidget() + self._widget.setLayout(self._layout) + self._update_state() + + @property + def toplevel(self): + return self._widget + + def destroy(self): + self._session.voicemailAvailable.disconnect(self._on_voicemail_downloaded) + self._session.draft.recipientsChanged.disconnect(self._on_recipients_changed) + self._invalidate_token() + + def _invalidate_token(self): + if self._token is not None: + self._token.invalidate() + self._token.error.disconnect(self._on_play_error) + self._token.stateChange.connect(self._on_play_state) + self._token.invalidated.connect(self._on_play_invalidated) + + def _show_download(self, messageId): + if self._visibleWidget is self._downloadWidget: + return + self._hide() + self._layout.addWidget(self._downloadWidget) + self._visibleWidget = self._downloadWidget + self._visibleWidget.show() + + def _show_player(self, messageId): + if self._visibleWidget is self._playerWidget: + return + self._hide() + self._layout.addWidget(self._playerWidget) + self._visibleWidget = self._playerWidget + self._visibleWidget.show() + + def _hide(self): + if self._visibleWidget is None: + return + self._visibleWidget.hide() + self._layout.removeWidget(self._visibleWidget) + self._visibleWidget = None + + def _update_play_state(self): + if self._token is not None and self._token.isValid: + self._playButton.setText("Stop") + else: + self._playButton.setText("Play") + + def _update_state(self): + if self._session.draft.get_num_contacts() != 1: + self._hide() + return + + (cid, ) = self._session.draft.get_contacts() + messageId = self._session.draft.get_message_id(cid) + if messageId is None: + self._hide() + return + + if self._session.is_available(messageId): + self._show_player(messageId) + else: + self._show_download(messageId) + if self._token is not None: + self._token.invalidate() + + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_save(self, arg): + with qui_utils.notify_error(self._app.errorLog): + targetPath = QtGui.QFileDialog.getSaveFileName(None, caption="Save Voicemail", filter="Audio File (*.mp3)") + targetPath = unicode(targetPath) + if not targetPath: + return + + (cid, ) = self._session.draft.get_contacts() + messageId = self._session.draft.get_message_id(cid) + sourcePath = self._session.voicemail_path(messageId) + import shutil + shutil.copy2(sourcePath, targetPath) + + @misc_utils.log_exception(_moduleLogger) + def _on_play_error(self, error): + with qui_utils.notify_error(self._app.errorLog): + self._app.errorLog.push_error(error) + + @misc_utils.log_exception(_moduleLogger) + def _on_play_invalidated(self): + with qui_utils.notify_error(self._app.errorLog): + self._playButton.show() + self._pauseButton.hide() + self._resumeButton.hide() + self._stopButton.hide() + self._invalidate_token() + + @misc_utils.log_exception(_moduleLogger) + def _on_play_state(self, state): + with qui_utils.notify_error(self._app.errorLog): + if state == self._token.STATE_PLAY: + self._playButton.hide() + self._pauseButton.show() + self._resumeButton.hide() + self._stopButton.show() + elif state == self._token.STATE_PAUSE: + self._playButton.hide() + self._pauseButton.hide() + self._resumeButton.show() + self._stopButton.show() + elif state == self._token.STATE_STOP: + self._playButton.show() + self._pauseButton.hide() + self._resumeButton.hide() + self._stopButton.hide() + + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_play(self, arg): + with qui_utils.notify_error(self._app.errorLog): + (cid, ) = self._session.draft.get_contacts() + messageId = self._session.draft.get_message_id(cid) + sourcePath = self._session.voicemail_path(messageId) + + self._invalidate_token() + uri = "file://%s" % sourcePath + self._token = self._app.streamHandler.set_file(uri) + self._token.stateChange.connect(self._on_play_state) + self._token.invalidated.connect(self._on_play_invalidated) + self._token.error.connect(self._on_play_error) + self._token.play() + + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_pause(self, arg): + with qui_utils.notify_error(self._app.errorLog): + self._token.pause() + + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_resume(self, arg): + with qui_utils.notify_error(self._app.errorLog): + self._token.play() + + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_stop(self, arg): + with qui_utils.notify_error(self._app.errorLog): + self._token.stop() - # @bug on n900 no scrolling for history qtextedit + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_download(self, arg): + with qui_utils.notify_error(self._app.errorLog): + (cid, ) = self._session.draft.get_contacts() + messageId = self._session.draft.get_message_id(cid) + self._session.download_voicemail(messageId) + self._hide() + + @qt_compat.Slot() + @misc_utils.log_exception(_moduleLogger) + def _on_recipients_changed(self): + with qui_utils.notify_error(self._app.errorLog): + self._update_state() + + @qt_compat.Slot(str, str) + @misc_utils.log_exception(_moduleLogger) + def _on_voicemail_downloaded(self, messageId, filepath): + with qui_utils.notify_error(self._app.errorLog): + self._update_state() + + +class SMSEntryWindow(qwrappers.WindowWrapper): + + MAX_CHAR = 160 + # @bug Somehow a window is being destroyed on object creation which causes glitches on Maemo 5 def __init__(self, parent, app, session, errorLog): + qwrappers.WindowWrapper.__init__(self, parent, app) self._session = session + self._session.messagesUpdated.connect(self._on_refresh_history) + self._session.historyUpdated.connect(self._on_refresh_history) self._session.draft.recipientsChanged.connect(self._on_recipients_changed) - self._session.draft.called.connect(self._on_op_finished) + + self._session.draft.sendingMessage.connect(self._on_op_started) + self._session.draft.calling.connect(self._on_op_started) + self._session.draft.calling.connect(self._on_calling_started) + self._session.draft.cancelling.connect(self._on_op_started) + self._session.draft.sentMessage.connect(self._on_op_finished) + self._session.draft.called.connect(self._on_op_finished) self._session.draft.cancelled.connect(self._on_op_finished) + self._session.draft.error.connect(self._on_op_error) self._errorLog = errorLog - self._targetLayout = QtGui.QVBoxLayout() - self._targetList = QtGui.QWidget() - self._targetList.setLayout(self._targetLayout) - self._history = QtGui.QTextEdit() - self._history.setReadOnly(True) - self._history.setAlignment(QtCore.Qt.AlignLeft | QtCore.Qt.AlignBottom) - self._history.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded) - self._history.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff) + self._errorDisplay = qui_utils.ErrorDisplay(self._errorLog) + + self._targetList = ContactList(self._app, self._session) + self._history = QtGui.QLabel() + self._history.setTextFormat(QtCore.Qt.RichText) + self._history.setWordWrap(True) + self._voicemailPlayer = VoicemailPlayer(self._app, self._session, self._errorLog) self._smsEntry = QtGui.QTextEdit() self._smsEntry.textChanged.connect(self._on_letter_count_changed) self._entryLayout = QtGui.QVBoxLayout() - self._entryLayout.addWidget(self._targetList) + self._entryLayout.addWidget(self._targetList.toplevel) self._entryLayout.addWidget(self._history) + self._entryLayout.addWidget(self._voicemailPlayer.toplevel, 0) self._entryLayout.addWidget(self._smsEntry) self._entryLayout.setContentsMargins(0, 0, 0, 0) self._entryWidget = QtGui.QWidget() @@ -190,59 +830,102 @@ class SMSEntryWindow(object): self._scrollEntry.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAsNeeded) self._scrollEntry.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff) - self._characterCountLabel = QtGui.QLabel("0 (0)") + self._characterCountLabel = QtGui.QLabel("") self._singleNumberSelector = QtGui.QComboBox() + self._cids = [] + self._singleNumberSelector.activated.connect(self._on_single_change_number) self._smsButton = QtGui.QPushButton("SMS") self._smsButton.clicked.connect(self._on_sms_clicked) + self._smsButton.setEnabled(False) self._dialButton = QtGui.QPushButton("Dial") self._dialButton.clicked.connect(self._on_call_clicked) + self._cancelButton = QtGui.QPushButton("Cancel Call") + self._cancelButton.clicked.connect(self._on_cancel_clicked) + self._cancelButton.setVisible(False) self._buttonLayout = QtGui.QHBoxLayout() self._buttonLayout.addWidget(self._characterCountLabel) self._buttonLayout.addWidget(self._singleNumberSelector) self._buttonLayout.addWidget(self._smsButton) self._buttonLayout.addWidget(self._dialButton) + self._buttonLayout.addWidget(self._cancelButton) - self._layout = QtGui.QVBoxLayout() + self._layout.addWidget(self._errorDisplay.toplevel) self._layout.addWidget(self._scrollEntry) self._layout.addLayout(self._buttonLayout) + self._layout.setDirection(QtGui.QBoxLayout.TopToBottom) - centralWidget = QtGui.QWidget() - centralWidget.setLayout(self._layout) - - self._window = QtGui.QMainWindow(parent) - qui_utils.set_autorient(self._window, True) - qui_utils.set_stackable(self._window, True) self._window.setWindowTitle("Contact") - self._window.setCentralWidget(centralWidget) + self._window.closed.connect(self._on_close_window) + self._window.hidden.connect(self._on_close_window) + self._window.resized.connect(self._on_window_resized) - self._closeWindowAction = QtGui.QAction(None) - self._closeWindowAction.setText("Close") - self._closeWindowAction.setShortcut(QtGui.QKeySequence("CTRL+w")) - self._closeWindowAction.triggered.connect(self._on_close_window) + self._scrollTimer = QtCore.QTimer() + self._scrollTimer.setInterval(100) + self._scrollTimer.setSingleShot(True) + self._scrollTimer.timeout.connect(self._on_delayed_scroll_to_bottom) - fileMenu = self._window.menuBar().addMenu("&File") - fileMenu.addAction(self._closeWindowAction) - fileMenu.addAction(app.quitAction) - viewMenu = self._window.menuBar().addMenu("&View") - viewMenu.addAction(app.fullscreenAction) + self._smsEntry.setPlainText(self._session.draft.message) + self._update_letter_count() + self._update_target_fields() + self.set_fullscreen(self._app.fullscreenAction.isChecked()) + self.update_orientation(self._app.orientation) + + def close(self): + if self._window is None: + # Already closed + return + window = self._window + try: + message = unicode(self._smsEntry.toPlainText()) + self._session.draft.message = message + self.hide() + except AttributeError: + _moduleLogger.exception("Oh well") + except RuntimeError: + _moduleLogger.exception("Oh well") + + def destroy(self): + self._session.messagesUpdated.disconnect(self._on_refresh_history) + self._session.historyUpdated.disconnect(self._on_refresh_history) + self._session.draft.recipientsChanged.disconnect(self._on_recipients_changed) + self._session.draft.sendingMessage.disconnect(self._on_op_started) + self._session.draft.calling.disconnect(self._on_op_started) + self._session.draft.calling.disconnect(self._on_calling_started) + self._session.draft.cancelling.disconnect(self._on_op_started) + self._session.draft.sentMessage.disconnect(self._on_op_finished) + self._session.draft.called.disconnect(self._on_op_finished) + self._session.draft.cancelled.disconnect(self._on_op_finished) + self._session.draft.error.disconnect(self._on_op_error) + self._voicemailPlayer.destroy() + window = self._window + self._window = None + try: + window.close() + window.destroy() + except AttributeError: + _moduleLogger.exception("Oh well") + except RuntimeError: + _moduleLogger.exception("Oh well") - self._window.show() - self._update_recipients() + def update_orientation(self, orientation): + qwrappers.WindowWrapper.update_orientation(self, orientation) + self._scroll_to_bottom() def _update_letter_count(self): - count = self._smsEntry.toPlainText().size() + count = len(self._smsEntry.toPlainText()) numTexts, numCharInText = divmod(count, self.MAX_CHAR) numTexts += 1 numCharsLeftInText = self.MAX_CHAR - numCharInText self._characterCountLabel.setText("%d (%d)" % (numCharsLeftInText, numTexts)) def _update_button_state(self): + self._cancelButton.setEnabled(True) if self._session.draft.get_num_contacts() == 0: self._dialButton.setEnabled(False) self._smsButton.setEnabled(False) elif self._session.draft.get_num_contacts() == 1: - count = self._smsEntry.toPlainText().size() + count = len(self._smsEntry.toPlainText()) if count == 0: self._dialButton.setEnabled(True) self._smsButton.setEnabled(False) @@ -251,77 +934,69 @@ class SMSEntryWindow(object): self._smsButton.setEnabled(True) else: self._dialButton.setEnabled(False) - self._smsButton.setEnabled(True) + count = len(self._smsEntry.toPlainText()) + if count == 0: + self._smsButton.setEnabled(False) + else: + self._smsButton.setEnabled(True) - def _update_recipients(self): + def _update_history(self, cid): draftContactsCount = self._session.draft.get_num_contacts() - if draftContactsCount == 0: - self._window.hide() - elif draftContactsCount == 1: - (cid, ) = self._session.draft.get_contacts() - title = self._session.draft.get_title(cid) + if draftContactsCount != 1: + self._history.setVisible(False) + else: description = self._session.draft.get_description(cid) - numbers = self._session.draft.get_numbers(cid) self._targetList.setVisible(False) if description: - self._history.setHtml(description) + self._history.setText(description) self._history.setVisible(True) else: - self._history.setHtml("") + self._history.setText("") self._history.setVisible(False) - self._populate_number_selector(self._singleNumberSelector, cid, numbers) + + def _update_target_fields(self): + draftContactsCount = self._session.draft.get_num_contacts() + if draftContactsCount == 0: + self.hide() + del self._cids[:] + elif draftContactsCount == 1: + (cid, ) = self._session.draft.get_contacts() + title = self._session.draft.get_title(cid) + numbers = self._session.draft.get_numbers(cid) + + self._targetList.setVisible(False) + self._update_history(cid) + self._populate_number_selector(self._singleNumberSelector, cid, 0, numbers) + self._cids = [cid] self._scroll_to_bottom() self._window.setWindowTitle(title) - self._window.show() self._smsEntry.setFocus(QtCore.Qt.OtherFocusReason) + self.show() + self._window.raise_() else: self._targetList.setVisible(True) - while self._targetLayout.count(): - removedLayoutItem = self._targetLayout.takeAt(self._targetLayout.count()-1) - removedWidget = removedLayoutItem.widget() - removedWidget.close() - for cid in self._session.draft.get_contacts(): - title = self._session.draft.get_title(cid) - description = self._session.draft.get_description(cid) - numbers = self._session.draft.get_numbers(cid) - - titleLabel = QtGui.QLabel(title) - numberSelector = QtGui.QComboBox() - self._populate_number_selector(numberSelector, cid, numbers) - deleteButton = QtGui.QPushButton("Delete") - callback = functools.partial( - self._on_remove_contact, - cid - ) - callback.__name__ = "b" - deleteButton.clicked.connect(callback) - - rowLayout = QtGui.QHBoxLayout() - rowLayout.addWidget(titleLabel) - rowLayout.addWidget(numberSelector) - rowLayout.addWidget(deleteButton) - rowWidget = QtGui.QWidget() - rowWidget.setLayout(rowLayout) - self._targetLayout.addWidget(rowWidget) - self._history.setHtml("") + self._targetList.update() + self._history.setText("") self._history.setVisible(False) self._singleNumberSelector.setVisible(False) self._scroll_to_bottom() self._window.setWindowTitle("Contacts") - self._window.show() self._smsEntry.setFocus(QtCore.Qt.OtherFocusReason) + self.show() + self._window.raise_() - def _populate_number_selector(self, selector, cid, numbers): - while 0 < selector.count(): - selector.removeItem(0) + def _populate_number_selector(self, selector, cid, cidIndex, numbers): + selector.clear() + selectedNumber = self._session.draft.get_selected_number(cid) if len(numbers) == 1: - numbers, defaultIndex = _get_contact_numbers(self._session, cid, numbers[0]) + # If no alt numbers available, check the address book + numbers, defaultIndex = _get_contact_numbers(self._session, cid, selectedNumber, numbers[0][1]) else: - defaultIndex = 0 + defaultIndex = _index_number(numbers, selectedNumber) for number, description in numbers: if description: @@ -335,64 +1010,148 @@ class SMSEntryWindow(object): selector.setCurrentIndex(defaultIndex) else: selector.setEnabled(False) - callback = functools.partial( - self._on_change_number, - cid - ) - callback.__name__ = "thanks partials for not having names and pyqt for requiring them" - selector.currentIndexChanged.connect( - QtCore.pyqtSlot(int)(callback) - ) def _scroll_to_bottom(self): - self._scrollEntry.ensureWidgetVisible(self._smsEntry) + self._scrollTimer.start() + + @misc_utils.log_exception(_moduleLogger) + def _on_delayed_scroll_to_bottom(self): + with qui_utils.notify_error(self._app.errorLog): + self._scrollEntry.ensureWidgetVisible(self._smsEntry) @misc_utils.log_exception(_moduleLogger) def _on_sms_clicked(self, arg): - message = str(self._smsEntry.toPlainText()) - self._session.draft.send(message) - self._smsEntry.setPlainText("") + with qui_utils.notify_error(self._app.errorLog): + message = unicode(self._smsEntry.toPlainText()) + self._session.draft.message = message + self._session.draft.send() @misc_utils.log_exception(_moduleLogger) def _on_call_clicked(self, arg): - self._session.draft.call() - self._smsEntry.setPlainText("") + with qui_utils.notify_error(self._app.errorLog): + message = unicode(self._smsEntry.toPlainText()) + self._session.draft.message = message + self._session.draft.call() + @qt_compat.Slot() @misc_utils.log_exception(_moduleLogger) - def _on_remove_contact(self, cid, toggled): - self._session.draft.remove_contact(cid) + def _on_cancel_clicked(self, message): + with qui_utils.notify_error(self._app.errorLog): + self._session.draft.cancel() @misc_utils.log_exception(_moduleLogger) - def _on_change_number(self, cid, index): - # Exception thrown when the first item is removed - numbers = self._session.draft.get_numbers(cid) - number = numbers[index][0] - self._session.draft.set_selected_number(cid, number) + def _on_single_change_number(self, index): + with qui_utils.notify_error(self._app.errorLog): + # Exception thrown when the first item is removed + cid = self._cids[0] + try: + numbers = self._session.draft.get_numbers(cid) + except KeyError: + _moduleLogger.error("Contact no longer available (or bizarre error): %r (%r)" % (cid, index)) + return + number = numbers[index][0] + self._session.draft.set_selected_number(cid, number) - @QtCore.pyqtSlot() + @qt_compat.Slot() + @misc_utils.log_exception(_moduleLogger) + def _on_refresh_history(self): + with qui_utils.notify_error(self._app.errorLog): + draftContactsCount = self._session.draft.get_num_contacts() + if draftContactsCount != 1: + # Changing contact count will automatically refresh it + return + (cid, ) = self._session.draft.get_contacts() + self._update_history(cid) + + @qt_compat.Slot() @misc_utils.log_exception(_moduleLogger) def _on_recipients_changed(self): - self._update_recipients() + with qui_utils.notify_error(self._app.errorLog): + self._update_target_fields() + self._update_button_state() - @QtCore.pyqtSlot() + @qt_compat.Slot() + @misc_utils.log_exception(_moduleLogger) + def _on_op_started(self): + with qui_utils.notify_error(self._app.errorLog): + self._smsEntry.setReadOnly(True) + self._smsButton.setVisible(False) + self._dialButton.setVisible(False) + self.show() + + @qt_compat.Slot() + @misc_utils.log_exception(_moduleLogger) + def _on_calling_started(self): + with qui_utils.notify_error(self._app.errorLog): + self._cancelButton.setVisible(True) + + @qt_compat.Slot() @misc_utils.log_exception(_moduleLogger) def _on_op_finished(self): - self._window.hide() + with qui_utils.notify_error(self._app.errorLog): + self._smsEntry.setPlainText("") + self._smsEntry.setReadOnly(False) + self._cancelButton.setVisible(False) + self._smsButton.setVisible(True) + self._dialButton.setVisible(True) + self.close() + self.destroy() + + @qt_compat.Slot() + @misc_utils.log_exception(_moduleLogger) + def _on_op_error(self, message): + with qui_utils.notify_error(self._app.errorLog): + self._smsEntry.setReadOnly(False) + self._cancelButton.setVisible(False) + self._smsButton.setVisible(True) + self._dialButton.setVisible(True) + + self._errorLog.push_error(message) - @QtCore.pyqtSlot() + @qt_compat.Slot() @misc_utils.log_exception(_moduleLogger) def _on_letter_count_changed(self): - self._update_letter_count() - self._update_button_state() + with qui_utils.notify_error(self._app.errorLog): + self._update_letter_count() + self._update_button_state() + + @qt_compat.Slot() + @misc_utils.log_exception(_moduleLogger) + def _on_window_resized(self): + with qui_utils.notify_error(self._app.errorLog): + self._scroll_to_bottom() - @QtCore.pyqtSlot() - @QtCore.pyqtSlot(bool) + @qt_compat.Slot() + @qt_compat.Slot(bool) @misc_utils.log_exception(_moduleLogger) def _on_close_window(self, checked = True): - self._window.hide() + with qui_utils.notify_error(self._app.errorLog): + self.close() + + +def _index_number(numbers, default): + uglyDefault = misc_utils.make_ugly(default) + uglyContactNumbers = list( + misc_utils.make_ugly(contactNumber) + for (contactNumber, _) in numbers + ) + defaultMatches = [ + misc_utils.similar_ugly_numbers(uglyDefault, contactNumber) + for contactNumber in uglyContactNumbers + ] + try: + defaultIndex = defaultMatches.index(True) + except ValueError: + defaultIndex = -1 + _moduleLogger.warn( + "Could not find contact number %s among %r" % ( + default, numbers + ) + ) + return defaultIndex -def _get_contact_numbers(session, contactId, numberDescription): +def _get_contact_numbers(session, contactId, number, description): contactPhoneNumbers = [] if contactId and contactId != "0": try: @@ -401,31 +1160,13 @@ def _get_contact_numbers(session, contactId, numberDescription): except KeyError: contactPhoneNumbers = [] contactPhoneNumbers = [ - (contactPhoneNumber["phoneNumber"], contactPhoneNumber["phoneType"]) + (contactPhoneNumber["phoneNumber"], contactPhoneNumber.get("phoneType", "Unknown")) for contactPhoneNumber in contactPhoneNumbers ] - if contactPhoneNumbers: - uglyContactNumbers = ( - misc_utils.make_ugly(contactNumber) - for (contactNumber, _) in contactPhoneNumbers - ) - defaultMatches = [ - misc_utils.similar_ugly_numbers(numberDescription[0], contactNumber) - for contactNumber in uglyContactNumbers - ] - try: - defaultIndex = defaultMatches.index(True) - except ValueError: - contactPhoneNumbers.append(numberDescription) - defaultIndex = len(contactPhoneNumbers)-1 - _moduleLogger.warn( - "Could not find contact %r's number %s among %r" % ( - contactId, numberDescription, contactPhoneNumbers - ) - ) - - if not contactPhoneNumbers: - contactPhoneNumbers = [numberDescription] - defaultIndex = -1 + defaultIndex = _index_number(contactPhoneNumbers, number) + + if not contactPhoneNumbers or defaultIndex == -1: + contactPhoneNumbers += [(number, description)] + defaultIndex = 0 return contactPhoneNumbers, defaultIndex