Grey out Apply button when there are no changes

Resolves 

What this commit does:
* Whenever the Apply button is pressed, and if the save was successful, then the Apply button is disabled.
* Each subwidget used by EditEntryWidget has now a signal called `widgetUpdated` that is emitted when the widgets' internal content changes. The EditEntryWidget subscribes to that signal to know when to enable the Apply button (by calling `entryUpdated()`).
* There are some views that are not isolated in their own widgets (`m_advancedUi`, for example) so in those cases I invoked `entryUpdated()` directly whenever I detected an update:
  * some updates occur directly in a Qt widget like when editing the text of a QLineItem, so in that case I connected the widget's signals directly to the `entryUpdated()` slot.
  * some updates occur in EditEntryWidget, so in those cases the invocation to `entryUpdated()` is made as soon as the change is detected (for example when the user has confirmed an action in a dialog).

A known problem: there are some situations when the Apply button will get enabled even if there are no changes, this is because the app changes the value of a field by itself so it's considered an update (for example, clicking on the "Reveal" button changes the text shown in a text field).
The solution to this can be a bit complicated: disabling temporarily the `entryUpdated()` whenever the app is going to do an action with such side-effects.
So I preferred to let the Apply button get enabled in those cases.
This commit is contained in:
Daniel Wilches 2018-03-10 19:31:43 -08:00
parent e92d5e80ee
commit 78ef6f0d04
8 changed files with 102 additions and 2 deletions

@ -117,6 +117,14 @@ bool EditWidget::readOnly() const
return m_readOnly; return m_readOnly;
} }
void EditWidget::enableApplyButton(bool enabled)
{
QPushButton* applyButton = m_ui->buttonBox->button(QDialogButtonBox::Apply);
if (applyButton) {
applyButton->setEnabled(enabled);
}
}
void EditWidget::showMessage(const QString& text, MessageWidget::MessageType type) void EditWidget::showMessage(const QString& text, MessageWidget::MessageType type)
{ {
m_ui->messageWidget->setCloseButtonVisible(false); m_ui->messageWidget->setCloseButtonVisible(false);

@ -47,6 +47,7 @@ public:
QLabel* headlineLabel(); QLabel* headlineLabel();
void setReadOnly(bool readOnly); void setReadOnly(bool readOnly);
bool readOnly() const; bool readOnly() const;
void enableApplyButton(bool enabled);
signals: signals:
void apply(); void apply();

@ -66,6 +66,13 @@ EditWidgetIcons::EditWidgetIcons(QWidget* parent)
connect(m_ui->deleteButton, SIGNAL(clicked()), SLOT(removeCustomIcon())); connect(m_ui->deleteButton, SIGNAL(clicked()), SLOT(removeCustomIcon()));
connect(m_ui->faviconButton, SIGNAL(clicked()), SLOT(downloadFavicon())); connect(m_ui->faviconButton, SIGNAL(clicked()), SLOT(downloadFavicon()));
connect(m_ui->defaultIconsRadio, SIGNAL(toggled(bool)), this, SIGNAL(widgetUpdated()));
connect(m_ui->defaultIconsRadio, SIGNAL(toggled(bool)), this, SIGNAL(widgetUpdated()));
connect(m_ui->defaultIconsView->selectionModel(), SIGNAL(selectionChanged(const QItemSelection&, const QItemSelection&)),
this, SIGNAL(widgetUpdated()));
connect(m_ui->customIconsView->selectionModel(), SIGNAL(selectionChanged(const QItemSelection&, const QItemSelection&)),
this, SIGNAL(widgetUpdated()));
m_ui->faviconButton->setVisible(false); m_ui->faviconButton->setVisible(false);
} }
@ -268,6 +275,8 @@ void EditWidgetIcons::addCustomIcon(const QImage& icon)
updateRadioButtonCustomIcons(); updateRadioButtonCustomIcons();
QModelIndex index = m_customIconModel->indexFromUuid(uuid); QModelIndex index = m_customIconModel->indexFromUuid(uuid);
m_ui->customIconsView->setCurrentIndex(index); m_ui->customIconsView->setCurrentIndex(index);
emit widgetUpdated();
} }
} }
@ -347,6 +356,8 @@ void EditWidgetIcons::removeCustomIcon()
} else { } else {
m_ui->defaultIconsView->setCurrentIndex(m_defaultIconModel->index(Group::DefaultIconNumber)); m_ui->defaultIconsView->setCurrentIndex(m_defaultIconModel->index(Group::DefaultIconNumber));
} }
emit widgetUpdated();
} }
} }
} }

