mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-27 23:07:11 -05:00
Support editing attribute values in a separate text edit.
This commit is contained in:
parent
3ab81ea776
commit
1d39368f8d
@ -110,11 +110,25 @@ void EntryAttributes::rename(const QString& oldKey, const QString& newKey)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (m_attributes.contains(newKey)) {
|
||||||
|
Q_ASSERT(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
QString data = value(oldKey);
|
QString data = value(oldKey);
|
||||||
bool protect = isProtected(oldKey);
|
bool protect = isProtected(oldKey);
|
||||||
|
|
||||||
remove(oldKey);
|
Q_EMIT aboutToRename(oldKey, newKey);
|
||||||
set(newKey, data, protect);
|
|
||||||
|
m_attributes.remove(oldKey);
|
||||||
|
m_attributes.insert(newKey, data);
|
||||||
|
if (protect) {
|
||||||
|
m_protectedAttributes.remove(oldKey);
|
||||||
|
m_protectedAttributes.insert(newKey);
|
||||||
|
}
|
||||||
|
|
||||||
|
Q_EMIT modified();
|
||||||
|
Q_EMIT renamed(oldKey, newKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
void EntryAttributes::copyCustomKeysFrom(const EntryAttributes* other)
|
void EntryAttributes::copyCustomKeysFrom(const EntryAttributes* other)
|
||||||
|
@ -53,6 +53,8 @@ Q_SIGNALS:
|
|||||||
void added(QString key);
|
void added(QString key);
|
||||||
void aboutToBeRemoved(QString key);
|
void aboutToBeRemoved(QString key);
|
||||||
void removed(QString key);
|
void removed(QString key);
|
||||||
|
void aboutToRename(QString oldKey, QString newKey);
|
||||||
|
void renamed(QString oldKey, QString newKey);
|
||||||
void aboutToBeReset();
|
void aboutToBeReset();
|
||||||
void reset();
|
void reset();
|
||||||
|
|
||||||
|
@ -61,13 +61,21 @@ EditEntryWidget::EditEntryWidget(QWidget* parent)
|
|||||||
m_advancedUi->setupUi(m_advancedWidget);
|
m_advancedUi->setupUi(m_advancedWidget);
|
||||||
m_ui->stackedWidget->addWidget(m_advancedWidget);
|
m_ui->stackedWidget->addWidget(m_advancedWidget);
|
||||||
|
|
||||||
m_attachmentsModel = new EntryAttachmentsModel(m_advancedWidget);
|
|
||||||
m_advancedUi->attachmentsView->setModel(m_attachmentsModel);
|
|
||||||
m_entryAttributes = new EntryAttributes(this);
|
|
||||||
|
|
||||||
m_attributesModel = new EntryAttributesModel(m_advancedWidget);
|
|
||||||
m_advancedUi->attributesView->setModel(m_attributesModel);
|
|
||||||
m_entryAttachments = new EntryAttachments(this);
|
m_entryAttachments = new EntryAttachments(this);
|
||||||
|
m_attachmentsModel = new EntryAttachmentsModel(m_advancedWidget);
|
||||||
|
m_attachmentsModel->setEntryAttachments(m_entryAttachments);
|
||||||
|
m_advancedUi->attachmentsView->setModel(m_attachmentsModel);
|
||||||
|
|
||||||
|
m_entryAttributes = new EntryAttributes(this);
|
||||||
|
m_attributesModel = new EntryAttributesModel(m_advancedWidget);
|
||||||
|
m_attributesModel->setEntryAttributes(m_entryAttributes);
|
||||||
|
m_advancedUi->attributesView->setModel(m_attributesModel);
|
||||||
|
connect(m_advancedUi->addAttributeButton, SIGNAL(clicked()), SLOT(insertAttribute()));
|
||||||
|
connect(m_advancedUi->editAttributeButton, SIGNAL(clicked()), SLOT(editCurrentAttribute()));
|
||||||
|
connect(m_advancedUi->removeAttributeButton, SIGNAL(clicked()), SLOT(removeCurrentAttribute()));
|
||||||
|
connect(m_advancedUi->attributesView->selectionModel(),
|
||||||
|
SIGNAL(currentChanged(QModelIndex,QModelIndex)),
|
||||||
|
SLOT(updateCurrentAttribute()));
|
||||||
|
|
||||||
Q_ASSERT(m_ui->categoryList->model()->rowCount() == m_ui->stackedWidget->count());
|
Q_ASSERT(m_ui->categoryList->model()->rowCount() == m_ui->stackedWidget->count());
|
||||||
|
|
||||||
@ -114,12 +122,17 @@ void EditEntryWidget::loadEntry(Entry* entry, bool create, const QString& groupN
|
|||||||
m_notesUi->notesEdit->setPlainText(entry->notes());
|
m_notesUi->notesEdit->setPlainText(entry->notes());
|
||||||
|
|
||||||
m_entryAttributes->copyCustomKeysFrom(entry->attributes());
|
m_entryAttributes->copyCustomKeysFrom(entry->attributes());
|
||||||
m_attributesModel->setEntryAttributes(m_entryAttributes);
|
|
||||||
*m_entryAttachments = *entry->attachments();
|
*m_entryAttachments = *entry->attachments();
|
||||||
m_attachmentsModel->setEntryAttachments(m_entryAttachments);
|
|
||||||
|
|
||||||
m_ui->categoryList->setCurrentRow(0);
|
m_ui->categoryList->setCurrentRow(0);
|
||||||
|
|
||||||
|
if (m_attributesModel->rowCount() != 0) {
|
||||||
|
m_advancedUi->attributesView->setCurrentIndex(m_attributesModel->index(0, 0));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
m_advancedUi->attributesEdit->setEnabled(false);
|
||||||
|
}
|
||||||
|
|
||||||
m_mainUi->titleEdit->setFocus();
|
m_mainUi->titleEdit->setFocus();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -130,6 +143,14 @@ void EditEntryWidget::saveEntry()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (m_advancedUi->attributesView->currentIndex().isValid()) {
|
||||||
|
QString key = m_attributesModel->keyByIndex(m_advancedUi->attributesView->currentIndex());
|
||||||
|
m_entryAttributes->set(key, m_advancedUi->attributesEdit->toPlainText(),
|
||||||
|
m_entryAttributes->isProtected(key));
|
||||||
|
}
|
||||||
|
|
||||||
|
m_currentAttribute = QPersistentModelIndex();
|
||||||
|
|
||||||
m_entry->beginUpdate();
|
m_entry->beginUpdate();
|
||||||
|
|
||||||
m_entry->setTitle(m_mainUi->titleEdit->text());
|
m_entry->setTitle(m_mainUi->titleEdit->text());
|
||||||
@ -142,19 +163,23 @@ void EditEntryWidget::saveEntry()
|
|||||||
m_entry->setNotes(m_notesUi->notesEdit->toPlainText());
|
m_entry->setNotes(m_notesUi->notesEdit->toPlainText());
|
||||||
|
|
||||||
m_entry->attributes()->copyCustomKeysFrom(m_entryAttributes);
|
m_entry->attributes()->copyCustomKeysFrom(m_entryAttributes);
|
||||||
m_entryAttributes->clear();
|
|
||||||
*m_entry->attachments() = *m_entryAttachments;
|
*m_entry->attachments() = *m_entryAttachments;
|
||||||
m_entryAttachments->clear();
|
|
||||||
|
|
||||||
m_entry->endUpdate();
|
m_entry->endUpdate();
|
||||||
|
|
||||||
m_entry = 0;
|
m_entry = 0;
|
||||||
|
m_entryAttributes->clear();
|
||||||
|
m_entryAttachments->clear();
|
||||||
|
|
||||||
Q_EMIT editFinished(true);
|
Q_EMIT editFinished(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditEntryWidget::cancel()
|
void EditEntryWidget::cancel()
|
||||||
{
|
{
|
||||||
m_entry = 0;
|
m_entry = 0;
|
||||||
|
m_entryAttributes->clear();
|
||||||
|
m_entryAttachments->clear();
|
||||||
|
|
||||||
Q_EMIT editFinished(false);
|
Q_EMIT editFinished(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -185,3 +210,65 @@ void EditEntryWidget::setPasswordCheckColors()
|
|||||||
}
|
}
|
||||||
m_mainUi->passwordRepeatEdit->setPalette(pal);
|
m_mainUi->passwordRepeatEdit->setPalette(pal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EditEntryWidget::insertAttribute()
|
||||||
|
{
|
||||||
|
QString name = tr("New attribute");
|
||||||
|
int i = 1;
|
||||||
|
|
||||||
|
while (m_entryAttributes->keys().contains(name)) {
|
||||||
|
name = QString("%1 %2").arg(tr("New attribute")).arg(i);
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_entryAttributes->set(name, "");
|
||||||
|
QModelIndex index = m_attributesModel->indexByKey(name);
|
||||||
|
|
||||||
|
m_advancedUi->attributesView->setCurrentIndex(index);
|
||||||
|
m_advancedUi->attributesView->edit(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditEntryWidget::editCurrentAttribute()
|
||||||
|
{
|
||||||
|
QModelIndex index = m_advancedUi->attributesView->currentIndex();
|
||||||
|
|
||||||
|
if (index.isValid()) {
|
||||||
|
m_advancedUi->attributesView->edit(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditEntryWidget::removeCurrentAttribute()
|
||||||
|
{
|
||||||
|
QModelIndex index = m_advancedUi->attributesView->currentIndex();
|
||||||
|
|
||||||
|
if (index.isValid()) {
|
||||||
|
m_entryAttributes->remove(m_attributesModel->keyByIndex(index));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditEntryWidget::updateCurrentAttribute()
|
||||||
|
{
|
||||||
|
QModelIndex newIndex = m_advancedUi->attributesView->currentIndex();
|
||||||
|
|
||||||
|
if (m_currentAttribute != newIndex) {
|
||||||
|
if (m_currentAttribute.isValid()) {
|
||||||
|
QString key = m_attributesModel->keyByIndex(m_currentAttribute);
|
||||||
|
m_entryAttributes->set(key, m_advancedUi->attributesEdit->toPlainText(),
|
||||||
|
m_entryAttributes->isProtected(key));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newIndex.isValid()) {
|
||||||
|
QString key = m_attributesModel->keyByIndex(newIndex);
|
||||||
|
m_advancedUi->attributesEdit->setPlainText(m_entryAttributes->value(key));
|
||||||
|
m_advancedUi->attributesEdit->setEnabled(true);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
m_advancedUi->attributesEdit->setPlainText("");
|
||||||
|
m_advancedUi->attributesEdit->setEnabled(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_advancedUi->editAttributeButton->setEnabled(newIndex.isValid());
|
||||||
|
m_advancedUi->removeAttributeButton->setEnabled(newIndex.isValid());
|
||||||
|
m_currentAttribute = newIndex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
#ifndef KEEPASSX_EDITENTRYWIDGET_H
|
#ifndef KEEPASSX_EDITENTRYWIDGET_H
|
||||||
#define KEEPASSX_EDITENTRYWIDGET_H
|
#define KEEPASSX_EDITENTRYWIDGET_H
|
||||||
|
|
||||||
|
#include <QtCore/QAbstractItemModel>
|
||||||
#include <QtCore/QScopedPointer>
|
#include <QtCore/QScopedPointer>
|
||||||
|
|
||||||
#include "gui/DialogyWidget.h"
|
#include "gui/DialogyWidget.h"
|
||||||
@ -59,6 +60,10 @@ private Q_SLOTS:
|
|||||||
void cancel();
|
void cancel();
|
||||||
void togglePassword(bool checked);
|
void togglePassword(bool checked);
|
||||||
void setPasswordCheckColors();
|
void setPasswordCheckColors();
|
||||||
|
void insertAttribute();
|
||||||
|
void editCurrentAttribute();
|
||||||
|
void removeCurrentAttribute();
|
||||||
|
void updateCurrentAttribute();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool passwordsEqual();
|
bool passwordsEqual();
|
||||||
@ -76,6 +81,7 @@ private:
|
|||||||
EntryAttributesModel* m_attributesModel;
|
EntryAttributesModel* m_attributesModel;
|
||||||
EntryAttachments* m_entryAttachments;
|
EntryAttachments* m_entryAttachments;
|
||||||
EntryAttributes* m_entryAttributes;
|
EntryAttributes* m_entryAttributes;
|
||||||
|
QPersistentModelIndex m_currentAttribute;
|
||||||
|
|
||||||
Q_DISABLE_COPY(EditEntryWidget)
|
Q_DISABLE_COPY(EditEntryWidget)
|
||||||
};
|
};
|
||||||
|
@ -18,8 +18,11 @@
|
|||||||
</property>
|
</property>
|
||||||
<layout class="QHBoxLayout" name="horizontalLayout">
|
<layout class="QHBoxLayout" name="horizontalLayout">
|
||||||
<item>
|
<item>
|
||||||
<widget class="QTableView" name="attributesView">
|
<widget class="AttributesListView" name="attributesView"/>
|
||||||
<property name="showGrid">
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QPlainTextEdit" name="attributesEdit">
|
||||||
|
<property name="enabled">
|
||||||
<bool>false</bool>
|
<bool>false</bool>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
@ -35,6 +38,9 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="editAttributeButton">
|
<widget class="QPushButton" name="editAttributeButton">
|
||||||
|
<property name="enabled">
|
||||||
|
<bool>false</bool>
|
||||||
|
</property>
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Edit</string>
|
<string>Edit</string>
|
||||||
</property>
|
</property>
|
||||||
@ -42,6 +48,9 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="removeAttributeButton">
|
<widget class="QPushButton" name="removeAttributeButton">
|
||||||
|
<property name="enabled">
|
||||||
|
<bool>false</bool>
|
||||||
|
</property>
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Remove</string>
|
<string>Remove</string>
|
||||||
</property>
|
</property>
|
||||||
@ -121,6 +130,13 @@
|
|||||||
</item>
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
|
<customwidgets>
|
||||||
|
<customwidget>
|
||||||
|
<class>AttributesListView</class>
|
||||||
|
<extends>QListView</extends>
|
||||||
|
<header>gui/EditEntryWidget_p.h</header>
|
||||||
|
</customwidget>
|
||||||
|
</customwidgets>
|
||||||
<resources/>
|
<resources/>
|
||||||
<connections/>
|
<connections/>
|
||||||
</ui>
|
</ui>
|
||||||
|
@ -59,4 +59,27 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class AttributesListView : public QListView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit AttributesListView(QWidget* parent = 0) : QListView(parent)
|
||||||
|
{
|
||||||
|
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Expanding);
|
||||||
|
setItemDelegate(new CategoryListViewDelegate(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual QSize sizeHint() const
|
||||||
|
{
|
||||||
|
QSize sizeHint = QListView::sizeHint();
|
||||||
|
|
||||||
|
int width = sizeHintForColumn(0) + frameWidth() * 2 + 5;
|
||||||
|
if (verticalScrollBar()->isVisible()) {
|
||||||
|
width += verticalScrollBar()->width();
|
||||||
|
}
|
||||||
|
sizeHint.setWidth(width);
|
||||||
|
|
||||||
|
return sizeHint;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
#endif // KEEPASSX_EDITENTRYWIDGET_P_H
|
#endif // KEEPASSX_EDITENTRYWIDGET_P_H
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
EntryAttributesModel::EntryAttributesModel(QObject* parent)
|
EntryAttributesModel::EntryAttributesModel(QObject* parent)
|
||||||
: QAbstractListModel(parent)
|
: QAbstractListModel(parent)
|
||||||
, m_entryAttributes(0)
|
, m_entryAttributes(0)
|
||||||
|
, m_nextRenameDataChange(false)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -43,6 +44,9 @@ void EntryAttributesModel::setEntryAttributes(EntryAttributes* entryAttributes)
|
|||||||
connect(m_entryAttributes, SIGNAL(added(QString)), SLOT(attributeAdd()));
|
connect(m_entryAttributes, SIGNAL(added(QString)), SLOT(attributeAdd()));
|
||||||
connect(m_entryAttributes, SIGNAL(aboutToBeRemoved(QString)), SLOT(attributeAboutToRemove(QString)));
|
connect(m_entryAttributes, SIGNAL(aboutToBeRemoved(QString)), SLOT(attributeAboutToRemove(QString)));
|
||||||
connect(m_entryAttributes, SIGNAL(removed(QString)), SLOT(attributeRemove()));
|
connect(m_entryAttributes, SIGNAL(removed(QString)), SLOT(attributeRemove()));
|
||||||
|
connect(m_entryAttributes, SIGNAL(aboutToRename(QString,QString)),
|
||||||
|
SLOT(attributeAboutToRename(QString,QString)));
|
||||||
|
connect(m_entryAttributes, SIGNAL(renamed(QString,QString)), SLOT(attributeRename(QString,QString)));
|
||||||
connect(m_entryAttributes, SIGNAL(aboutToBeReset()), SLOT(aboutToReset()));
|
connect(m_entryAttributes, SIGNAL(aboutToBeReset()), SLOT(aboutToReset()));
|
||||||
connect(m_entryAttributes, SIGNAL(reset()), SLOT(reset()));
|
connect(m_entryAttributes, SIGNAL(reset()), SLOT(reset()));
|
||||||
}
|
}
|
||||||
@ -171,6 +175,46 @@ void EntryAttributesModel::attributeRemove()
|
|||||||
endRemoveRows();
|
endRemoveRows();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EntryAttributesModel::attributeAboutToRename(QString oldKey, QString newKey)
|
||||||
|
{
|
||||||
|
int oldRow = m_attributes.indexOf(oldKey);
|
||||||
|
|
||||||
|
QList<QString> rows = m_attributes;
|
||||||
|
rows.removeOne(oldKey);
|
||||||
|
rows.append(newKey);
|
||||||
|
qSort(rows);
|
||||||
|
int newRow = rows.indexOf(newKey);
|
||||||
|
if (newRow > oldRow) {
|
||||||
|
newRow++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (oldRow != newRow) {
|
||||||
|
bool result = beginMoveRows(QModelIndex(), oldRow, oldRow, QModelIndex(), newRow);
|
||||||
|
Q_UNUSED(result);
|
||||||
|
Q_ASSERT(result);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
m_nextRenameDataChange = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void EntryAttributesModel::attributeRename(QString oldKey, QString newKey)
|
||||||
|
{
|
||||||
|
Q_UNUSED(oldKey);
|
||||||
|
|
||||||
|
updateAttributes();
|
||||||
|
|
||||||
|
if (!m_nextRenameDataChange) {
|
||||||
|
endMoveRows();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
m_nextRenameDataChange = false;
|
||||||
|
|
||||||
|
QModelIndex keyIndex = index(m_attributes.indexOf(newKey), 0);
|
||||||
|
Q_EMIT dataChanged(keyIndex, keyIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void EntryAttributesModel::aboutToReset()
|
void EntryAttributesModel::aboutToReset()
|
||||||
{
|
{
|
||||||
beginResetModel();
|
beginResetModel();
|
||||||
|
@ -44,6 +44,8 @@ private Q_SLOTS:
|
|||||||
void attributeAdd();
|
void attributeAdd();
|
||||||
void attributeAboutToRemove(QString key);
|
void attributeAboutToRemove(QString key);
|
||||||
void attributeRemove();
|
void attributeRemove();
|
||||||
|
void attributeAboutToRename(QString oldKey, QString newKey);
|
||||||
|
void attributeRename(QString oldKey, QString newKey);
|
||||||
void aboutToReset();
|
void aboutToReset();
|
||||||
void reset();
|
void reset();
|
||||||
|
|
||||||
@ -52,6 +54,7 @@ private:
|
|||||||
|
|
||||||
EntryAttributes* m_entryAttributes;
|
EntryAttributes* m_entryAttributes;
|
||||||
QList<QString> m_attributes;
|
QList<QString> m_attributes;
|
||||||
|
bool m_nextRenameDataChange;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // KEEPASSX_ENTRYATTRIBUTESMODEL_H
|
#endif // KEEPASSX_ENTRYATTRIBUTESMODEL_H
|
||||||
|
Loading…
x
Reference in New Issue
Block a user