/* * Copyright (C) 2014 Felix Geyer * Copyright (C) 2020 KeePassXC Team * * 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 . */ #include "PasswordWidget.h" #include "ui_PasswordWidget.h" #include "core/Config.h" #include "core/PasswordHealth.h" #include "gui/Font.h" #include "gui/Icons.h" #include "gui/PasswordGeneratorWidget.h" #include "gui/osutils/OSUtils.h" #include "gui/styles/StateColorPalette.h" #include #include #include #include PasswordWidget::PasswordWidget(QWidget* parent) : QWidget(parent) , m_ui(new Ui::PasswordWidget()) { m_ui->setupUi(this); setFocusProxy(m_ui->passwordEdit); m_ui->passwordEdit->installEventFilter(this); const QIcon errorIcon = icons()->icon("dialog-error"); m_errorAction = m_ui->passwordEdit->addAction(errorIcon, QLineEdit::TrailingPosition); m_errorAction->setVisible(false); m_errorAction->setToolTip(tr("Passwords do not match")); const QIcon correctIcon = icons()->icon("dialog-ok"); m_correctAction = m_ui->passwordEdit->addAction(correctIcon, QLineEdit::TrailingPosition); m_correctAction->setVisible(false); m_correctAction->setToolTip(tr("Passwords match so far")); setEchoMode(QLineEdit::Password); // use a monospace font for the password field QFont passwordFont = Font::fixedFont(); passwordFont.setLetterSpacing(QFont::PercentageSpacing, 110); m_ui->passwordEdit->setFont(passwordFont); // Prevent conflicts with global Mac shortcuts (force Control on all platforms) #ifdef Q_OS_MAC constexpr auto modifier = Qt::MetaModifier; #else constexpr auto modifier = Qt::ControlModifier; #endif m_toggleVisibleAction = new QAction( icons()->onOffIcon("password-show", false), tr("Toggle Password (%1)").arg(QKeySequence(modifier + Qt::Key_H).toString(QKeySequence::NativeText)), this); m_toggleVisibleAction->setCheckable(true); m_toggleVisibleAction->setShortcut(modifier + Qt::Key_H); m_toggleVisibleAction->setShortcutContext(Qt::WidgetShortcut); m_ui->passwordEdit->addAction(m_toggleVisibleAction, QLineEdit::TrailingPosition); connect(m_toggleVisibleAction, &QAction::triggered, this, &PasswordWidget::setShowPassword); m_passwordGeneratorAction = new QAction( icons()->icon("password-generator"), tr("Generate Password (%1)").arg(QKeySequence(modifier + Qt::Key_G).toString(QKeySequence::NativeText)), this); m_passwordGeneratorAction->setShortcut(modifier + Qt::Key_G); m_passwordGeneratorAction->setShortcutContext(Qt::WidgetShortcut); m_ui->passwordEdit->addAction(m_passwordGeneratorAction, QLineEdit::TrailingPosition); m_passwordGeneratorAction->setVisible(false); m_capslockAction = new QAction(icons()->icon("dialog-warning", true, StateColorPalette().color(StateColorPalette::Error)), tr("Warning: Caps Lock enabled!"), this); m_ui->passwordEdit->addAction(m_capslockAction, QLineEdit::LeadingPosition); m_capslockAction->setVisible(false); // Reset the password strength bar, hidden by default updatePasswordStrength(""); m_ui->qualityProgressBar->setVisible(false); connect(m_ui->passwordEdit, &QLineEdit::textChanged, this, [this](const QString& pwd) { updatePasswordStrength(pwd); emit textChanged(pwd); }); } PasswordWidget::~PasswordWidget() { } void PasswordWidget::setQualityVisible(bool state) { m_ui->qualityProgressBar->setVisible(state); } QString PasswordWidget::text() { return m_ui->passwordEdit->text(); } void PasswordWidget::setText(const QString& text) { m_ui->passwordEdit->setText(text); } void PasswordWidget::setEchoMode(QLineEdit::EchoMode mode) { m_ui->passwordEdit->setEchoMode(mode); } void PasswordWidget::clear() { m_ui->passwordEdit->clear(); } void PasswordWidget::setClearButtonEnabled(bool enabled) { m_ui->passwordEdit->setClearButtonEnabled(enabled); } void PasswordWidget::selectAll() { m_ui->passwordEdit->selectAll(); } void PasswordWidget::setReadOnly(bool state) { m_ui->passwordEdit->setReadOnly(state); } void PasswordWidget::setRepeatPartner(PasswordWidget* repeatPartner) { m_repeatPasswordWidget = repeatPartner; m_repeatPasswordWidget->setParentPasswordEdit(this); connect(m_ui->passwordEdit, SIGNAL(textChanged(QString)), m_repeatPasswordWidget, SLOT(updateRepeatStatus())); } void PasswordWidget::setParentPasswordEdit(PasswordWidget* parent) { m_parentPasswordWidget = parent; // Hide actions m_toggleVisibleAction->setVisible(false); m_passwordGeneratorAction->setVisible(false); connect(m_ui->passwordEdit, SIGNAL(textChanged(QString)), this, SLOT(updateRepeatStatus())); } void PasswordWidget::enablePasswordGenerator() { if (!m_passwordGeneratorAction->isVisible()) { m_passwordGeneratorAction->setVisible(true); connect(m_passwordGeneratorAction, &QAction::triggered, this, &PasswordWidget::popupPasswordGenerator); } } void PasswordWidget::setShowPassword(bool show) { setEchoMode(show ? QLineEdit::Normal : QLineEdit::Password); m_toggleVisibleAction->setIcon(icons()->onOffIcon("password-show", show)); m_toggleVisibleAction->setChecked(show); if (m_repeatPasswordWidget) { m_repeatPasswordWidget->setEchoMode(show ? QLineEdit::Normal : QLineEdit::Password); } } bool PasswordWidget::isPasswordVisible() const { return m_ui->passwordEdit->echoMode() == QLineEdit::Normal; } void PasswordWidget::popupPasswordGenerator() { auto generator = PasswordGeneratorWidget::popupGenerator(this); generator->setPasswordVisible(isPasswordVisible()); generator->setPasswordLength(text().length()); connect(generator, SIGNAL(appliedPassword(QString)), SLOT(setText(QString))); if (m_repeatPasswordWidget) { connect(generator, SIGNAL(appliedPassword(QString)), m_repeatPasswordWidget, SLOT(setText(QString))); } } void PasswordWidget::updateRepeatStatus() { static const auto stylesheetTemplate = QStringLiteral("QLineEdit { background: %1; }"); if (!m_parentPasswordWidget) { return; } const auto otherPassword = m_parentPasswordWidget->text(); const auto password = text(); if (otherPassword != password) { bool isCorrect = false; StateColorPalette statePalette; QColor color = statePalette.color(StateColorPalette::ColorRole::Error); if (!password.isEmpty() && otherPassword.startsWith(password)) { color = statePalette.color(StateColorPalette::ColorRole::Incomplete); isCorrect = true; } setStyleSheet(stylesheetTemplate.arg(color.name())); m_correctAction->setVisible(isCorrect); m_errorAction->setVisible(!isCorrect); } else { m_correctAction->setVisible(false); m_errorAction->setVisible(false); setStyleSheet(""); } } bool PasswordWidget::eventFilter(QObject* watched, QEvent* event) { if (watched == m_ui->passwordEdit) { auto type = event->type(); if (isVisible() && (type == QEvent::KeyPress || type == QEvent::KeyRelease || type == QEvent::FocusIn)) { checkCapslockState(); } if (type == QEvent::FocusIn || type == QEvent::FocusOut || type == QEvent::Hide) { osUtils->setUserInputProtection(type == QEvent::FocusIn); } } // Continue with normal operations return false; } void PasswordWidget::checkCapslockState() { if (m_parentPasswordWidget) { return; } bool newCapslockState = osUtils->isCapslockEnabled(); if (newCapslockState != m_capslockState) { m_capslockState = newCapslockState; m_capslockAction->setVisible(newCapslockState); // Force repaint to avoid rendering glitches of QLineEdit contents repaint(); if (newCapslockState) { QTimer::singleShot( 150, [this] { QToolTip::showText(mapToGlobal(rect().bottomLeft()), m_capslockAction->text()); }); } else if (QToolTip::isVisible()) { QToolTip::hideText(); } } } void PasswordWidget::updatePasswordStrength(const QString& password) { if (password.isEmpty()) { m_ui->qualityProgressBar->setValue(0); m_ui->qualityProgressBar->setToolTip((tr(""))); return; } PasswordHealth health(password); m_ui->qualityProgressBar->setValue(std::min(int(health.entropy()), m_ui->qualityProgressBar->maximum())); QString style = m_ui->qualityProgressBar->styleSheet(); QRegularExpression re("(QProgressBar::chunk\\s*\\{.*?background-color:)[^;]+;", QRegularExpression::CaseInsensitiveOption | QRegularExpression::DotMatchesEverythingOption); style.replace(re, "\\1 %1;"); StateColorPalette qualityPalette; switch (health.quality()) { case PasswordHealth::Quality::Bad: case PasswordHealth::Quality::Poor: m_ui->qualityProgressBar->setStyleSheet( style.arg(qualityPalette.color(StateColorPalette::HealthCritical).name())); m_ui->qualityProgressBar->setToolTip(tr("Quality: %1").arg(tr("Poor", "Password quality"))); break; case PasswordHealth::Quality::Weak: m_ui->qualityProgressBar->setStyleSheet(style.arg(qualityPalette.color(StateColorPalette::HealthBad).name())); m_ui->qualityProgressBar->setToolTip(tr("Quality: %1").arg(tr("Weak", "Password quality"))); break; case PasswordHealth::Quality::Good: m_ui->qualityProgressBar->setStyleSheet(style.arg(qualityPalette.color(StateColorPalette::HealthOk).name())); m_ui->qualityProgressBar->setToolTip(tr("Quality: %1").arg(tr("Good", "Password quality"))); break; case PasswordHealth::Quality::Excellent: m_ui->qualityProgressBar->setStyleSheet( style.arg(qualityPalette.color(StateColorPalette::HealthExcellent).name())); m_ui->qualityProgressBar->setToolTip(tr("Quality: %1").arg(tr("Excellent", "Password quality"))); break; } }