Several improvements to tags editing

* Fix #7602 - Allow spaces in tag names
* Fix #7528 - Allow pasting text into the tags field. Text is split by comma creating tags for each section of text. If there are no commas then the pasted text becomes a tag.
* Fix tags editing not causing the entry to be marked as modified.
This commit is contained in:
Jonathan White 2022-03-28 22:49:03 -04:00
parent ad61d71c6e
commit 02602da257
4 changed files with 26 additions and 9 deletions

View file

@ -192,7 +192,7 @@ QString Entry::tags() const
QStringList Entry::tagList() const QStringList Entry::tagList() const
{ {
static QRegExp rx("(\\ |\\,|\\t|\\;)"); static QRegExp rx("(\\,|\\t|\\;)");
auto taglist = tags().split(rx, QString::SkipEmptyParts); auto taglist = tags().split(rx, QString::SkipEmptyParts);
std::sort(taglist.begin(), taglist.end()); std::sort(taglist.begin(), taglist.end());
return taglist; return taglist;

View file

@ -441,6 +441,7 @@ void EditEntryWidget::setupEntryUpdate()
#ifdef WITH_XC_NETWORKING #ifdef WITH_XC_NETWORKING
connect(m_mainUi->urlEdit, SIGNAL(textChanged(QString)), this, SLOT(updateFaviconButtonEnable(QString))); connect(m_mainUi->urlEdit, SIGNAL(textChanged(QString)), this, SLOT(updateFaviconButtonEnable(QString)));
#endif #endif
connect(m_mainUi->tagsList, SIGNAL(tagsEdited()), this, SLOT(setModified()));
connect(m_mainUi->expireCheck, SIGNAL(stateChanged(int)), this, SLOT(setModified())); connect(m_mainUi->expireCheck, SIGNAL(stateChanged(int)), this, SLOT(setModified()));
connect(m_mainUi->expireDatePicker, SIGNAL(dateTimeChanged(QDateTime)), this, SLOT(setModified())); connect(m_mainUi->expireDatePicker, SIGNAL(dateTimeChanged(QDateTime)), this, SLOT(setModified()));
connect(m_mainUi->notesEdit, SIGNAL(textChanged()), this, SLOT(setModified())); connect(m_mainUi->notesEdit, SIGNAL(textChanged()), this, SLOT(setModified()));

View file

@ -25,6 +25,7 @@
#include "TagsEdit.h" #include "TagsEdit.h"
#include "gui/MainWindow.h" #include "gui/MainWindow.h"
#include <QApplication> #include <QApplication>
#include <QClipboard>
#include <QCompleter> #include <QCompleter>
#include <QDebug> #include <QDebug>
#include <QPainter> #include <QPainter>
@ -701,6 +702,7 @@ void TagsEdit::mousePressEvent(QMouseEvent* event)
if (i <= impl->editing_index) { if (i <= impl->editing_index) {
--impl->editing_index; --impl->editing_index;
} }
emit tagsEdited();
found = true; found = true;
break; break;
} }
@ -797,6 +799,15 @@ void TagsEdit::keyPressEvent(QKeyEvent* event)
} else if (event == QKeySequence::SelectNextChar) { } else if (event == QKeySequence::SelectNextChar) {
impl->moveCursor(impl->text_layout.nextCursorPosition(impl->cursor), true); impl->moveCursor(impl->text_layout.nextCursorPosition(impl->cursor), true);
event->accept(); event->accept();
} else if (event == QKeySequence::Paste) {
auto clipboard = QApplication::clipboard();
if (clipboard) {
for (auto tagtext : clipboard->text().split(",")) {
impl->currentText().insert(impl->cursor, tagtext);
impl->editNewTag(impl->editing_index + 1);
}
}
event->accept();
} else { } else {
switch (event->key()) { switch (event->key()) {
case Qt::Key_Left: case Qt::Key_Left:
@ -839,7 +850,10 @@ void TagsEdit::keyPressEvent(QKeyEvent* event)
} }
event->accept(); event->accept();
break; break;
case Qt::Key_Space: case Qt::Key_Return:
case Qt::Key_Enter:
case Qt::Key_Comma:
case Qt::Key_Semicolon:
if (!impl->currentText().isEmpty()) { if (!impl->currentText().isEmpty()) {
impl->editNewTag(impl->editing_index + 1); impl->editNewTag(impl->editing_index + 1);
} }

View file

@ -107,6 +107,8 @@ void TestGui::init()
config()->set(Config::GUI_ShowTrayIcon, true); config()->set(Config::GUI_ShowTrayIcon, true);
// Disable the update check first time alert // Disable the update check first time alert
config()->set(Config::UpdateCheckMessageShown, true); config()->set(Config::UpdateCheckMessageShown, true);
// Disable quick unlock
config()->set(Config::Security_QuickUnlock, false);
// Copy the test database file to the temporary file // Copy the test database file to the temporary file
auto origFilePath = QDir(KEEPASSX_TEST_DATA_DIR).absoluteFilePath("NewDatabase.kdbx"); auto origFilePath = QDir(KEEPASSX_TEST_DATA_DIR).absoluteFilePath("NewDatabase.kdbx");
@ -451,15 +453,15 @@ void TestGui::testEditEntry()
// Test tags // Test tags
auto* tags = editEntryWidget->findChild<TagsEdit*>("tagsList"); auto* tags = editEntryWidget->findChild<TagsEdit*>("tagsList");
QTest::keyClicks(tags, "_tag1"); QTest::keyClicks(tags, "_tag1");
QTest::keyClick(tags, Qt::Key_Space); QTest::keyClick(tags, Qt::Key_Return);
QCOMPARE(tags->tags().last(), QString("_tag1")); QCOMPARE(tags->tags().last(), QString("_tag1"));
QTest::keyClick(tags, Qt::Key_Space); QTest::keyClicks(tags, "tag 2"); // adds another tag
QTest::keyClicks(tags, "_tag2"); // adds another tag QTest::keyClick(tags, Qt::Key_Return);
QCOMPARE(tags->tags().last(), QString("_tag2")); QCOMPARE(tags->tags().last(), QString("tag 2"));
QTest::keyClick(tags, Qt::Key_Backspace); // Back into editing last tag QTest::keyClick(tags, Qt::Key_Backspace); // Back into editing last tag
QTest::keyClicks(tags, "gers"); QTest::keyClicks(tags, "_is!awesome");
QTest::keyClick(tags, Qt::Key_Space); QTest::keyClick(tags, Qt::Key_Return);
QCOMPARE(tags->tags().last(), QString("_taggers")); QCOMPARE(tags->tags().last(), QString("tag 2_is!awesome"));
// Test entry colors (simulate choosing a color) // Test entry colors (simulate choosing a color)
editEntryWidget->setCurrentPage(1); editEntryWidget->setCurrentPage(1);