keepassxc/src/gui/EntryPreviewWidget.cpp

353 lines
12 KiB
C++
Raw Normal View History

/*
* Copyright (C) 2012 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 "EntryPreviewWidget.h"
#include "ui_EntryPreviewWidget.h"
2017-12-13 15:22:31 -05:00
#include <QDesktopServices>
2018-03-31 16:01:30 -04:00
#include <QDir>
#include "core/Config.h"
#include "core/FilePath.h"
2017-12-13 15:22:31 -05:00
#include "entry/EntryAttachmentsModel.h"
2018-03-31 16:01:30 -04:00
#include "gui/Clipboard.h"
#if defined(WITH_XC_KEESHARE)
#include "keeshare/KeeShare.h"
#endif
2018-03-31 16:01:30 -04:00
namespace
{
constexpr int GeneralTabIndex = 0;
2017-12-25 08:53:22 -05:00
}
EntryPreviewWidget::EntryPreviewWidget(QWidget* parent)
: QWidget(parent)
, m_ui(new Ui::EntryPreviewWidget())
, m_locked(false)
, m_currentEntry(nullptr)
, m_currentGroup(nullptr)
, m_selectedTabEntry(0)
, m_selectedTabGroup(0)
{
m_ui->setupUi(this);
2017-12-25 08:53:22 -05:00
// Entry
m_ui->entryTotpButton->setIcon(filePath()->icon("actions", "chronometer"));
m_ui->entryCloseButton->setIcon(filePath()->icon("actions", "dialog-close"));
m_ui->togglePasswordButton->setIcon(filePath()->onOffIcon("actions", "password-show"));
2017-12-25 08:53:22 -05:00
m_ui->entryAttachmentsWidget->setReadOnly(true);
m_ui->entryAttachmentsWidget->setButtonsVisible(false);
2017-12-13 15:22:31 -05:00
2017-12-25 08:53:22 -05:00
connect(m_ui->entryTotpButton, SIGNAL(toggled(bool)), m_ui->entryTotpWidget, SLOT(setVisible(bool)));
connect(m_ui->entryCloseButton, SIGNAL(clicked()), SLOT(hide()));
connect(m_ui->togglePasswordButton, SIGNAL(clicked(bool)), SLOT(setPasswordVisible(bool)));
2017-12-25 08:53:22 -05:00
connect(m_ui->entryTabWidget, SIGNAL(tabBarClicked(int)), SLOT(updateTabIndexes()), Qt::QueuedConnection);
connect(&m_totpTimer, SIGNAL(timeout()), this, SLOT(updateTotpLabel()));
2017-12-13 15:22:31 -05:00
2017-12-25 08:53:22 -05:00
// Group
m_ui->groupCloseButton->setIcon(filePath()->icon("actions", "dialog-close"));
connect(m_ui->groupCloseButton, SIGNAL(clicked()), SLOT(hide()));
2017-12-25 08:53:22 -05:00
connect(m_ui->groupTabWidget, SIGNAL(tabBarClicked(int)), SLOT(updateTabIndexes()), Qt::QueuedConnection);
}
EntryPreviewWidget::~EntryPreviewWidget()
{
}
void EntryPreviewWidget::setEntry(Entry* selectedEntry)
{
2017-10-28 14:57:11 -04:00
if (!selectedEntry) {
2017-12-25 08:53:22 -05:00
hide();
2017-10-28 14:57:11 -04:00
return;
}
m_currentEntry = selectedEntry;
2017-12-25 08:53:22 -05:00
updateEntryHeaderLine();
updateEntryTotp();
updateEntryGeneralTab();
updateEntryNotesTab();
updateEntryAttributesTab();
updateEntryAttachmentsTab();
updateEntryAutotypeTab();
setVisible(!config()->get("GUI/HidePreviewPanel").toBool());
2017-12-25 08:53:22 -05:00
m_ui->stackedWidget->setCurrentWidget(m_ui->pageEntry);
2018-03-31 16:01:30 -04:00
const int tabIndex = m_ui->entryTabWidget->isTabEnabled(m_selectedTabEntry) ? m_selectedTabEntry : GeneralTabIndex;
2017-12-25 08:53:22 -05:00
Q_ASSERT(m_ui->entryTabWidget->isTabEnabled(GeneralTabIndex));
m_ui->entryTabWidget->setCurrentIndex(tabIndex);
}
void EntryPreviewWidget::setGroup(Group* selectedGroup)
2017-12-25 08:53:22 -05:00
{
if (!selectedGroup) {
hide();
return;
}
2017-12-25 08:53:22 -05:00
m_currentGroup = selectedGroup;
updateGroupHeaderLine();
updateGroupGeneralTab();
updateGroupNotesTab();
2017-09-27 13:18:13 -04:00
#if defined(WITH_XC_KEESHARE)
updateGroupSharingTab();
#endif
setVisible(!config()->get("GUI/HidePreviewPanel").toBool());
2017-12-25 08:53:22 -05:00
m_ui->stackedWidget->setCurrentWidget(m_ui->pageGroup);
2018-03-31 16:01:30 -04:00
const int tabIndex = m_ui->groupTabWidget->isTabEnabled(m_selectedTabGroup) ? m_selectedTabGroup : GeneralTabIndex;
2017-12-25 08:53:22 -05:00
Q_ASSERT(m_ui->groupTabWidget->isTabEnabled(GeneralTabIndex));
m_ui->groupTabWidget->setCurrentIndex(tabIndex);
}
void EntryPreviewWidget::setDatabaseMode(DatabaseWidget::Mode mode)
2017-12-25 08:53:22 -05:00
{
m_locked = mode == DatabaseWidget::Mode::LockedMode;
2017-12-25 08:53:22 -05:00
if (m_locked) {
return;
}
2017-12-25 08:53:22 -05:00
if (mode == DatabaseWidget::Mode::ViewMode) {
2017-12-25 08:53:22 -05:00
if (m_ui->stackedWidget->currentWidget() == m_ui->pageGroup) {
setGroup(m_currentGroup);
} else {
setEntry(m_currentEntry);
}
}
2017-12-25 08:53:22 -05:00
}
void EntryPreviewWidget::updateEntryHeaderLine()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
const QString title = m_currentEntry->resolveMultiplePlaceholders(m_currentEntry->title());
m_ui->entryTitleLabel->setRawText(hierarchy(m_currentEntry->group(), title));
2017-12-25 08:53:22 -05:00
m_ui->entryIcon->setPixmap(preparePixmap(m_currentEntry->iconPixmap(), 16));
}
void EntryPreviewWidget::updateEntryTotp()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
const bool hasTotp = m_currentEntry->hasTotp();
m_ui->entryTotpButton->setVisible(hasTotp);
m_ui->entryTotpWidget->hide();
m_ui->entryTotpButton->setChecked(false);
if (hasTotp) {
m_totpTimer.start(1000);
2017-12-25 08:53:22 -05:00
updateTotpLabel();
} else {
m_ui->entryTotpLabel->clear();
m_totpTimer.stop();
2017-12-25 08:53:22 -05:00
}
}
void EntryPreviewWidget::setPasswordVisible(bool state)
2017-12-25 08:53:22 -05:00
{
const QString password = m_currentEntry->resolveMultiplePlaceholders(m_currentEntry->password());
auto flags = m_ui->entryPasswordLabel->textInteractionFlags();
if (state) {
2017-12-29 03:46:16 -05:00
m_ui->entryPasswordLabel->setRawText(password);
2017-12-25 08:53:22 -05:00
m_ui->entryPasswordLabel->setToolTip(password);
m_ui->entryPasswordLabel->setTextInteractionFlags(flags | Qt::TextSelectableByMouse);
} else {
m_ui->entryPasswordLabel->setTextInteractionFlags(flags & ~Qt::TextSelectableByMouse);
2017-12-29 03:46:16 -05:00
m_ui->entryPasswordLabel->setToolTip({});
if (password.isEmpty() && config()->get("security/passwordemptynodots").toBool()) {
m_ui->entryPasswordLabel->setRawText("");
} else {
m_ui->entryPasswordLabel->setRawText(QString("\u25cf").repeated(6));
}
}
}
void EntryPreviewWidget::updateEntryGeneralTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
m_ui->entryUsernameLabel->setText(m_currentEntry->resolveMultiplePlaceholders(m_currentEntry->username()));
if (config()->get("security/HidePasswordPreviewPanel").toBool()) {
// Hide password
setPasswordVisible(false);
// Show the password toggle button if there are dots in the label
m_ui->togglePasswordButton->setVisible(!m_ui->entryPasswordLabel->rawText().isEmpty());
m_ui->togglePasswordButton->setChecked(false);
} else {
// Show password
setPasswordVisible(true);
m_ui->togglePasswordButton->setVisible(false);
}
m_ui->entryUrlLabel->setRawText(m_currentEntry->displayUrl());
2017-12-29 03:46:16 -05:00
const QString url = m_currentEntry->webUrl();
if (!url.isEmpty()) {
// URL is well formed and can be opened in a browser
2017-12-29 03:46:16 -05:00
m_ui->entryUrlLabel->setUrl(url);
m_ui->entryUrlLabel->setCursor(Qt::PointingHandCursor);
} else {
2017-12-29 03:46:16 -05:00
m_ui->entryUrlLabel->setUrl({});
m_ui->entryUrlLabel->setCursor(Qt::ArrowCursor);
}
2017-12-25 08:53:22 -05:00
const TimeInfo entryTime = m_currentEntry->timeInfo();
2018-03-31 16:01:30 -04:00
const QString expires =
entryTime.expires() ? entryTime.expiryTime().toLocalTime().toString(Qt::DefaultLocaleShortDate) : tr("Never");
2017-12-25 08:53:22 -05:00
m_ui->entryExpirationLabel->setText(expires);
}
2017-09-27 13:18:13 -04:00
void EntryPreviewWidget::updateEntryNotesTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
const QString notes = m_currentEntry->notes();
setTabEnabled(m_ui->entryTabWidget, m_ui->entryNotesTab, !notes.isEmpty());
m_ui->entryNotesEdit->setText(notes);
2017-12-25 08:53:22 -05:00
}
2017-09-27 13:18:13 -04:00
void EntryPreviewWidget::updateEntryAttributesTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
m_ui->entryAttributesEdit->clear();
const EntryAttributes* attributes = m_currentEntry->attributes();
const QStringList customAttributes = attributes->customKeys();
const bool haveAttributes = !customAttributes.isEmpty();
2017-12-25 08:53:22 -05:00
setTabEnabled(m_ui->entryTabWidget, m_ui->entryAttributesTab, haveAttributes);
if (haveAttributes) {
QString attributesText;
2017-09-27 13:18:13 -04:00
for (const QString& key : customAttributes) {
QString value = m_currentEntry->attributes()->value(key);
if (m_currentEntry->attributes()->isProtected(key)) {
2017-10-24 08:08:22 -04:00
value = "<i>" + tr("[PROTECTED]") + "</i>";
2017-09-27 13:18:13 -04:00
}
attributesText.append(tr("<b>%1</b>: %2", "attributes line").arg(key, value).append("<br/>"));
2017-09-27 13:18:13 -04:00
}
2017-12-25 08:53:22 -05:00
m_ui->entryAttributesEdit->setText(attributesText);
2017-09-27 13:18:13 -04:00
}
2017-12-25 08:53:22 -05:00
}
void EntryPreviewWidget::updateEntryAttachmentsTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
2017-12-13 15:22:31 -05:00
const bool hasAttachments = !m_currentEntry->attachments()->isEmpty();
2017-12-25 08:53:22 -05:00
setTabEnabled(m_ui->entryTabWidget, m_ui->entryAttachmentsTab, hasAttachments);
m_ui->entryAttachmentsWidget->setEntryAttachments(m_currentEntry->attachments());
}
2017-12-13 15:22:31 -05:00
void EntryPreviewWidget::updateEntryAutotypeTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentEntry);
m_ui->entryAutotypeTree->clear();
2017-10-24 08:08:22 -04:00
QList<QTreeWidgetItem*> items;
2017-12-25 08:53:22 -05:00
const AutoTypeAssociations* autotypeAssociations = m_currentEntry->autoTypeAssociations();
const auto associations = autotypeAssociations->getAll();
for (const auto& assoc : associations) {
2018-03-31 16:01:30 -04:00
const QString sequence =
assoc.sequence.isEmpty() ? m_currentEntry->effectiveAutoTypeSequence() : assoc.sequence;
2017-12-25 08:53:22 -05:00
items.append(new QTreeWidgetItem(m_ui->entryAutotypeTree, {assoc.window, sequence}));
}
2017-12-25 08:53:22 -05:00
m_ui->entryAutotypeTree->addTopLevelItems(items);
setTabEnabled(m_ui->entryTabWidget, m_ui->entryAutotypeTab, !items.isEmpty());
}
void EntryPreviewWidget::updateGroupHeaderLine()
{
2017-12-25 08:53:22 -05:00
Q_ASSERT(m_currentGroup);
m_ui->groupTitleLabel->setRawText(hierarchy(m_currentGroup, {}));
2017-12-25 08:53:22 -05:00
m_ui->groupIcon->setPixmap(preparePixmap(m_currentGroup->iconPixmap(), 32));
}
void EntryPreviewWidget::updateGroupGeneralTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentGroup);
const QString searchingText = m_currentGroup->resolveSearchingEnabled() ? tr("Enabled") : tr("Disabled");
m_ui->groupSearchingLabel->setText(searchingText);
2017-12-25 08:53:22 -05:00
const QString autotypeText = m_currentGroup->resolveAutoTypeEnabled() ? tr("Enabled") : tr("Disabled");
m_ui->groupAutotypeLabel->setText(autotypeText);
2017-12-25 08:53:22 -05:00
const TimeInfo groupTime = m_currentGroup->timeInfo();
2018-03-31 16:01:30 -04:00
const QString expiresText =
groupTime.expires() ? groupTime.expiryTime().toString(Qt::DefaultLocaleShortDate) : tr("Never");
2017-12-25 08:53:22 -05:00
m_ui->groupExpirationLabel->setText(expiresText);
}
void EntryPreviewWidget::updateGroupNotesTab()
2017-12-25 08:53:22 -05:00
{
Q_ASSERT(m_currentGroup);
const QString notes = m_currentGroup->notes();
setTabEnabled(m_ui->groupTabWidget, m_ui->groupNotesTab, !notes.isEmpty());
m_ui->groupNotesEdit->setText(notes);
}
#if defined(WITH_XC_KEESHARE)
void EntryPreviewWidget::updateGroupSharingTab()
{
Q_ASSERT(m_currentGroup);
setTabEnabled(m_ui->groupTabWidget, m_ui->groupShareTab, KeeShare::isShared(m_currentGroup));
auto reference = KeeShare::referenceOf(m_currentGroup);
m_ui->groupShareTypeLabel->setText(KeeShare::referenceTypeLabel(reference));
m_ui->groupSharePathLabel->setText(reference.path);
}
#endif
void EntryPreviewWidget::updateTotpLabel()
2017-12-25 08:53:22 -05:00
{
if (!m_locked && m_currentEntry && m_currentEntry->hasTotp()) {
2017-12-25 08:53:22 -05:00
const QString totpCode = m_currentEntry->totp();
const QString firstHalf = totpCode.left(totpCode.size() / 2);
const QString secondHalf = totpCode.mid(totpCode.size() / 2);
m_ui->entryTotpLabel->setText(firstHalf + " " + secondHalf);
} else {
2017-12-25 08:53:22 -05:00
m_ui->entryTotpLabel->clear();
m_totpTimer.stop();
}
2017-12-25 08:53:22 -05:00
}
void EntryPreviewWidget::updateTabIndexes()
2017-12-25 08:53:22 -05:00
{
m_selectedTabEntry = m_ui->entryTabWidget->currentIndex();
m_selectedTabGroup = m_ui->groupTabWidget->currentIndex();
}
void EntryPreviewWidget::setTabEnabled(QTabWidget* tabWidget, QWidget* widget, bool enabled)
{
2017-12-25 08:53:22 -05:00
const int tabIndex = tabWidget->indexOf(widget);
Q_ASSERT(tabIndex != -1);
tabWidget->setTabEnabled(tabIndex, enabled);
}
QPixmap EntryPreviewWidget::preparePixmap(const QPixmap& pixmap, int size)
{
2017-12-25 08:53:22 -05:00
if (pixmap.width() > size || pixmap.height() > size) {
return pixmap.scaled(size, size, Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
}
2017-12-25 08:53:22 -05:00
return pixmap;
}
QString EntryPreviewWidget::hierarchy(const Group* group, const QString& title)
{
2017-12-29 03:46:16 -05:00
const QString separator(" / ");
2017-12-25 08:53:22 -05:00
QStringList hierarchy = group->hierarchy();
hierarchy.removeFirst();
hierarchy.append(title);
2017-12-29 03:46:16 -05:00
return QString("%1%2").arg(separator, hierarchy.join(separator));
2017-10-25 23:29:01 -04:00
}