Implement Auto-Type {PICKCHARS}

* Closes #725

Support Auto-Type {PICKCHARS} placeholder. Open a dialog that lets you pick characters of an entry's password by their position. Supports typing {TAB} in between characters to move between fields (if necessary). Also supports using arrow keys to quickly navigate around the choice grid.
This commit is contained in:
Jonathan White 2021-02-13 22:08:29 -05:00
parent 027ff9f2bf
commit 813ab47e29
No known key found for this signature in database
GPG Key ID: 440FC65F2E0C6E01
6 changed files with 335 additions and 0 deletions

View File

@ -47,6 +47,7 @@ image::autotype_entry_sequences.png[]
|{DELAY X} |Delay typing start by X milliseconds
|{CLEARFIELD} |Clear the input field before typing
|{TOTP} |Insert calculated TOTP value (if configured)
|{PICKCHARS} |Pick specific password characters from a dialog
|{<ACTION> X} |Repeat <ACTION> X times (e.g., {SPACE 5} inserts five spaces)
|===
+

View File

@ -273,6 +273,7 @@ set(autotype_SOURCES
autotype/AutoTypeMatchModel.cpp
autotype/AutoTypeMatchView.cpp
autotype/AutoTypeSelectDialog.cpp
autotype/PickcharsDialog.cpp
autotype/ShortcutWidget.cpp
autotype/WindowSelectComboBox.cpp)

View File