@ -62,6 +62,7 @@ public slots:
signals: signals:
void messageEditEntry(QString, MessageWidget::MessageType); void messageEditEntry(QString, MessageWidget::MessageType);
void messageEditEntryDismiss(); void messageEditEntryDismiss();
void widgetUpdated();
private slots: private slots:
void downloadFavicon(); void downloadFavicon();

@ -95,6 +95,7 @@ EditEntryWidget::EditEntryWidget(QWidget* parent)
#endif #endif
setupProperties(); setupProperties();
setupHistory(); setupHistory();
setupEntryUpdate();
connect(this, SIGNAL(accepted()), SLOT(acceptEntry())); connect(this, SIGNAL(accepted()), SLOT(acceptEntry()));
connect(this, SIGNAL(rejected()), SLOT(cancel())); connect(this, SIGNAL(rejected()), SLOT(cancel()));
@ -227,6 +228,59 @@ void EditEntryWidget::setupHistory()
connect(m_historyUi->deleteAllButton, SIGNAL(clicked()), SLOT(deleteAllHistoryEntries())); connect(m_historyUi->deleteAllButton, SIGNAL(clicked()), SLOT(deleteAllHistoryEntries()));
} }
void EditEntryWidget::setupEntryUpdate()
{
// Entry tab
connect(m_mainUi->titleEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->usernameEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->passwordEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->passwordRepeatEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->urlEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->expireCheck, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->notesEnabled, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->expireDatePicker, SIGNAL(dateTimeChanged(const QDateTime&)), this, SLOT(setUnsavedChanges()));
connect(m_mainUi->notesEdit, SIGNAL(textChanged()), this, SLOT(setUnsavedChanges()));
// Advanced tab
connect(m_advancedUi->attributesEdit, SIGNAL(textChanged()), this, SLOT(setUnsavedChanges()));
connect(m_advancedUi->protectAttributeButton, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_advancedUi->fgColorCheckBox, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_advancedUi->bgColorCheckBox, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_advancedUi->attachmentsWidget, SIGNAL(widgetUpdated()), this, SLOT(setUnsavedChanges()));
// Icon tab
connect(m_iconsWidget, SIGNAL(widgetUpdated()), this, SLOT(setUnsavedChanges()));
// Auto-Type tab
connect(m_autoTypeUi->enableButton, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->customWindowSequenceButton, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->inheritSequenceButton, SIGNAL(toggled(bool)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->customSequenceButton, SIGNAL(toggled(bool)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->windowSequenceEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->sequenceEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->windowTitleCombo, SIGNAL(currentIndexChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_autoTypeUi->windowTitleCombo, SIGNAL(editTextChanged(const QString&)), this, SLOT(setUnsavedChanges()));
// Properties and History tabs don't need extra connections
#ifdef WITH_XC_SSHAGENT
// SSH Agent tab
if (config()->get("SSHAgent", false).toBool()) {
connect(m_sshAgentUi->attachmentRadioButton, SIGNAL(toggled(bool)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->externalFileRadioButton, SIGNAL(toggled(bool)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->attachmentComboBox, SIGNAL(currentIndexChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->attachmentComboBox, SIGNAL(editTextChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->externalFileEdit, SIGNAL(textChanged(const QString&)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->publicKeyEdit, SIGNAL(textChanged()), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->addKeyToAgentCheckBox, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->removeKeyFromAgentCheckBox, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->requireUserConfirmationCheckBox, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->lifetimeCheckBox, SIGNAL(stateChanged(int)), this, SLOT(setUnsavedChanges()));
connect(m_sshAgentUi->lifetimeSpinBox, SIGNAL(valueChanged(int)), this, SLOT(setUnsavedChanges()));
}
#endif
}
void EditEntryWidget::emitHistoryEntryActivated(const QModelIndex& index) void EditEntryWidget::emitHistoryEntryActivated(const QModelIndex& index)
{ {
Q_ASSERT(!m_history); Q_ASSERT(!m_history);
@ -581,7 +635,6 @@ void EditEntryWidget::loadEntry(Entry* entry, bool create, bool history, const Q
m_database = database; m_database = database;
m_create = create; m_create = create;
m_history = history; m_history = history;
m_saved = false;
if (history) { if (history) {
setHeadline(QString("%1 > %2").arg(parentName, tr("Entry history"))); setHeadline(QString("%1 > %2").arg(parentName, tr("Entry history")));
@ -601,6 +654,9 @@ void EditEntryWidget::loadEntry(Entry* entry, bool create, bool history, const Q
setCurrentPage(0); setCurrentPage(0);
setPageHidden(m_historyWidget, m_history || m_entry->historyItems().count() < 1); setPageHidden(m_historyWidget, m_history || m_entry->historyItems().count() < 1);
// Force the user to Save/Apply/Discard new entries
setUnsavedChanges(m_create);
} }
void EditEntryWidget::setForms(const Entry* entry, bool restore) void EditEntryWidget::setForms(const Entry* entry, bool restore)
@ -780,7 +836,7 @@ bool EditEntryWidget::commitEntry()
} }
updateEntryData(m_entry); updateEntryData(m_entry);
m_saved = true; setUnsavedChanges(false);
if (!m_create) { if (!m_create) {
m_entry->endUpdate(); m_entry->endUpdate();
@ -940,6 +996,8 @@ void EditEntryWidget::insertAttribute()
m_advancedUi->attributesView->setCurrentIndex(index); m_advancedUi->attributesView->setCurrentIndex(index);
m_advancedUi->attributesView->edit(index); m_advancedUi->attributesView->edit(index);
setUnsavedChanges(true);
} }
void EditEntryWidget::editCurrentAttribute() void EditEntryWidget::editCurrentAttribute()
@ -950,6 +1008,7 @@ void EditEntryWidget::editCurrentAttribute()
if (index.isValid()) { if (index.isValid()) {
m_advancedUi->attributesView->edit(index); m_advancedUi->attributesView->edit(index);
setUnsavedChanges(true);
} }
} }
@ -963,6 +1022,7 @@ void EditEntryWidget::removeCurrentAttribute()
if (MessageBox::question(this, tr("Confirm Remove"), tr("Are you sure you want to remove this attribute?"), if (MessageBox::question(this, tr("Confirm Remove"), tr("Are you sure you want to remove this attribute?"),
QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes) { QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes) {
m_entryAttributes->remove(m_attributesModel->keyByIndex(index)); m_entryAttributes->remove(m_attributesModel->keyByIndex(index));
setUnsavedChanges(true);
} }
} }
} }
@ -1047,9 +1107,11 @@ void EditEntryWidget::revealCurrentAttribute()
if (! m_advancedUi->attributesEdit->isEnabled()) { if (! m_advancedUi->attributesEdit->isEnabled()) {
QModelIndex index = m_advancedUi->attributesView->currentIndex(); QModelIndex index = m_advancedUi->attributesView->currentIndex();
if (index.isValid()) { if (index.isValid()) {
bool oldBlockSignals = m_advancedUi->attributesEdit->blockSignals(true);
QString key = m_attributesModel->keyByIndex(index); QString key = m_attributesModel->keyByIndex(index);
m_advancedUi->attributesEdit->setPlainText(m_entryAttributes->value(key)); m_advancedUi->attributesEdit->setPlainText(m_entryAttributes->value(key));
m_advancedUi->attributesEdit->setEnabled(true); m_advancedUi->attributesEdit->setEnabled(true);
m_advancedUi->attributesEdit->blockSignals(oldBlockSignals);
} }
} }
} }
@ -1083,6 +1145,7 @@ void EditEntryWidget::insertAutoTypeAssoc()
m_autoTypeUi->assocView->setCurrentIndex(newIndex); m_autoTypeUi->assocView->setCurrentIndex(newIndex);
loadCurrentAssoc(newIndex); loadCurrentAssoc(newIndex);
m_autoTypeUi->windowTitleCombo->setFocus(); m_autoTypeUi->windowTitleCombo->setFocus();
setUnsavedChanges(true);
} }
void EditEntryWidget::removeAutoTypeAssoc() void EditEntryWidget::removeAutoTypeAssoc()
@ -1091,6 +1154,7 @@ void EditEntryWidget::removeAutoTypeAssoc()
if (currentIndex.isValid()) { if (currentIndex.isValid()) {
m_autoTypeAssoc->remove(currentIndex.row()); m_autoTypeAssoc->remove(currentIndex.row());
setUnsavedChanges(true);
} }
} }
@ -1153,6 +1217,7 @@ void EditEntryWidget::restoreHistoryEntry()
QModelIndex index = m_sortModel->mapToSource(m_historyUi->historyView->currentIndex()); QModelIndex index = m_sortModel->mapToSource(m_historyUi->historyView->currentIndex());
if (index.isValid()) { if (index.isValid()) {
setForms(m_historyModel->entryFromIndex(index), true); setForms(m_historyModel->entryFromIndex(index), true);
setUnsavedChanges(true);
} }
} }
@ -1166,6 +1231,7 @@ void EditEntryWidget::deleteHistoryEntry()
} else { } else {
m_historyUi->deleteAllButton->setEnabled(false); m_historyUi->deleteAllButton->setEnabled(false);
} }
setUnsavedChanges(true);
} }
} }
@ -1178,6 +1244,7 @@ void EditEntryWidget::deleteAllHistoryEntries()
else { else {
m_historyUi->deleteAllButton->setEnabled(false); m_historyUi->deleteAllButton->setEnabled(false);
} }
setUnsavedChanges(true);
} }
QMenu* EditEntryWidget::createPresetsMenu() QMenu* EditEntryWidget::createPresetsMenu()
@ -1229,5 +1296,12 @@ void EditEntryWidget::pickColor()
QColor newColor = colorDialog.getColor(oldColor); QColor newColor = colorDialog.getColor(oldColor);
if (newColor.isValid()) { if (newColor.isValid()) {
setupColorButton(isForeground, newColor); setupColorButton(isForeground, newColor);
setUnsavedChanges(true);
} }
} }
void EditEntryWidget::setUnsavedChanges(bool hasUnsaved)
{
m_saved = !hasUnsaved;
enableApplyButton(hasUnsaved);
}

