2017-08-17 15:02:21 -04:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
#include "EntryPreviewWidget.h"
|
2019-06-28 22:10:47 -04:00
|
|
|
#include "Font.h"
|
2018-10-08 21:11:44 -04:00
|
|
|
#include "ui_EntryPreviewWidget.h"
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2017-12-13 15:22:31 -05:00
|
|
|
#include <QDesktopServices>
|
2018-03-31 16:01:30 -04:00
|
|
|
#include <QDir>
|
2017-08-17 15:02:21 -04:00
|
|
|
|
|
|
|
#include "core/Config.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"
|
2020-10-05 20:41:00 -04:00
|
|
|
#include "gui/Icons.h"
|
2019-01-03 02:46:32 -05:00
|
|
|
#if defined(WITH_XC_KEESHARE)
|
2018-10-01 10:26:24 -04:00
|
|
|
#include "keeshare/KeeShare.h"
|
|
|
|
#endif
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2019-01-20 09:50:20 -05:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
constexpr int GeneralTabIndex = 0;
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
EntryPreviewWidget::EntryPreviewWidget(QWidget* parent)
|
2019-01-16 22:53:29 -05:00
|
|
|
: QWidget(parent)
|
|
|
|
, m_ui(new Ui::EntryPreviewWidget())
|
|
|
|
, m_locked(false)
|
|
|
|
, m_currentEntry(nullptr)
|
|
|
|
, m_currentGroup(nullptr)
|
|
|
|
, m_selectedTabEntry(0)
|
|
|
|
, m_selectedTabGroup(0)
|
2017-08-17 15:02:21 -04:00
|
|
|
{
|
|
|
|
m_ui->setupUi(this);
|
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
// Entry
|
2020-10-05 20:41:00 -04:00
|
|
|
m_ui->entryTotpButton->setIcon(icons()->icon("chronometer"));
|
|
|
|
m_ui->entryCloseButton->setIcon(icons()->icon("dialog-close"));
|
2021-01-31 14:08:03 -05:00
|
|
|
m_ui->togglePasswordButton->setIcon(icons()->onOffIcon("password-show", true));
|
|
|
|
m_ui->toggleEntryNotesButton->setIcon(icons()->onOffIcon("password-show", true));
|
|
|
|
m_ui->toggleGroupNotesButton->setIcon(icons()->onOffIcon("password-show", true));
|
2017-08-17 15:02:21 -04:00
|
|
|
|
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
|
|
|
|
2019-11-09 12:16:05 -05:00
|
|
|
// Match background of read-only text edit fields with the window
|
|
|
|
m_ui->entryPasswordLabel->setBackgroundRole(QPalette::Window);
|
|
|
|
m_ui->entryUsernameLabel->setBackgroundRole(QPalette::Window);
|
|
|
|
m_ui->entryNotesTextEdit->setBackgroundRole(QPalette::Window);
|
|
|
|
m_ui->groupNotesTextEdit->setBackgroundRole(QPalette::Window);
|
|
|
|
// Align notes text with label text
|
|
|
|
m_ui->entryNotesTextEdit->document()->setDocumentMargin(0);
|
|
|
|
m_ui->groupNotesTextEdit->document()->setDocumentMargin(0);
|
|
|
|
|
2019-05-18 10:55:57 -04:00
|
|
|
connect(m_ui->entryUrlLabel, SIGNAL(linkActivated(QString)), SLOT(openEntryUrl()));
|
|
|
|
|
2019-06-23 10:02:02 -04:00
|
|
|
connect(m_ui->entryTotpButton, SIGNAL(toggled(bool)), m_ui->entryTotpLabel, SLOT(setVisible(bool)));
|
2018-10-24 08:11:18 -04:00
|
|
|
connect(m_ui->entryCloseButton, SIGNAL(clicked()), SLOT(hide()));
|
2018-11-16 10:00:59 -05:00
|
|
|
connect(m_ui->togglePasswordButton, SIGNAL(clicked(bool)), SLOT(setPasswordVisible(bool)));
|
2019-06-28 21:48:25 -04:00
|
|
|
connect(m_ui->toggleEntryNotesButton, SIGNAL(clicked(bool)), SLOT(setEntryNotesVisible(bool)));
|
|
|
|
connect(m_ui->toggleGroupNotesButton, SIGNAL(clicked(bool)), SLOT(setGroupNotesVisible(bool)));
|
2017-12-25 08:53:22 -05:00
|
|
|
connect(m_ui->entryTabWidget, SIGNAL(tabBarClicked(int)), SLOT(updateTabIndexes()), Qt::QueuedConnection);
|
2019-05-18 10:55:57 -04:00
|
|
|
connect(&m_totpTimer, SIGNAL(timeout()), SLOT(updateTotpLabel()));
|
2017-12-13 15:22:31 -05:00
|
|
|
|
2020-06-28 10:01:24 -04:00
|
|
|
connect(config(), &Config::changed, this, [this](Config::ConfigKey key) {
|
|
|
|
if (key == Config::GUI_HidePreviewPanel) {
|
|
|
|
setVisible(!config()->get(Config::GUI_HidePreviewPanel).toBool());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
// Group
|
2020-10-05 20:41:00 -04:00
|
|
|
m_ui->groupCloseButton->setIcon(icons()->icon("dialog-close"));
|
2018-10-24 08:11:18 -04:00
|
|
|
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);
|
2019-10-22 11:47:43 -04:00
|
|
|
|
2020-04-21 18:46:21 -04:00
|
|
|
setFocusProxy(m_ui->entryTabWidget);
|
|
|
|
|
2019-10-22 11:47:43 -04:00
|
|
|
#if !defined(WITH_XC_KEESHARE)
|
|
|
|
removeTab(m_ui->groupTabWidget, m_ui->groupShareTab);
|
|
|
|
#endif
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
EntryPreviewWidget::~EntryPreviewWidget()
|
2017-08-17 15:02:21 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::setEntry(Entry* selectedEntry)
|
2017-08-17 15:02:21 -04:00
|
|
|
{
|
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;
|
|
|
|
}
|
|
|
|
|
2017-08-17 15:02:21 -04:00
|
|
|
m_currentEntry = selectedEntry;
|
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
updateEntryHeaderLine();
|
|
|
|
updateEntryTotp();
|
|
|
|
updateEntryGeneralTab();
|
2019-06-23 10:02:02 -04:00
|
|
|
updateEntryAdvancedTab();
|
2017-12-25 08:53:22 -05:00
|
|
|
updateEntryAutotypeTab();
|
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
setVisible(!config()->get(Config::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);
|
|
|
|
}
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::setGroup(Group* selectedGroup)
|
2017-12-25 08:53:22 -05:00
|
|
|
{
|
|
|
|
if (!selectedGroup) {
|
|
|
|
hide();
|
|
|
|
return;
|
2017-09-29 14:37:37 -04:00
|
|
|
}
|
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
m_currentGroup = selectedGroup;
|
|
|
|
updateGroupHeaderLine();
|
|
|
|
updateGroupGeneralTab();
|
2017-09-27 13:18:13 -04:00
|
|
|
|
2019-01-03 02:46:32 -05:00
|
|
|
#if defined(WITH_XC_KEESHARE)
|
2018-10-01 10:26:24 -04:00
|
|
|
updateGroupSharingTab();
|
|
|
|
#endif
|
2018-12-18 22:28:56 -05:00
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
setVisible(!config()->get(Config::GUI_HidePreviewPanel).toBool());
|
2017-08-17 15:02:21 -04:00
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::setDatabaseMode(DatabaseWidget::Mode mode)
|
2017-12-25 08:53:22 -05:00
|
|
|
{
|
2018-11-22 05:47:31 -05:00
|
|
|
m_locked = mode == DatabaseWidget::Mode::LockedMode;
|
2017-12-25 08:53:22 -05:00
|
|
|
if (m_locked) {
|
|
|
|
return;
|
2017-09-28 14:48:49 -04:00
|
|
|
}
|
2017-12-25 08:53:22 -05:00
|
|
|
|
2018-11-22 05:47:31 -05:00
|
|
|
if (mode == DatabaseWidget::Mode::ViewMode) {
|
2020-01-10 22:28:31 -05:00
|
|
|
if (m_currentGroup && m_ui->stackedWidget->currentWidget() == m_ui->pageGroup) {
|
2017-12-25 08:53:22 -05:00
|
|
|
setGroup(m_currentGroup);
|
2020-01-10 22:28:31 -05:00
|
|
|
} else if (m_currentEntry) {
|
2017-12-25 08:53:22 -05:00
|
|
|
setEntry(m_currentEntry);
|
2020-01-10 22:28:31 -05:00
|
|
|
} else {
|
|
|
|
hide();
|
2017-08-30 18:52:05 -04:00
|
|
|
}
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04: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());
|
2018-02-24 20:00:31 -05:00
|
|
|
m_ui->entryTitleLabel->setRawText(hierarchy(m_currentEntry->group(), title));
|
2020-05-29 10:00:07 -04:00
|
|
|
m_ui->entryIcon->setPixmap(m_currentEntry->iconPixmap(IconSize::Large));
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
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);
|
2019-06-23 10:02:02 -04:00
|
|
|
m_ui->entryTotpLabel->hide();
|
2017-12-25 08:53:22 -05:00
|
|
|
m_ui->entryTotpButton->setChecked(false);
|
|
|
|
|
|
|
|
if (hasTotp) {
|
2018-09-05 16:20:57 -04:00
|
|
|
m_totpTimer.start(1000);
|
2017-12-25 08:53:22 -05:00
|
|
|
updateTotpLabel();
|
|
|
|
} else {
|
|
|
|
m_ui->entryTotpLabel->clear();
|
2018-09-05 16:20:57 -04:00
|
|
|
m_totpTimer.stop();
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
|
|
|
}
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2018-11-16 10:00:59 -05:00
|
|
|
void EntryPreviewWidget::setPasswordVisible(bool state)
|
2017-12-25 08:53:22 -05:00
|
|
|
{
|
2018-11-16 10:00:59 -05:00
|
|
|
const QString password = m_currentEntry->resolveMultiplePlaceholders(m_currentEntry->password());
|
|
|
|
if (state) {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->entryPasswordLabel->setText(password);
|
|
|
|
m_ui->entryPasswordLabel->setCursorPosition(0);
|
2020-09-20 11:28:03 -04:00
|
|
|
m_ui->entryPasswordLabel->setFont(Font::fixedFont());
|
2020-05-10 21:35:08 -04:00
|
|
|
} else if (password.isEmpty() && !config()->get(Config::Security_PasswordEmptyPlaceholder).toBool()) {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->entryPasswordLabel->setText("");
|
2017-08-17 15:02:21 -04:00
|
|
|
} else {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->entryPasswordLabel->setText(QString("\u25cf").repeated(6));
|
2018-11-16 10:00:59 -05:00
|
|
|
}
|
2021-01-31 14:08:03 -05:00
|
|
|
m_ui->togglePasswordButton->setIcon(icons()->onOffIcon("password-show", state));
|
2018-11-16 10:00:59 -05:00
|
|
|
}
|
|
|
|
|
2019-06-28 21:48:25 -04:00
|
|
|
void EntryPreviewWidget::setEntryNotesVisible(bool state)
|
2019-06-23 10:02:02 -04:00
|
|
|
{
|
2019-11-09 12:16:05 -05:00
|
|
|
setNotesVisible(m_ui->entryNotesTextEdit, m_currentEntry->notes(), state);
|
2021-01-31 14:08:03 -05:00
|
|
|
m_ui->toggleEntryNotesButton->setIcon(icons()->onOffIcon("password-show", state));
|
2019-06-28 21:48:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void EntryPreviewWidget::setGroupNotesVisible(bool state)
|
|
|
|
{
|
2019-11-09 12:16:05 -05:00
|
|
|
setNotesVisible(m_ui->groupNotesTextEdit, m_currentGroup->notes(), state);
|
2021-01-31 14:08:03 -05:00
|
|
|
m_ui->toggleGroupNotesButton->setIcon(icons()->onOffIcon("password-show", state));
|
2019-06-28 21:48:25 -04:00
|
|
|
}
|
2019-06-23 10:02:02 -04:00
|
|
|
|
2019-11-09 12:16:05 -05:00
|
|
|
void EntryPreviewWidget::setNotesVisible(QTextEdit* notesWidget, const QString& notes, bool state)
|
2019-06-28 21:48:25 -04:00
|
|
|
{
|
2019-06-23 10:02:02 -04:00
|
|
|
if (state) {
|
2019-11-09 12:16:05 -05:00
|
|
|
notesWidget->setPlainText(notes);
|
|
|
|
notesWidget->moveCursor(QTextCursor::Start);
|
|
|
|
notesWidget->ensureCursorVisible();
|
2019-06-23 10:02:02 -04:00
|
|
|
} else {
|
2019-11-09 12:16:05 -05:00
|
|
|
if (!notes.isEmpty()) {
|
|
|
|
notesWidget->setPlainText(QString("\u25cf").repeated(6));
|
2019-06-23 10:02:02 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
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()));
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->entryUsernameLabel->setCursorPosition(0);
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::Security_HidePasswordPreviewPanel).toBool()) {
|
2018-11-16 10:00:59 -05:00
|
|
|
// Hide password
|
|
|
|
setPasswordVisible(false);
|
|
|
|
// Show the password toggle button if there are dots in the label
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->togglePasswordButton->setVisible(!m_ui->entryPasswordLabel->text().isEmpty());
|
2018-11-16 10:00:59 -05:00
|
|
|
m_ui->togglePasswordButton->setChecked(false);
|
2017-08-17 15:02:21 -04:00
|
|
|
} else {
|
2018-11-16 10:00:59 -05:00
|
|
|
// Show password
|
|
|
|
setPasswordVisible(true);
|
|
|
|
m_ui->togglePasswordButton->setVisible(false);
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::Security_HideNotes).toBool()) {
|
2019-06-28 21:48:25 -04:00
|
|
|
setEntryNotesVisible(false);
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->toggleEntryNotesButton->setVisible(!m_ui->entryNotesTextEdit->toPlainText().isEmpty());
|
2019-06-28 21:48:25 -04:00
|
|
|
m_ui->toggleEntryNotesButton->setChecked(false);
|
2019-06-23 10:02:02 -04:00
|
|
|
} else {
|
2019-06-28 21:48:25 -04:00
|
|
|
setEntryNotesVisible(true);
|
|
|
|
m_ui->toggleEntryNotesButton->setVisible(false);
|
2019-06-23 10:02:02 -04:00
|
|
|
}
|
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::GUI_MonospaceNotes).toBool()) {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->entryNotesTextEdit->setFont(Font::fixedFont());
|
2019-06-28 22:10:47 -04:00
|
|
|
} else {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->entryNotesTextEdit->setFont(Font::defaultFont());
|
2019-06-28 22:10:47 -04:00
|
|
|
}
|
|
|
|
|
2018-07-14 17:08:04 -04:00
|
|
|
m_ui->entryUrlLabel->setRawText(m_currentEntry->displayUrl());
|
2019-05-18 10:55:57 -04:00
|
|
|
const QString url = m_currentEntry->url();
|
2017-10-25 00:01:29 -04:00
|
|
|
if (!url.isEmpty()) {
|
|
|
|
// URL is well formed and can be opened in a browser
|
2019-06-19 20:48:35 -04:00
|
|
|
m_ui->entryUrlLabel->setUrl(m_currentEntry->resolveMultiplePlaceholders(url));
|
2018-07-14 17:08:04 -04:00
|
|
|
m_ui->entryUrlLabel->setCursor(Qt::PointingHandCursor);
|
2019-05-18 10:55:57 -04:00
|
|
|
m_ui->entryUrlLabel->setOpenExternalLinks(false);
|
2017-08-17 15:02:21 -04:00
|
|
|
} else {
|
2017-12-29 03:46:16 -05:00
|
|
|
m_ui->entryUrlLabel->setUrl({});
|
2018-07-14 17:08:04 -04:00
|
|
|
m_ui->entryUrlLabel->setCursor(Qt::ArrowCursor);
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
|
|
|
|
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 =
|
2019-01-20 09:50:20 -05:00
|
|
|
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
|
|
|
|
2019-06-23 10:02:02 -04:00
|
|
|
void EntryPreviewWidget::updateEntryAdvancedTab()
|
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();
|
2019-06-23 10:02:02 -04:00
|
|
|
const bool hasAttributes = !customAttributes.isEmpty();
|
|
|
|
const bool hasAttachments = !m_currentEntry->attachments()->isEmpty();
|
|
|
|
|
|
|
|
setTabEnabled(m_ui->entryTabWidget, m_ui->entryAdvancedTab, hasAttributes || hasAttachments);
|
|
|
|
if (hasAttributes) {
|
2020-12-19 10:49:53 -05:00
|
|
|
QString attributesText("<table>");
|
2017-09-27 13:18:13 -04:00
|
|
|
for (const QString& key : customAttributes) {
|
2020-12-19 10:49:53 -05:00
|
|
|
QString value;
|
2017-09-27 13:18:13 -04:00
|
|
|
if (m_currentEntry->attributes()->isProtected(key)) {
|
2017-10-24 08:08:22 -04:00
|
|
|
value = "<i>" + tr("[PROTECTED]") + "</i>";
|
2020-12-19 10:49:53 -05:00
|
|
|
} else {
|
|
|
|
value = m_currentEntry->attributes()->value(key).toHtmlEscaped();
|
|
|
|
value.replace('\n', QLatin1String("<br/>"));
|
2017-09-27 13:18:13 -04:00
|
|
|
}
|
2020-12-19 10:49:53 -05:00
|
|
|
attributesText.append(tr("<tr><td><b>%1</b>:</td><td>%2</td></tr>", "attributes line").arg(key, value));
|
2017-09-27 13:18:13 -04:00
|
|
|
}
|
2020-12-19 10:49:53 -05:00
|
|
|
attributesText.append("</table>");
|
2017-12-25 08:53:22 -05:00
|
|
|
m_ui->entryAttributesEdit->setText(attributesText);
|
2017-09-27 13:18:13 -04:00
|
|
|
}
|
2017-09-29 14:37:37 -04:00
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
m_ui->entryAttachmentsWidget->setEntryAttachments(m_currentEntry->attachments());
|
|
|
|
}
|
2017-12-13 15:22:31 -05:00
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::updateEntryAutotypeTab()
|
2017-12-25 08:53:22 -05:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_currentEntry);
|
2020-11-01 11:24:47 -05:00
|
|
|
|
|
|
|
m_ui->entrySequenceLabel->setText(m_currentEntry->effectiveAutoTypeSequence());
|
2017-12-25 08:53:22 -05:00
|
|
|
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 =
|
2019-01-20 09:50:20 -05:00
|
|
|
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-09-29 14:37:37 -04:00
|
|
|
}
|
2017-09-29 16:02:07 -04:00
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
m_ui->entryAutotypeTree->addTopLevelItems(items);
|
2020-11-01 11:24:47 -05:00
|
|
|
setTabEnabled(m_ui->entryTabWidget, m_ui->entryAutotypeTab, m_currentEntry->autoTypeEnabled());
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::updateGroupHeaderLine()
|
2017-08-17 15:02:21 -04:00
|
|
|
{
|
2017-12-25 08:53:22 -05:00
|
|
|
Q_ASSERT(m_currentGroup);
|
2018-02-24 20:00:31 -05:00
|
|
|
m_ui->groupTitleLabel->setRawText(hierarchy(m_currentGroup, {}));
|
2020-05-29 10:00:07 -04:00
|
|
|
m_ui->groupIcon->setPixmap(m_currentGroup->iconPixmap(IconSize::Large));
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
2017-09-29 14:37:37 -04:00
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
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-08-17 15:02:21 -04:00
|
|
|
|
2017-12-25 08:53:22 -05:00
|
|
|
const QString autotypeText = m_currentGroup->resolveAutoTypeEnabled() ? tr("Enabled") : tr("Disabled");
|
|
|
|
m_ui->groupAutotypeLabel->setText(autotypeText);
|
2017-08-17 15:02:21 -04:00
|
|
|
|
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 =
|
2019-01-20 09:50:20 -05:00
|
|
|
groupTime.expires() ? groupTime.expiryTime().toString(Qt::DefaultLocaleShortDate) : tr("Never");
|
2017-12-25 08:53:22 -05:00
|
|
|
m_ui->groupExpirationLabel->setText(expiresText);
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::Security_HideNotes).toBool()) {
|
2019-06-28 21:48:25 -04:00
|
|
|
setGroupNotesVisible(false);
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->toggleGroupNotesButton->setVisible(!m_ui->groupNotesTextEdit->toPlainText().isEmpty());
|
2019-06-28 21:48:25 -04:00
|
|
|
m_ui->toggleGroupNotesButton->setChecked(false);
|
|
|
|
} else {
|
|
|
|
setGroupNotesVisible(true);
|
|
|
|
m_ui->toggleGroupNotesButton->setVisible(false);
|
|
|
|
}
|
2019-06-28 22:10:47 -04:00
|
|
|
|
2020-04-25 19:31:38 -04:00
|
|
|
if (config()->get(Config::GUI_MonospaceNotes).toBool()) {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->groupNotesTextEdit->setFont(Font::fixedFont());
|
2019-06-28 22:10:47 -04:00
|
|
|
} else {
|
2019-11-09 12:16:05 -05:00
|
|
|
m_ui->groupNotesTextEdit->setFont(Font::defaultFont());
|
2019-06-28 22:10:47 -04:00
|
|
|
}
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
2017-08-17 15:02:21 -04:00
|
|
|
|
2019-01-03 02:46:32 -05:00
|
|
|
#if defined(WITH_XC_KEESHARE)
|
2018-12-18 22:28:56 -05:00
|
|
|
void EntryPreviewWidget::updateGroupSharingTab()
|
2018-10-01 10:26:24 -04:00
|
|
|
{
|
|
|
|
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
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::updateTotpLabel()
|
2017-12-25 08:53:22 -05:00
|
|
|
{
|
2018-09-05 16:20:57 -04: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);
|
2017-08-17 15:02:21 -04:00
|
|
|
} else {
|
2017-12-25 08:53:22 -05:00
|
|
|
m_ui->entryTotpLabel->clear();
|
2018-09-05 16:20:57 -04:00
|
|
|
m_totpTimer.stop();
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
2017-12-25 08:53:22 -05:00
|
|
|
}
|
2017-09-29 16:02:07 -04:00
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::updateTabIndexes()
|
2017-12-25 08:53:22 -05:00
|
|
|
{
|
|
|
|
m_selectedTabEntry = m_ui->entryTabWidget->currentIndex();
|
|
|
|
m_selectedTabGroup = m_ui->groupTabWidget->currentIndex();
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
|
|
|
|
2019-05-18 10:55:57 -04:00
|
|
|
void EntryPreviewWidget::openEntryUrl()
|
|
|
|
{
|
|
|
|
if (m_currentEntry) {
|
|
|
|
emit entryUrlActivated(m_currentEntry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-22 11:47:43 -04:00
|
|
|
void EntryPreviewWidget::removeTab(QTabWidget* tabWidget, QWidget* widget)
|
|
|
|
{
|
|
|
|
const int tabIndex = tabWidget->indexOf(widget);
|
|
|
|
Q_ASSERT(tabIndex != -1);
|
|
|
|
tabWidget->removeTab(tabIndex);
|
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
void EntryPreviewWidget::setTabEnabled(QTabWidget* tabWidget, QWidget* widget, bool enabled)
|
2017-08-17 15:02:21 -04:00
|
|
|
{
|
2017-12-25 08:53:22 -05:00
|
|
|
const int tabIndex = tabWidget->indexOf(widget);
|
|
|
|
Q_ASSERT(tabIndex != -1);
|
|
|
|
tabWidget->setTabEnabled(tabIndex, enabled);
|
2017-08-17 15:02:21 -04:00
|
|
|
}
|
|
|
|
|
2018-10-08 21:11:44 -04:00
|
|
|
QString EntryPreviewWidget::hierarchy(const Group* group, const QString& title)
|
2017-11-11 17:25:20 -05:00
|
|
|
{
|
2020-05-26 11:58:52 -04:00
|
|
|
QString groupList = QString("%1").arg(group->hierarchy().join(" / "));
|
|
|
|
return title.isEmpty() ? groupList : QStringLiteral("%1 / %2").arg(groupList, title);
|
2017-10-25 23:29:01 -04:00
|
|
|
}
|