@ -27,6 +27,7 @@
#include "autotype/AutoTypePlatformPlugin.h"
#include "autotype/AutoTypeSelectDialog.h"
#include "autotype/PickcharsDialog.h"
#include "core/Config.h"
#include "core/Database.h"
#include "core/Entry.h"
@ -564,6 +565,27 @@ AutoType::parseActions(const QString& entrySequence, const Entry* entry, QString
for (const auto& ch : totp) {
actions << QSharedPointer<AutoTypeKey>::create(ch);
}
} else if (placeholder == "pickchars") {
if (error) {
// Ignore this if we are syntax checking
continue;
}
// Show pickchars dialog for entry's password
auto password = entry->resolvePlaceholder(entry->password());
if (!password.isEmpty()) {
PickcharsDialog pickcharsDialog(password);
if (pickcharsDialog.exec() == QDialog::Accepted && !pickcharsDialog.selectedChars().isEmpty()) {
auto chars = pickcharsDialog.selectedChars();
auto iter = chars.begin();
while (iter != chars.end()) {
actions << QSharedPointer<AutoTypeKey>::create(*iter);
++iter;
if (pickcharsDialog.pressTab() && iter != chars.end()) {
actions << QSharedPointer<AutoTypeKey>::create(g_placeholderToKey["tab"]);
}
}
}
}
} else if (placeholder == "beep" || placeholder.startsWith("vkey")
|| placeholder.startsWith("appactivate")) {
// Ignore these commands

View File

@ -0,0 +1,172 @@
/*
* Copyright (C) 2021 Team KeePassXC <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 "PickcharsDialog.h"
#include "ui_PickcharsDialog.h"
#include "core/Entry.h"
#include "gui/Icons.h"
#include <QPushButton>
#include <QShortcut>
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
#include <QScreen>
#else
#include <QDesktopWidget>
#endif
PickcharsDialog::PickcharsDialog(const QString& string, QWidget* parent)
: QDialog(parent)
, m_ui(new Ui::PickcharsDialog())
{
if (string.isEmpty()) {
reject();
}
// Places the window on the active (virtual) desktop instead of where the main window is.
setAttribute(Qt::WA_X11BypassTransientForHint);
setWindowFlags((windowFlags() | Qt::WindowStaysOnTopHint | Qt::MSWindowsFixedSizeDialogHint)
& ~Qt::WindowContextHelpButtonHint);
setWindowIcon(icons()->applicationIcon());
m_ui->setupUi(this);
// Increase max columns with longer passwords for better display
int width = 10;
if (string.length() >= 100) {
width = 20;
} else if (string.length() >= 60) {
width = 15;
}
int count = 0;
for (const auto& ch : string) {
auto btn = new QPushButton(QString::number(count + 1));
btn->setProperty("char", ch);
btn->setProperty("count", count);
connect(btn, &QPushButton::clicked, this, &PickcharsDialog::charSelected);
m_ui->charsGrid->addWidget(btn, count / width, count % width);
m_lastSelected = count;
++count;
}
// Prevent stretched buttons
if (m_ui->charsGrid->rowCount() == 1 && m_ui->charsGrid->columnCount() < 5) {
m_ui->charsGrid->addItem(new QSpacerItem(5, 5, QSizePolicy::MinimumExpanding), count / width, count % width);
}
m_ui->charsGrid->itemAtPosition(0, 0)->widget()->setFocus();
connect(m_ui->buttonBox, &QDialogButtonBox::accepted, this, &QDialog::accept);
connect(m_ui->buttonBox, &QDialogButtonBox::rejected, this, &QDialog::reject);
// Navigate grid layout using up/down/left/right motion
new QShortcut(Qt::Key_Up, this, SLOT(upPressed()));
new QShortcut(Qt::Key_Down, this, SLOT(downPressed()));
// Remove last selected character
auto shortcut = new QShortcut(Qt::Key_Backspace, this);
connect(shortcut, &QShortcut::activated, this, [this] {
auto text = m_ui->selectedChars->text();
m_ui->selectedChars->setText(text.left(text.size() - 1));
});
// Submit the form
shortcut = new QShortcut(Qt::CTRL + Qt::Key_S, this);
connect(shortcut, &QShortcut::activated, this, [this] { accept(); });
}
void PickcharsDialog::upPressed()
{
auto focus = focusWidget();
if (!focus) {
return;
}
auto count = focus->property("count");
if (count.isValid()) {
// Lower bound not checked by QGridLayout::itemAt https://bugreports.qt.io/browse/QTBUG-91261
auto upCount = count.toInt() - m_ui->charsGrid->columnCount();
if (upCount >= 0) {
m_ui->charsGrid->itemAt(upCount)->widget()->setFocus();
}
} else if (focus == m_ui->selectedChars) {
// Move back to the last selected button
auto item = m_ui->charsGrid->itemAt(m_lastSelected);
if (item) {
item->widget()->setFocus();
}
} else if (focus == m_ui->pressTab) {
m_ui->selectedChars->setFocus();
}
}
void PickcharsDialog::downPressed()
{
auto focus = focusWidget();
if (!focus) {
return;
}
auto count = focus->property("count");
if (count.isValid()) {
auto item = m_ui->charsGrid->itemAt(count.toInt() + m_ui->charsGrid->columnCount());
if (item) {
item->widget()->setFocus();
} else {
// Store the currently selected button and move to the line edit
m_lastSelected = count.toInt();
m_ui->selectedChars->setFocus();
}
} else if (focus == m_ui->selectedChars) {
m_ui->pressTab->setFocus();
}
}
QString PickcharsDialog::selectedChars()
{
return m_ui->selectedChars->text();
}
bool PickcharsDialog::pressTab()
{
return m_ui->pressTab->isChecked();
}
void PickcharsDialog::charSelected()
{
auto btn = qobject_cast<QPushButton*>(sender());
if (!btn) {
return;
}
m_ui->selectedChars->setText(m_ui->selectedChars->text() + btn->property("char").toChar());
}
void PickcharsDialog::showEvent(QShowEvent* event)
{
QDialog::showEvent(event);
// Center on active screen
#if QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)
auto screen = QApplication::screenAt(QCursor::pos());
if (!screen) {
// screenAt can return a nullptr, default to the primary screen
screen = QApplication::primaryScreen();
}
QRect screenGeometry = screen->availableGeometry();
#else
QRect screenGeometry = QApplication::desktop()->availableGeometry(QCursor::pos());
#endif
move(screenGeometry.center().x() - (size().width() / 2), screenGeometry.center().y() - (size().height() / 2));
}

View File

@ -0,0 +1,52 @@
/*
* Copyright (C) 2021 Team KeePassXC <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_PICKCHARSDIALOG_H
#define KEEPASSXC_PICKCHARSDIALOG_H
#include <QDialog>
#include <QPointer>
#include <QString>
namespace Ui
{
class PickcharsDialog;
}
class PickcharsDialog : public QDialog
{
Q_OBJECT
public:
explicit PickcharsDialog(const QString& string, QWidget* parent = nullptr);
QString selectedChars();
bool pressTab();
protected:
void showEvent(QShowEvent*) override;
private slots:
void charSelected();
void upPressed();
void downPressed();
private:
QSharedPointer<Ui::PickcharsDialog> m_ui;
int m_lastSelected;
};
#endif // KEEPASSXC_PICKCHARSDIALOG_H

View File

@ -0,0 +1,87 @@
<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
<class>PickcharsDialog</class>
<widget class="QDialog" name="PickcharsDialog">
<property name="geometry">
<rect>
<x>0</x>
<y>0</y>
<width>418</width>
<height>188</height>
</rect>
</property>
<property name="windowTitle">
<string>KeePassXC - Pick Characters</string>
</property>
<layout class="QVBoxLayout" name="verticalLayout">
<property name="sizeConstraint">
<enum>QLayout::SetFixedSize</enum>
</property>
<item>
<widget class="QLabel" name="label">
<property name="sizePolicy">
<sizepolicy hsizetype="Preferred" vsizetype="Fixed">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="text">
<string>Select characters to type, navigate with arrow keys, Ctrl + S submits.</string>
</property>
</widget>
</item>
<item>
<layout class="QGridLayout" name="charsGrid"/>
</item>
<item>
<widget class="PasswordEdit" name="selectedChars">
<property name="sizePolicy">
<sizepolicy hsizetype="MinimumExpanding" vsizetype="Fixed">
<horstretch>0</horstretch>
<verstretch>0</verstretch>
</sizepolicy>
</property>
<property name="minimumSize">
<size>
<width>150</width>
<height>0</height>
</size>
</property>
</widget>
</item>
<item>
<layout class="QHBoxLayout" name="horizontalLayout">
<item>
<widget class="QCheckBox" name="pressTab">
<property name="text">
<string>Press &amp;Tab between characters</string>
</property>
<property name="checked">
<bool>true</bool>
</property>
</widget>
</item>
<item>
<widget class="QDialogButtonBox" name="buttonBox">
<property name="standardButtons">
<set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
</property>
</widget>
</item>
</layout>
</item>
</layout>
</widget>
<customwidgets>
<customwidget>
<class>PasswordEdit</class>
<extends>QLineEdit</extends>
<header>gui/PasswordEdit.h</header>
</customwidget>
</customwidgets>
<tabstops>
<tabstop>selectedChars</tabstop>
</tabstops>
<resources/>
<connections/>
</ui>