@ -100,6 +100,7 @@ private slots:
void useExpiryPreset(QAction* action); void useExpiryPreset(QAction* action);
void toggleHideNotes(bool visible); void toggleHideNotes(bool visible);
void pickColor(); void pickColor();
void setUnsavedChanges(bool hasUnsaved = true);
#ifdef WITH_XC_SSHAGENT #ifdef WITH_XC_SSHAGENT
void updateSSHAgent(); void updateSSHAgent();
void updateSSHAgentAttachment(); void updateSSHAgentAttachment();
@ -122,6 +123,7 @@ private:
#endif #endif
void setupProperties(); void setupProperties();
void setupHistory(); void setupHistory();
void setupEntryUpdate();
void setupColorButton(bool foreground, const QColor& color); void setupColorButton(bool foreground, const QColor& color);
bool passwordsEqual(); bool passwordsEqual();

@ -147,6 +147,7 @@ void EntryAttachmentsWidget::insertAttachments()
if (!insertAttachments(filenames, errorMessage)) { if (!insertAttachments(filenames, errorMessage)) {
errorOccurred(errorMessage); errorOccurred(errorMessage);
} }
emit widgetUpdated();
} }
void EntryAttachmentsWidget::removeSelectedAttachments() void EntryAttachmentsWidget::removeSelectedAttachments()
@ -170,6 +171,7 @@ void EntryAttachmentsWidget::removeSelectedAttachments()
keys.append(m_attachmentsModel->keyByIndex(index)); keys.append(m_attachmentsModel->keyByIndex(index));
} }
m_entryAttachments->remove(keys); m_entryAttachments->remove(keys);
emit widgetUpdated();
} }
} }

@ -39,6 +39,7 @@ signals:
void errorOccurred(const QString& error); void errorOccurred(const QString& error);
void readOnlyChanged(bool readOnly); void readOnlyChanged(bool readOnly);
void buttonsVisibleChanged(bool isButtonsVisible); void buttonsVisibleChanged(bool isButtonsVisible);
void widgetUpdated();
private slots: private slots:
void insertAttachments(); void insertAttachments();