mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-05 20:51:21 -05:00
0a3b19edf2
Default option is set to off
262 lines
8.7 KiB
C++
262 lines
8.7 KiB
C++
/*
|
|
* Copyright (C) 2011 Felix Geyer <debfx@fobos.de>
|
|
* Copyright (C) 2017 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 "Config.h"
|
|
|
|
#include <QCoreApplication>
|
|
#include <QDir>
|
|
#include <QSettings>
|
|
#include <QStandardPaths>
|
|
#include <QTemporaryFile>
|
|
|
|
/*
|
|
* Map of configuration file settings that are either deprecated, or have
|
|
* had their name changed. Entries in the map are of the form
|
|
* {oldName, newName}
|
|
* Set newName to empty string to remove the setting from the file.
|
|
*/
|
|
static const QMap<QString, QString> deprecationMap = {
|
|
// >2.3.4
|
|
{QStringLiteral("security/hidepassworddetails"), QStringLiteral("security/HidePasswordPreviewPanel")},
|
|
// >2.3.4
|
|
{QStringLiteral("GUI/HideDetailsView"), QStringLiteral("GUI/HidePreviewPanel")},
|
|
// >2.3.4
|
|
{QStringLiteral("GUI/DetailSplitterState"), QStringLiteral("GUI/PreviewSplitterState")},
|
|
// >2.3.4
|
|
{QStringLiteral("security/IconDownloadFallbackToGoogle"), QStringLiteral("security/IconDownloadFallback")},
|
|
};
|
|
|
|
Config* Config::m_instance(nullptr);
|
|
|
|
QVariant Config::get(const QString& key)
|
|
{
|
|
return m_settings->value(key, m_defaults.value(key));
|
|
}
|
|
|
|
QVariant Config::get(const QString& key, const QVariant& defaultValue)
|
|
{
|
|
return m_settings->value(key, defaultValue);
|
|
}
|
|
|
|
bool Config::hasAccessError()
|
|
{
|
|
return m_settings->status() & QSettings::AccessError;
|
|
}
|
|
|
|
QString Config::getFileName()
|
|
{
|
|
return m_settings->fileName();
|
|
}
|
|
|
|
void Config::set(const QString& key, const QVariant& value)
|
|
{
|
|
if (m_settings->contains(key) && m_settings->value(key) == value) {
|
|
return;
|
|
}
|
|
const bool surpressSignal = !m_settings->contains(key) && m_defaults.value(key) == value;
|
|
|
|
m_settings->setValue(key, value);
|
|
|
|
if (!surpressSignal) {
|
|
emit changed(key);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sync configuration with persistent storage.
|
|
*
|
|
* Usually, you don't need to call this method manually, but if you are writing
|
|
* configurations after an emitted \link QCoreApplication::aboutToQuit() signal,
|
|
* use it to guarantee your config values are persisted.
|
|
*/
|
|
void Config::sync()
|
|
{
|
|
m_settings->sync();
|
|
}
|
|
|
|
void Config::resetToDefaults()
|
|
{
|
|
for (const auto& setting : m_defaults.keys()) {
|
|
m_settings->setValue(setting, m_defaults.value(setting));
|
|
}
|
|
}
|
|
|
|
void Config::upgrade()
|
|
{
|
|
const auto keys = deprecationMap.keys();
|
|
for (const auto& setting : keys) {
|
|
if (m_settings->contains(setting)) {
|
|
if (!deprecationMap.value(setting).isEmpty()) {
|
|
// Add entry with new name and old entry's value
|
|
m_settings->setValue(deprecationMap.value(setting), m_settings->value(setting));
|
|
}
|
|
m_settings->remove(setting);
|
|
}
|
|
}
|
|
|
|
// > 2.3.4
|
|
if (m_settings->value("AutoSaveAfterEveryChange").toBool()) {
|
|
m_settings->setValue("AutoSaveOnExit", true);
|
|
}
|
|
|
|
// Setting defaults for 'hide window on copy' behavior, keeping the user's original setting
|
|
if (m_settings->value("HideWindowOnCopy").isNull()) {
|
|
m_settings->setValue("HideWindowOnCopy", m_settings->value("MinimizeOnCopy").toBool());
|
|
m_settings->setValue("MinimizeOnCopy", true);
|
|
}
|
|
}
|
|
|
|
Config::Config(const QString& fileName, QObject* parent)
|
|
: QObject(parent)
|
|
{
|
|
init(fileName);
|
|
}
|
|
|
|
Config::Config(QObject* parent)
|
|
: QObject(parent)
|
|
{
|
|
// Check if portable config is present. If not, find it in user's directory
|
|
QString portablePath = QCoreApplication::applicationDirPath() + "/keepassxc.ini";
|
|
if (QFile::exists(portablePath)) {
|
|
init(portablePath);
|
|
} else {
|
|
QString userPath;
|
|
QString homePath = QDir::homePath();
|
|
|
|
#if defined(Q_OS_UNIX) && !defined(Q_OS_MACOS)
|
|
// we can't use QStandardPaths on X11 as it uses XDG_DATA_HOME instead of XDG_CONFIG_HOME
|
|
QByteArray env = qgetenv("XDG_CONFIG_HOME");
|
|
if (env.isEmpty()) {
|
|
userPath = homePath;
|
|
userPath += "/.config";
|
|
} else if (env[0] == '/') {
|
|
userPath = QFile::decodeName(env);
|
|
} else {
|
|
userPath = homePath;
|
|
userPath += '/';
|
|
userPath += QFile::decodeName(env);
|
|
}
|
|
|
|
userPath += "/keepassxc/";
|
|
#else
|
|
userPath = QDir::fromNativeSeparators(QStandardPaths::writableLocation(QStandardPaths::DataLocation));
|
|
// storageLocation() appends the application name ("/keepassxc") to the end
|
|
userPath += "/";
|
|
#endif
|
|
|
|
#ifdef QT_DEBUG
|
|
userPath += "keepassxc_debug.ini";
|
|
#else
|
|
userPath += "keepassxc.ini";
|
|
#endif
|
|
|
|
init(userPath);
|
|
}
|
|
}
|
|
|
|
Config::~Config()
|
|
{
|
|
}
|
|
|
|
void Config::init(const QString& fileName)
|
|
{
|
|
m_settings.reset(new QSettings(fileName, QSettings::IniFormat));
|
|
upgrade();
|
|
connect(qApp, &QCoreApplication::aboutToQuit, this, &Config::sync);
|
|
|
|
m_defaults.insert("SingleInstance", true);
|
|
m_defaults.insert("RememberLastDatabases", true);
|
|
m_defaults.insert("NumberOfRememberedLastDatabases", 5);
|
|
m_defaults.insert("RememberLastKeyFiles", true);
|
|
m_defaults.insert("OpenPreviousDatabasesOnStartup", true);
|
|
m_defaults.insert("AutoSaveAfterEveryChange", true);
|
|
m_defaults.insert("AutoReloadOnChange", true);
|
|
m_defaults.insert("AutoSaveOnExit", true);
|
|
m_defaults.insert("BackupBeforeSave", false);
|
|
m_defaults.insert("UseAtomicSaves", true);
|
|
m_defaults.insert("SearchLimitGroup", false);
|
|
m_defaults.insert("MinimizeOnOpenUrl", false);
|
|
m_defaults.insert("HideWindowOnCopy", false);
|
|
m_defaults.insert("MinimizeOnCopy", true);
|
|
m_defaults.insert("MinimizeAfterUnlock", false);
|
|
m_defaults.insert("DropToBackgroundOnCopy", false);
|
|
m_defaults.insert("UseGroupIconOnEntryCreation", false);
|
|
m_defaults.insert("AutoTypeEntryTitleMatch", true);
|
|
m_defaults.insert("AutoTypeEntryURLMatch", true);
|
|
m_defaults.insert("AutoTypeDelay", 25);
|
|
m_defaults.insert("AutoTypeStartDelay", 500);
|
|
m_defaults.insert("UseGroupIconOnEntryCreation", true);
|
|
m_defaults.insert("IgnoreGroupExpansion", true);
|
|
m_defaults.insert("FaviconDownloadTimeout", 10);
|
|
m_defaults.insert("security/clearclipboard", true);
|
|
m_defaults.insert("security/clearclipboardtimeout", 10);
|
|
m_defaults.insert("security/clearsearch", true);
|
|
m_defaults.insert("security/clearsearchtimeout", 5);
|
|
m_defaults.insert("security/lockdatabaseidle", false);
|
|
m_defaults.insert("security/lockdatabaseidlesec", 240);
|
|
m_defaults.insert("security/lockdatabaseminimize", false);
|
|
m_defaults.insert("security/lockdatabasescreenlock", true);
|
|
m_defaults.insert("security/passwordsrepeat", false);
|
|
m_defaults.insert("security/passwordscleartext", false);
|
|
m_defaults.insert("security/passwordemptynodots", true);
|
|
m_defaults.insert("security/HidePasswordPreviewPanel", true);
|
|
m_defaults.insert("security/autotypeask", true);
|
|
m_defaults.insert("security/IconDownloadFallback", false);
|
|
m_defaults.insert("security/resettouchid", false);
|
|
m_defaults.insert("security/resettouchidtimeout", 30);
|
|
m_defaults.insert("security/resettouchidscreenlock", true);
|
|
m_defaults.insert("GUI/Language", "system");
|
|
m_defaults.insert("GUI/HideToolbar", false);
|
|
m_defaults.insert("GUI/MovableToolbar", false);
|
|
m_defaults.insert("GUI/ToolButtonStyle", Qt::ToolButtonIconOnly);
|
|
m_defaults.insert("GUI/ShowTrayIcon", false);
|
|
m_defaults.insert("GUI/DarkTrayIcon", false);
|
|
m_defaults.insert("GUI/MinimizeToTray", false);
|
|
m_defaults.insert("GUI/MinimizeOnClose", false);
|
|
m_defaults.insert("GUI/HideUsernames", false);
|
|
m_defaults.insert("GUI/HidePasswords", true);
|
|
m_defaults.insert("GUI/AdvancedSettings", false);
|
|
m_defaults.insert("GUI/MonospaceNotes", false);
|
|
}
|
|
|
|
Config* Config::instance()
|
|
{
|
|
if (!m_instance) {
|
|
m_instance = new Config(qApp);
|
|
}
|
|
|
|
return m_instance;
|
|
}
|
|
|
|
void Config::createConfigFromFile(const QString& file)
|
|
{
|
|
Q_ASSERT(!m_instance);
|
|
m_instance = new Config(file, qApp);
|
|
}
|
|
|
|
void Config::createTempFileInstance()
|
|
{
|
|
Q_ASSERT(!m_instance);
|
|
auto* tmpFile = new QTemporaryFile();
|
|
bool openResult = tmpFile->open();
|
|
Q_ASSERT(openResult);
|
|
Q_UNUSED(openResult);
|
|
m_instance = new Config(tmpFile->fileName(), qApp);
|
|
tmpFile->setParent(m_instance);
|
|
}
|