mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-14 00:39:53 -05:00
Fix crash when removing custom data
This commit is contained in:
parent
72c1783b5b
commit
77be468670
@ -71,6 +71,11 @@ void EditWidgetProperties::setCustomData(CustomData* customData)
|
|||||||
|
|
||||||
void EditWidgetProperties::removeSelectedPluginData()
|
void EditWidgetProperties::removeSelectedPluginData()
|
||||||
{
|
{
|
||||||
|
QModelIndexList indexes = m_ui->customDataTable->selectionModel()->selectedRows(0);
|
||||||
|
if (indexes.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
auto result = MessageBox::question(this,
|
auto result = MessageBox::question(this,
|
||||||
tr("Delete plugin data?"),
|
tr("Delete plugin data?"),
|
||||||
tr("Do you really want to delete the selected plugin data?\n"
|
tr("Do you really want to delete the selected plugin data?\n"
|
||||||
@ -82,15 +87,19 @@ void EditWidgetProperties::removeSelectedPluginData()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const QItemSelectionModel* itemSelectionModel = m_ui->customDataTable->selectionModel();
|
QStringList selectedData;
|
||||||
if (itemSelectionModel) {
|
for (const auto& index : indexes) {
|
||||||
for (const QModelIndex& index : itemSelectionModel->selectedRows(0)) {
|
|
||||||
const QString key = index.data().toString();
|
const QString key = index.data().toString();
|
||||||
|
selectedData.append(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::sort(selectedData.begin(), selectedData.end());
|
||||||
|
for (const auto& key : selectedData) {
|
||||||
m_customData->remove(key);
|
m_customData->remove(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void EditWidgetProperties::toggleRemoveButton(const QItemSelection& selected)
|
void EditWidgetProperties::toggleRemoveButton(const QItemSelection& selected)
|
||||||
{
|
{
|
||||||
|
BIN
tests/data/NewDatabaseBrowser.kdbx
Normal file
BIN
tests/data/NewDatabaseBrowser.kdbx
Normal file
Binary file not shown.
@ -16,5 +16,5 @@
|
|||||||
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/..)
|
include_directories(${CMAKE_CURRENT_SOURCE_DIR}/..)
|
||||||
|
|
||||||
add_unit_test(NAME testgui SOURCES TestGui.cpp ../util/TemporaryFile.cpp LIBS ${TEST_LIBRARIES})
|
add_unit_test(NAME testgui SOURCES TestGui.cpp ../util/TemporaryFile.cpp LIBS ${TEST_LIBRARIES})
|
||||||
|
add_unit_test(NAME testguibrowser SOURCES TestGuiBrowser.cpp ../util/TemporaryFile.cpp LIBS ${TEST_LIBRARIES})
|
||||||
add_unit_test(NAME testguipixmaps SOURCES TestGuiPixmaps.cpp LIBS ${TEST_LIBRARIES})
|
add_unit_test(NAME testguipixmaps SOURCES TestGuiPixmaps.cpp LIBS ${TEST_LIBRARIES})
|
||||||
|
189
tests/gui/TestGuiBrowser.cpp
Normal file
189
tests/gui/TestGuiBrowser.cpp
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2010 Felix Geyer <debfx@fobos.de>
|
||||||
|
* Copyright (C) 2019 KeePassXC Team <team@keepassxc.org>
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 2 or (at your option)
|
||||||
|
* version 3 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "TestGuiBrowser.h"
|
||||||
|
#include "TestGlobal.h"
|
||||||
|
#include "gui/Application.h"
|
||||||
|
|
||||||
|
#include <QAction>
|
||||||
|
#include <QApplication>
|
||||||
|
#include <QDebug>
|
||||||
|
#include <QDialogButtonBox>
|
||||||
|
#include <QLineEdit>
|
||||||
|
#include <QPushButton>
|
||||||
|
#include <QToolBar>
|
||||||
|
#include <QTableView>
|
||||||
|
|
||||||
|
#include "config-keepassx-tests.h"
|
||||||
|
#include "core/Bootstrap.h"
|
||||||
|
#include "core/Config.h"
|
||||||
|
#include "core/Database.h"
|
||||||
|
#include "core/Entry.h"
|
||||||
|
#include "core/Tools.h"
|
||||||
|
#include "crypto/Crypto.h"
|
||||||
|
#include "gui/DatabaseTabWidget.h"
|
||||||
|
#include "gui/DatabaseWidget.h"
|
||||||
|
#include "gui/FileDialog.h"
|
||||||
|
#include "gui/MessageBox.h"
|
||||||
|
#include "gui/PasswordEdit.h"
|
||||||
|
#include "gui/dbsettings/DatabaseSettingsDialog.h"
|
||||||
|
#include "gui/entry/EditEntryWidget.h"
|
||||||
|
#include "gui/entry/EntryView.h"
|
||||||
|
|
||||||
|
QTEST_MAIN(TestGuiBrowser)
|
||||||
|
|
||||||
|
void TestGuiBrowser::initTestCase()
|
||||||
|
{
|
||||||
|
QVERIFY(Crypto::init());
|
||||||
|
Config::createTempFileInstance();
|
||||||
|
// Disable autosave so we can test the modified file indicator
|
||||||
|
config()->set("AutoSaveAfterEveryChange", false);
|
||||||
|
config()->set("AutoSaveOnExit", false);
|
||||||
|
// Enable the tray icon so we can test hiding/restoring the windowQByteArray
|
||||||
|
config()->set("GUI/ShowTrayIcon", true);
|
||||||
|
// Disable advanced settings mode (activate within individual tests to test advanced settings)
|
||||||
|
config()->set("GUI/AdvancedSettings", false);
|
||||||
|
// Disable the update check first time alert
|
||||||
|
config()->set("UpdateCheckMessageShown", true);
|
||||||
|
|
||||||
|
m_mainWindow.reset(new MainWindow());
|
||||||
|
Bootstrap::restoreMainWindowState(*m_mainWindow);
|
||||||
|
m_tabWidget = m_mainWindow->findChild<DatabaseTabWidget*>("tabWidget");
|
||||||
|
m_mainWindow->show();
|
||||||
|
|
||||||
|
// Load the NewDatabase.kdbx file into temporary storage
|
||||||
|
QFile sourceDbFile(QString(KEEPASSX_TEST_DATA_DIR).append("/NewDatabaseBrowser.kdbx"));
|
||||||
|
QVERIFY(sourceDbFile.open(QIODevice::ReadOnly));
|
||||||
|
QVERIFY(Tools::readAllFromDevice(&sourceDbFile, m_dbData));
|
||||||
|
sourceDbFile.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Every test starts with opening the temp database
|
||||||
|
void TestGuiBrowser::init()
|
||||||
|
{
|
||||||
|
m_dbFile.reset(new TemporaryFile());
|
||||||
|
// Write the temp storage to a temp database file for use in our tests
|
||||||
|
QVERIFY(m_dbFile->open());
|
||||||
|
QCOMPARE(m_dbFile->write(m_dbData), static_cast<qint64>((m_dbData.size())));
|
||||||
|
m_dbFileName = QFileInfo(m_dbFile->fileName()).fileName();
|
||||||
|
m_dbFilePath = m_dbFile->fileName();
|
||||||
|
m_dbFile->close();
|
||||||
|
|
||||||
|
// make sure window is activated or focus tests may fail
|
||||||
|
m_mainWindow->activateWindow();
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
|
fileDialog()->setNextFileName(m_dbFilePath);
|
||||||
|
triggerAction("actionDatabaseOpen");
|
||||||
|
|
||||||
|
auto* databaseOpenWidget = m_tabWidget->currentDatabaseWidget()->findChild<QWidget*>("databaseOpenWidget");
|
||||||
|
QVERIFY(databaseOpenWidget);
|
||||||
|
auto* editPassword = databaseOpenWidget->findChild<QLineEdit*>("editPassword");
|
||||||
|
QVERIFY(editPassword);
|
||||||
|
editPassword->setFocus();
|
||||||
|
|
||||||
|
QTest::keyClicks(editPassword, "a");
|
||||||
|
QTest::keyClick(editPassword, Qt::Key_Enter);
|
||||||
|
|
||||||
|
m_dbWidget = m_tabWidget->currentDatabaseWidget();
|
||||||
|
m_db = m_dbWidget->database();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Every test ends with closing the temp database without saving
|
||||||
|
void TestGuiBrowser::cleanup()
|
||||||
|
{
|
||||||
|
// DO NOT save the database
|
||||||
|
m_db->markAsClean();
|
||||||
|
MessageBox::setNextAnswer(MessageBox::No);
|
||||||
|
triggerAction("actionDatabaseClose");
|
||||||
|
QApplication::processEvents();
|
||||||
|
MessageBox::setNextAnswer(MessageBox::NoButton);
|
||||||
|
|
||||||
|
if (m_dbWidget) {
|
||||||
|
delete m_dbWidget;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_dbFile->remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGuiBrowser::cleanupTestCase()
|
||||||
|
{
|
||||||
|
m_dbFile->remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGuiBrowser::testEntrySettings()
|
||||||
|
{
|
||||||
|
auto* toolBar = m_mainWindow->findChild<QToolBar*>("toolBar");
|
||||||
|
auto* entryView = m_dbWidget->findChild<EntryView*>("entryView");
|
||||||
|
|
||||||
|
entryView->setFocus();
|
||||||
|
QVERIFY(entryView->hasFocus());
|
||||||
|
|
||||||
|
// Select the first entry in the database
|
||||||
|
QModelIndex entryItem = entryView->model()->index(0, 1);
|
||||||
|
Entry* entry = entryView->entryFromIndex(entryItem);
|
||||||
|
clickIndex(entryItem, entryView, Qt::LeftButton);
|
||||||
|
|
||||||
|
auto* entryEditAction = m_mainWindow->findChild<QAction*>("actionEntryEdit");
|
||||||
|
QWidget* entryEditWidget = toolBar->widgetForAction(entryEditAction);
|
||||||
|
QTest::mouseClick(entryEditWidget, Qt::LeftButton);
|
||||||
|
QCOMPARE(m_dbWidget->currentMode(), DatabaseWidget::Mode::EditMode);
|
||||||
|
auto* editEntryWidget = m_dbWidget->findChild<EditEntryWidget*>("editEntryWidget");
|
||||||
|
|
||||||
|
// Switch to Properties page and select all rows from the custom data table
|
||||||
|
editEntryWidget->setCurrentPage(4);
|
||||||
|
auto customDataTableView = editEntryWidget->findChild<QTableView*>("customDataTable");
|
||||||
|
QVERIFY(customDataTableView);
|
||||||
|
QTest::mouseClick(customDataTableView, Qt::LeftButton);
|
||||||
|
QTest::keyClick(customDataTableView, 'a', Qt::ControlModifier);
|
||||||
|
|
||||||
|
// Remove the data
|
||||||
|
QCOMPARE(entry->customData()->size(), 2);
|
||||||
|
auto* removeButton = editEntryWidget->findChild<QPushButton*>("removeCustomDataButton");
|
||||||
|
QVERIFY(removeButton);
|
||||||
|
MessageBox::setNextAnswer(MessageBox::Delete);
|
||||||
|
QTest::mouseClick(removeButton, Qt::LeftButton);
|
||||||
|
|
||||||
|
// Apply the removal
|
||||||
|
auto* editEntryWidgetButtonBox = editEntryWidget->findChild<QDialogButtonBox*>("buttonBox");
|
||||||
|
QVERIFY(editEntryWidgetButtonBox);
|
||||||
|
auto* okButton = editEntryWidgetButtonBox->button(QDialogButtonBox::Ok);
|
||||||
|
QVERIFY(okButton);
|
||||||
|
QTRY_VERIFY(okButton->isEnabled());
|
||||||
|
QTest::mouseClick(okButton, Qt::LeftButton);
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
|
QCOMPARE(entry->customData()->size(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGuiBrowser::triggerAction(const QString& name)
|
||||||
|
{
|
||||||
|
auto* action = m_mainWindow->findChild<QAction*>(name);
|
||||||
|
QVERIFY(action);
|
||||||
|
QVERIFY(action->isEnabled());
|
||||||
|
action->trigger();
|
||||||
|
QApplication::processEvents();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGuiBrowser::clickIndex(const QModelIndex& index,
|
||||||
|
QAbstractItemView* view,
|
||||||
|
Qt::MouseButton button,
|
||||||
|
Qt::KeyboardModifiers stateKey)
|
||||||
|
{
|
||||||
|
QTest::mouseClick(view->viewport(), button, stateKey, view->visualRect(index).center());
|
||||||
|
}
|
65
tests/gui/TestGuiBrowser.h
Normal file
65
tests/gui/TestGuiBrowser.h
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2011 Felix Geyer <debfx@fobos.de>
|
||||||
|
* Copyright (C) 2019 KeePassXC Team <team@keepassxc.org>
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 2 or (at your option)
|
||||||
|
* version 3 of the License.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef KEEPASSXC_TESTGUIBROWSER_H
|
||||||
|
#define KEEPASSXC_TESTGUIBROWSER_H
|
||||||
|
|
||||||
|
#include "gui/MainWindow.h"
|
||||||
|
#include "util/TemporaryFile.h"
|
||||||
|
|
||||||
|
#include <QAbstractItemModel>
|
||||||
|
#include <QObject>
|
||||||
|
#include <QPointer>
|
||||||
|
#include <QScopedPointer>
|
||||||
|
#include <QSharedPointer>
|
||||||
|
|
||||||
|
class Database;
|
||||||
|
class DatabaseTabWidget;
|
||||||
|
class DatabaseWidget;
|
||||||
|
class QAbstractItemView;
|
||||||
|
|
||||||
|
class TestGuiBrowser : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void initTestCase();
|
||||||
|
void init();
|
||||||
|
void cleanup();
|
||||||
|
void cleanupTestCase();
|
||||||
|
|
||||||
|
void testEntrySettings();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void triggerAction(const QString& name);
|
||||||
|
void clickIndex(const QModelIndex& index,
|
||||||
|
QAbstractItemView* view,
|
||||||
|
Qt::MouseButton button,
|
||||||
|
Qt::KeyboardModifiers stateKey = 0);
|
||||||
|
|
||||||
|
QScopedPointer<MainWindow> m_mainWindow;
|
||||||
|
QPointer<DatabaseTabWidget> m_tabWidget;
|
||||||
|
QPointer<DatabaseWidget> m_dbWidget;
|
||||||
|
QSharedPointer<Database> m_db;
|
||||||
|
QByteArray m_dbData;
|
||||||
|
QScopedPointer<TemporaryFile> m_dbFile;
|
||||||
|
QString m_dbFileName;
|
||||||
|
QString m_dbFilePath;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // KEEPASSXC_TESTGUIBROWSER_H
|
Loading…
Reference in New Issue
Block a user