mirror of
https://github.com/citra-emu/citra.git
synced 2024-11-25 07:00:14 +00:00
citra_qt: improve touchscreen mapping UI
This commit is contained in:
parent
41facaece3
commit
09cba69b48
5
dist/qt_themes/qdarkstyle/style.qss
vendored
5
dist/qt_themes/qdarkstyle/style.qss
vendored
@ -1236,3 +1236,8 @@ QToolButton:disabled,
|
|||||||
QPlainTextEdit:disabled {
|
QPlainTextEdit:disabled {
|
||||||
background-color: #2b2e31;
|
background-color: #2b2e31;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* touchscreen mapping widget */
|
||||||
|
TouchScreenPreview {
|
||||||
|
qproperty-dotHighlightColor: #3daee9;
|
||||||
|
}
|
||||||
|
@ -72,6 +72,7 @@ add_executable(citra-qt
|
|||||||
configuration/configure_touch_from_button.cpp
|
configuration/configure_touch_from_button.cpp
|
||||||
configuration/configure_touch_from_button.h
|
configuration/configure_touch_from_button.h
|
||||||
configuration/configure_touch_from_button.ui
|
configuration/configure_touch_from_button.ui
|
||||||
|
configuration/configure_touch_widget.h
|
||||||
configuration/configure_ui.cpp
|
configuration/configure_ui.cpp
|
||||||
configuration/configure_ui.h
|
configuration/configure_ui.h
|
||||||
configuration/configure_ui.ui
|
configuration/configure_ui.ui
|
||||||
|
@ -5,10 +5,14 @@
|
|||||||
#include <QInputDialog>
|
#include <QInputDialog>
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
|
#include <QMouseEvent>
|
||||||
|
#include <QResizeEvent>
|
||||||
#include <QStandardItemModel>
|
#include <QStandardItemModel>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
#include "citra_qt/configuration/configure_touch_from_button.h"
|
#include "citra_qt/configuration/configure_touch_from_button.h"
|
||||||
|
#include "citra_qt/configuration/configure_touch_widget.h"
|
||||||
#include "common/param_package.h"
|
#include "common/param_package.h"
|
||||||
|
#include "core/3ds.h"
|
||||||
#include "input_common/main.h"
|
#include "input_common/main.h"
|
||||||
#include "ui_configure_touch_from_button.h"
|
#include "ui_configure_touch_from_button.h"
|
||||||
|
|
||||||
@ -64,15 +68,17 @@ static QString ButtonToText(const Common::ParamPackage& param) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ConfigureTouchFromButton::ConfigureTouchFromButton(
|
ConfigureTouchFromButton::ConfigureTouchFromButton(
|
||||||
QWidget* parent, std::vector<Settings::TouchFromButtonMap> touch_maps, int default_index)
|
QWidget* parent, const std::vector<Settings::TouchFromButtonMap>& touch_maps,
|
||||||
: QDialog(parent), touch_maps(touch_maps), selected_index(default_index),
|
const int default_index)
|
||||||
ui(std::make_unique<Ui::ConfigureTouchFromButton>()),
|
: QDialog(parent), ui(std::make_unique<Ui::ConfigureTouchFromButton>()), touch_maps(touch_maps),
|
||||||
timeout_timer(std::make_unique<QTimer>()), poll_timer(std::make_unique<QTimer>()) {
|
selected_index(default_index), timeout_timer(std::make_unique<QTimer>()),
|
||||||
|
poll_timer(std::make_unique<QTimer>()) {
|
||||||
|
|
||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
binding_list_model = std::make_unique<QStandardItemModel>(0, 3, this);
|
binding_list_model = std::make_unique<QStandardItemModel>(0, 3, this);
|
||||||
binding_list_model->setHorizontalHeaderLabels({tr("Button"), tr("X"), tr("Y")});
|
binding_list_model->setHorizontalHeaderLabels({tr("Button"), tr("X"), tr("Y")});
|
||||||
ui->binding_list->setModel(binding_list_model.get());
|
ui->binding_list->setModel(binding_list_model.get());
|
||||||
|
ui->bottom_screen->SetCoordLabel(ui->coord_label);
|
||||||
|
|
||||||
SetConfiguration();
|
SetConfiguration();
|
||||||
UpdateUiDisplay();
|
UpdateUiDisplay();
|
||||||
@ -85,7 +91,8 @@ void ConfigureTouchFromButton::showEvent(QShowEvent* ev) {
|
|||||||
QWidget::showEvent(ev);
|
QWidget::showEvent(ev);
|
||||||
|
|
||||||
// width values are not valid in the constructor
|
// width values are not valid in the constructor
|
||||||
const int w = ui->binding_list->contentsRect().width() / binding_list_model->columnCount();
|
const int w =
|
||||||
|
ui->binding_list->viewport()->contentsRect().width() / binding_list_model->columnCount();
|
||||||
if (w > 0) {
|
if (w > 0) {
|
||||||
ui->binding_list->setColumnWidth(0, w);
|
ui->binding_list->setColumnWidth(0, w);
|
||||||
ui->binding_list->setColumnWidth(1, w);
|
ui->binding_list->setColumnWidth(1, w);
|
||||||
@ -102,20 +109,11 @@ void ConfigureTouchFromButton::SetConfiguration() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ConfigureTouchFromButton::UpdateUiDisplay() {
|
void ConfigureTouchFromButton::UpdateUiDisplay() {
|
||||||
const bool have_maps = !touch_maps.empty();
|
|
||||||
|
|
||||||
ui->button_delete->setEnabled(touch_maps.size() > 1);
|
ui->button_delete->setEnabled(touch_maps.size() > 1);
|
||||||
ui->button_rename->setEnabled(have_maps);
|
|
||||||
ui->binding_list->setEnabled(have_maps);
|
|
||||||
ui->button_add_bind->setEnabled(have_maps);
|
|
||||||
ui->button_delete_bind->setEnabled(false);
|
ui->button_delete_bind->setEnabled(false);
|
||||||
|
|
||||||
binding_list_model->removeRows(0, binding_list_model->rowCount());
|
binding_list_model->removeRows(0, binding_list_model->rowCount());
|
||||||
|
|
||||||
if (!have_maps) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto& button_str : touch_maps[selected_index].buttons) {
|
for (const auto& button_str : touch_maps[selected_index].buttons) {
|
||||||
Common::ParamPackage package{button_str};
|
Common::ParamPackage package{button_str};
|
||||||
QStandardItem* button = new QStandardItem(ButtonToText(package));
|
QStandardItem* button = new QStandardItem(ButtonToText(package));
|
||||||
@ -124,6 +122,9 @@ void ConfigureTouchFromButton::UpdateUiDisplay() {
|
|||||||
QStandardItem* xcoord = new QStandardItem(QString::number(package.Get("x", 0)));
|
QStandardItem* xcoord = new QStandardItem(QString::number(package.Get("x", 0)));
|
||||||
QStandardItem* ycoord = new QStandardItem(QString::number(package.Get("y", 0)));
|
QStandardItem* ycoord = new QStandardItem(QString::number(package.Get("y", 0)));
|
||||||
binding_list_model->appendRow({button, xcoord, ycoord});
|
binding_list_model->appendRow({button, xcoord, ycoord});
|
||||||
|
|
||||||
|
int dot = ui->bottom_screen->AddDot(package.Get("x", 0), package.Get("y", 0));
|
||||||
|
button->setData(dot, data_role_dot);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -138,16 +139,22 @@ void ConfigureTouchFromButton::ConnectEvents() {
|
|||||||
&ConfigureTouchFromButton::DeleteMapping);
|
&ConfigureTouchFromButton::DeleteMapping);
|
||||||
connect(ui->button_rename, &QPushButton::clicked, this,
|
connect(ui->button_rename, &QPushButton::clicked, this,
|
||||||
&ConfigureTouchFromButton::RenameMapping);
|
&ConfigureTouchFromButton::RenameMapping);
|
||||||
connect(ui->button_add_bind, &QPushButton::clicked, this,
|
|
||||||
&ConfigureTouchFromButton::NewBinding);
|
|
||||||
connect(ui->button_delete_bind, &QPushButton::clicked, this,
|
connect(ui->button_delete_bind, &QPushButton::clicked, this,
|
||||||
&ConfigureTouchFromButton::DeleteBinding);
|
&ConfigureTouchFromButton::DeleteBinding);
|
||||||
connect(ui->binding_list, &QTreeView::doubleClicked, this,
|
connect(ui->binding_list, &QTreeView::doubleClicked, this,
|
||||||
&ConfigureTouchFromButton::EditBinding);
|
&ConfigureTouchFromButton::EditBinding);
|
||||||
connect(ui->binding_list->selectionModel(), &QItemSelectionModel::selectionChanged, this,
|
connect(ui->binding_list->selectionModel(), &QItemSelectionModel::selectionChanged, this,
|
||||||
[this](const QItemSelection& selected, const QItemSelection& deselected) {
|
&ConfigureTouchFromButton::OnBindingSelection);
|
||||||
ui->button_delete_bind->setEnabled(!selected.indexes().isEmpty());
|
connect(binding_list_model.get(), &QStandardItemModel::itemChanged, this,
|
||||||
});
|
&ConfigureTouchFromButton::OnBindingChanged);
|
||||||
|
connect(ui->binding_list->model(), &QStandardItemModel::rowsAboutToBeRemoved, this,
|
||||||
|
&ConfigureTouchFromButton::OnBindingDeleted);
|
||||||
|
connect(ui->bottom_screen, &TouchScreenPreview::DotAdded, this,
|
||||||
|
&ConfigureTouchFromButton::NewBinding);
|
||||||
|
connect(ui->bottom_screen, &TouchScreenPreview::DotSelected, this,
|
||||||
|
&ConfigureTouchFromButton::SetActiveBinding);
|
||||||
|
connect(ui->bottom_screen, &TouchScreenPreview::DotMoved, this,
|
||||||
|
&ConfigureTouchFromButton::SetCoordinates);
|
||||||
connect(ui->buttonBox, &QDialogButtonBox::accepted, this,
|
connect(ui->buttonBox, &QDialogButtonBox::accepted, this,
|
||||||
&ConfigureTouchFromButton::ApplyConfiguration);
|
&ConfigureTouchFromButton::ApplyConfiguration);
|
||||||
|
|
||||||
@ -169,10 +176,10 @@ void ConfigureTouchFromButton::SaveCurrentMapping() {
|
|||||||
auto& map = touch_maps[selected_index];
|
auto& map = touch_maps[selected_index];
|
||||||
map.buttons.clear();
|
map.buttons.clear();
|
||||||
for (int i = 0, rc = binding_list_model->rowCount(); i < rc; ++i) {
|
for (int i = 0, rc = binding_list_model->rowCount(); i < rc; ++i) {
|
||||||
auto bind_str = binding_list_model->index(i, 0)
|
const auto bind_str = binding_list_model->index(i, 0)
|
||||||
.data(Qt::ItemDataRole::UserRole + 1)
|
.data(Qt::ItemDataRole::UserRole + 1)
|
||||||
.toString()
|
.toString()
|
||||||
.toStdString();
|
.toStdString();
|
||||||
if (bind_str.empty()) {
|
if (bind_str.empty()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -197,7 +204,7 @@ void ConfigureTouchFromButton::NewMapping() {
|
|||||||
SaveCurrentMapping();
|
SaveCurrentMapping();
|
||||||
}
|
}
|
||||||
touch_maps.emplace_back(Settings::TouchFromButtonMap{name.toStdString(), {}});
|
touch_maps.emplace_back(Settings::TouchFromButtonMap{name.toStdString(), {}});
|
||||||
selected_index = touch_maps.size() - 1;
|
selected_index = static_cast<int>(touch_maps.size()) - 1;
|
||||||
|
|
||||||
ui->mapping->addItem(name);
|
ui->mapping->addItem(name);
|
||||||
ui->mapping->setCurrentIndex(selected_index);
|
ui->mapping->setCurrentIndex(selected_index);
|
||||||
@ -226,7 +233,7 @@ void ConfigureTouchFromButton::RenameMapping() {
|
|||||||
touch_maps[selected_index].name = new_name.toStdString();
|
touch_maps[selected_index].name = new_name.toStdString();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ConfigureTouchFromButton::GetButtonInput(int row_index, bool is_new) {
|
void ConfigureTouchFromButton::GetButtonInput(const int row_index, const bool is_new) {
|
||||||
binding_list_model->item(row_index, 0)->setText(tr("[press key]"));
|
binding_list_model->item(row_index, 0)->setText(tr("[press key]"));
|
||||||
|
|
||||||
input_setter = [this, row_index, is_new](const Common::ParamPackage& params,
|
input_setter = [this, row_index, is_new](const Common::ParamPackage& params,
|
||||||
@ -253,15 +260,21 @@ void ConfigureTouchFromButton::GetButtonInput(int row_index, bool is_new) {
|
|||||||
|
|
||||||
grabKeyboard();
|
grabKeyboard();
|
||||||
grabMouse();
|
grabMouse();
|
||||||
|
qApp->setOverrideCursor(QCursor(Qt::CursorShape::ArrowCursor));
|
||||||
timeout_timer->start(5000); // Cancel after 5 seconds
|
timeout_timer->start(5000); // Cancel after 5 seconds
|
||||||
poll_timer->start(200); // Check for new inputs every 200ms
|
poll_timer->start(200); // Check for new inputs every 200ms
|
||||||
}
|
}
|
||||||
|
|
||||||
void ConfigureTouchFromButton::NewBinding() {
|
void ConfigureTouchFromButton::NewBinding(const QPoint& pos) {
|
||||||
QStandardItem* button = new QStandardItem();
|
QStandardItem* button = new QStandardItem();
|
||||||
button->setEditable(false);
|
button->setEditable(false);
|
||||||
binding_list_model->appendRow(
|
QStandardItem* xcoord = new QStandardItem(QString::number(pos.x()));
|
||||||
{button, new QStandardItem(QStringLiteral("0")), new QStandardItem(QStringLiteral("0"))});
|
QStandardItem* ycoord = new QStandardItem(QString::number(pos.y()));
|
||||||
|
|
||||||
|
int dot_id = ui->bottom_screen->AddDot(pos.x(), pos.y());
|
||||||
|
button->setData(dot_id, data_role_dot);
|
||||||
|
|
||||||
|
binding_list_model->appendRow({button, xcoord, ycoord});
|
||||||
ui->binding_list->setFocus();
|
ui->binding_list->setFocus();
|
||||||
ui->binding_list->setCurrentIndex(button->index());
|
ui->binding_list->setCurrentIndex(button->index());
|
||||||
|
|
||||||
@ -277,13 +290,86 @@ void ConfigureTouchFromButton::EditBinding(const QModelIndex& qi) {
|
|||||||
void ConfigureTouchFromButton::DeleteBinding() {
|
void ConfigureTouchFromButton::DeleteBinding() {
|
||||||
const int row_index = ui->binding_list->currentIndex().row();
|
const int row_index = ui->binding_list->currentIndex().row();
|
||||||
if (row_index >= 0) {
|
if (row_index >= 0) {
|
||||||
|
ui->bottom_screen->RemoveDot(
|
||||||
|
binding_list_model->index(row_index, 0).data(data_role_dot).toInt());
|
||||||
binding_list_model->removeRow(row_index);
|
binding_list_model->removeRow(row_index);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ConfigureTouchFromButton::SetPollingResult(const Common::ParamPackage& params, bool cancel) {
|
void ConfigureTouchFromButton::OnBindingSelection(const QItemSelection& selected,
|
||||||
|
const QItemSelection& deselected) {
|
||||||
|
ui->button_delete_bind->setEnabled(!selected.isEmpty());
|
||||||
|
if (!selected.isEmpty()) {
|
||||||
|
const auto dot_data = selected.indexes().first().data(data_role_dot);
|
||||||
|
if (dot_data.isValid()) {
|
||||||
|
ui->bottom_screen->HighlightDot(dot_data.toInt());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!deselected.isEmpty()) {
|
||||||
|
const auto dot_data = deselected.indexes().first().data(data_role_dot);
|
||||||
|
if (dot_data.isValid()) {
|
||||||
|
ui->bottom_screen->HighlightDot(dot_data.toInt(), false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConfigureTouchFromButton::OnBindingChanged(QStandardItem* item) {
|
||||||
|
if (item->column() == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const bool blocked = binding_list_model->blockSignals(true);
|
||||||
|
item->setText(QString::number(std::clamp(
|
||||||
|
item->text().toInt(), 0,
|
||||||
|
(item->column() == 1 ? Core::kScreenBottomWidth : Core::kScreenBottomHeight) - 1)));
|
||||||
|
binding_list_model->blockSignals(blocked);
|
||||||
|
|
||||||
|
const auto dot_data = binding_list_model->index(item->row(), 0).data(data_role_dot);
|
||||||
|
if (dot_data.isValid()) {
|
||||||
|
ui->bottom_screen->MoveDot(dot_data.toInt(),
|
||||||
|
binding_list_model->item(item->row(), 1)->text().toInt(),
|
||||||
|
binding_list_model->item(item->row(), 2)->text().toInt());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConfigureTouchFromButton::OnBindingDeleted(const QModelIndex& parent, int first, int last) {
|
||||||
|
for (int i = first; i <= last; ++i) {
|
||||||
|
auto ix = binding_list_model->index(i, 0);
|
||||||
|
if (!ix.isValid()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto dot_data = ix.data(data_role_dot);
|
||||||
|
if (dot_data.isValid()) {
|
||||||
|
ui->bottom_screen->RemoveDot(dot_data.toInt());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConfigureTouchFromButton::SetActiveBinding(const int dot_id) {
|
||||||
|
for (int i = 0; i < binding_list_model->rowCount(); ++i) {
|
||||||
|
if (binding_list_model->index(i, 0).data(data_role_dot) == dot_id) {
|
||||||
|
ui->binding_list->setCurrentIndex(binding_list_model->index(i, 0));
|
||||||
|
ui->binding_list->setFocus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConfigureTouchFromButton::SetCoordinates(const int dot_id, const QPoint& pos) {
|
||||||
|
for (int i = 0; i < binding_list_model->rowCount(); ++i) {
|
||||||
|
if (binding_list_model->item(i, 0)->data(data_role_dot) == dot_id) {
|
||||||
|
binding_list_model->item(i, 1)->setText(QString::number(pos.x()));
|
||||||
|
binding_list_model->item(i, 2)->setText(QString::number(pos.y()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConfigureTouchFromButton::SetPollingResult(const Common::ParamPackage& params,
|
||||||
|
const bool cancel) {
|
||||||
releaseKeyboard();
|
releaseKeyboard();
|
||||||
releaseMouse();
|
releaseMouse();
|
||||||
|
qApp->restoreOverrideCursor();
|
||||||
timeout_timer->stop();
|
timeout_timer->stop();
|
||||||
poll_timer->stop();
|
poll_timer->stop();
|
||||||
for (auto& poller : device_pollers) {
|
for (auto& poller : device_pollers) {
|
||||||
@ -296,8 +382,14 @@ void ConfigureTouchFromButton::SetPollingResult(const Common::ParamPackage& para
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ConfigureTouchFromButton::keyPressEvent(QKeyEvent* event) {
|
void ConfigureTouchFromButton::keyPressEvent(QKeyEvent* event) {
|
||||||
if (!input_setter || !event)
|
if (!input_setter && event->key() == Qt::Key_Delete) {
|
||||||
|
DeleteBinding();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!input_setter) {
|
||||||
return QDialog::keyPressEvent(event);
|
return QDialog::keyPressEvent(event);
|
||||||
|
}
|
||||||
|
|
||||||
if (event->key() != Qt::Key_Escape) {
|
if (event->key() != Qt::Key_Escape) {
|
||||||
SetPollingResult(Common::ParamPackage{InputCommon::GenerateKeyboardParam(event->key())},
|
SetPollingResult(Common::ParamPackage{InputCommon::GenerateKeyboardParam(event->key())},
|
||||||
@ -312,10 +404,210 @@ void ConfigureTouchFromButton::ApplyConfiguration() {
|
|||||||
accept();
|
accept();
|
||||||
}
|
}
|
||||||
|
|
||||||
const int ConfigureTouchFromButton::GetSelectedIndex() {
|
int ConfigureTouchFromButton::GetSelectedIndex() const {
|
||||||
return selected_index;
|
return selected_index;
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::vector<Settings::TouchFromButtonMap> ConfigureTouchFromButton::GetMaps() {
|
std::vector<Settings::TouchFromButtonMap> ConfigureTouchFromButton::GetMaps() const {
|
||||||
return touch_maps;
|
return touch_maps;
|
||||||
};
|
}
|
||||||
|
|
||||||
|
TouchScreenPreview::TouchScreenPreview(QWidget* parent) : QFrame(parent) {
|
||||||
|
setBackgroundRole(QPalette::ColorRole::Base);
|
||||||
|
}
|
||||||
|
|
||||||
|
TouchScreenPreview::~TouchScreenPreview() = default;
|
||||||
|
|
||||||
|
void TouchScreenPreview::SetCoordLabel(QLabel* const label) {
|
||||||
|
coord_label = label;
|
||||||
|
}
|
||||||
|
|
||||||
|
int TouchScreenPreview::AddDot(const int device_x, const int device_y) {
|
||||||
|
QFont dot_font{QStringLiteral("monospace")};
|
||||||
|
dot_font.setStyleHint(QFont::Monospace);
|
||||||
|
dot_font.setPointSize(20);
|
||||||
|
|
||||||
|
QLabel* dot = new QLabel(this);
|
||||||
|
dot->setAttribute(Qt::WA_TranslucentBackground);
|
||||||
|
dot->setFont(dot_font);
|
||||||
|
dot->setText(QChar(0xD7)); // U+00D7 Multiplication Sign
|
||||||
|
dot->setAlignment(Qt::AlignmentFlag::AlignCenter);
|
||||||
|
dot->setProperty(prop_id, ++max_dot_id);
|
||||||
|
dot->setProperty(prop_x, device_x);
|
||||||
|
dot->setProperty(prop_y, device_y);
|
||||||
|
dot->setCursor(Qt::CursorShape::PointingHandCursor);
|
||||||
|
dot->setMouseTracking(true);
|
||||||
|
dot->installEventFilter(this);
|
||||||
|
dot->show();
|
||||||
|
PositionDot(dot, device_x, device_y);
|
||||||
|
dots.emplace_back(max_dot_id, dot);
|
||||||
|
return max_dot_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::RemoveDot(const int id) {
|
||||||
|
for (auto dot_it = dots.begin(); dot_it < dots.end(); ++dot_it) {
|
||||||
|
if (dot_it->first == id) {
|
||||||
|
dot_it->second->deleteLater();
|
||||||
|
dots.erase(dot_it);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::HighlightDot(const int id, const bool active) const {
|
||||||
|
for (const auto& dot : dots) {
|
||||||
|
if (dot.first == id) {
|
||||||
|
// use color property from the stylesheet, or fall back to the default palette
|
||||||
|
if (dot_highlight_color.isValid()) {
|
||||||
|
dot.second->setStyleSheet(
|
||||||
|
active ? QStringLiteral("color: %1").arg(dot_highlight_color.name())
|
||||||
|
: QString{});
|
||||||
|
} else {
|
||||||
|
dot.second->setForegroundRole(active ? QPalette::ColorRole::LinkVisited
|
||||||
|
: QPalette::ColorRole::NoRole);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::MoveDot(const int id, const int device_x, const int device_y) const {
|
||||||
|
for (const auto& dot : dots) {
|
||||||
|
if (dot.first == id) {
|
||||||
|
dot.second->setProperty(prop_x, device_x);
|
||||||
|
dot.second->setProperty(prop_y, device_y);
|
||||||
|
PositionDot(dot.second, device_x, device_y);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::resizeEvent(QResizeEvent* event) {
|
||||||
|
if (ignore_resize) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const int target_width = std::min(width(), height() * 4 / 3);
|
||||||
|
const int target_height = std::min(height(), width() * 3 / 4);
|
||||||
|
if (target_width == width() && target_height == height()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ignore_resize = true;
|
||||||
|
setGeometry((parentWidget()->contentsRect().width() - target_width) / 2, y(), target_width,
|
||||||
|
target_height);
|
||||||
|
ignore_resize = false;
|
||||||
|
|
||||||
|
if (event->oldSize().width() != target_width || event->oldSize().height() != target_height) {
|
||||||
|
for (const auto& dot : dots) {
|
||||||
|
PositionDot(dot.second);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::mouseMoveEvent(QMouseEvent* event) {
|
||||||
|
if (!coord_label) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const auto point = MapToDeviceCoords(event->x(), event->y());
|
||||||
|
if (point.has_value()) {
|
||||||
|
coord_label->setText(QStringLiteral("X: %1, Y: %2").arg(point->x()).arg(point->y()));
|
||||||
|
} else {
|
||||||
|
coord_label->clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::leaveEvent(QEvent* event) {
|
||||||
|
if (coord_label) {
|
||||||
|
coord_label->clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::mousePressEvent(QMouseEvent* event) {
|
||||||
|
if (event->button() == Qt::MouseButton::LeftButton) {
|
||||||
|
const auto pos = MapToDeviceCoords(event->x(), event->y());
|
||||||
|
if (pos.has_value()) {
|
||||||
|
emit DotAdded(*pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TouchScreenPreview::eventFilter(QObject* obj, QEvent* event) {
|
||||||
|
switch (event->type()) {
|
||||||
|
case QEvent::Type::MouseButtonPress: {
|
||||||
|
const auto mouse_event = static_cast<QMouseEvent*>(event);
|
||||||
|
if (mouse_event->button() != Qt::MouseButton::LeftButton) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
emit DotSelected(obj->property(prop_id).toInt());
|
||||||
|
|
||||||
|
drag_state.dot = qobject_cast<QLabel*>(obj);
|
||||||
|
drag_state.start_pos = mouse_event->globalPos();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
case QEvent::Type::MouseMove: {
|
||||||
|
if (!drag_state.dot) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
const auto mouse_event = static_cast<QMouseEvent*>(event);
|
||||||
|
if (!drag_state.active) {
|
||||||
|
drag_state.active =
|
||||||
|
(mouse_event->globalPos() - drag_state.start_pos).manhattanLength() >=
|
||||||
|
QApplication::startDragDistance();
|
||||||
|
if (!drag_state.active) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
auto current_pos = mapFromGlobal(mouse_event->globalPos());
|
||||||
|
current_pos.setX(std::clamp(current_pos.x(), contentsMargins().left(),
|
||||||
|
contentsMargins().left() + contentsRect().width()));
|
||||||
|
current_pos.setY(std::clamp(current_pos.y(), contentsMargins().top(),
|
||||||
|
contentsMargins().top() + contentsRect().height()));
|
||||||
|
const auto device_coord = MapToDeviceCoords(current_pos.x(), current_pos.y());
|
||||||
|
if (device_coord.has_value()) {
|
||||||
|
drag_state.dot->setProperty(prop_x, device_coord->x());
|
||||||
|
drag_state.dot->setProperty(prop_y, device_coord->y());
|
||||||
|
PositionDot(drag_state.dot, device_coord->x(), device_coord->y());
|
||||||
|
emit DotMoved(drag_state.dot->property(prop_id).toInt(), *device_coord);
|
||||||
|
if (coord_label) {
|
||||||
|
coord_label->setText(
|
||||||
|
QStringLiteral("X: %1, Y: %2").arg(device_coord->x()).arg(device_coord->y()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
case QEvent::Type::MouseButtonRelease: {
|
||||||
|
drag_state.dot.clear();
|
||||||
|
drag_state.active = false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return obj->eventFilter(obj, event);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<QPoint> TouchScreenPreview::MapToDeviceCoords(const int screen_x,
|
||||||
|
const int screen_y) const {
|
||||||
|
const float t_x = 0.5f + static_cast<float>(screen_x - contentsMargins().left()) *
|
||||||
|
(Core::kScreenBottomWidth - 1) / contentsRect().width();
|
||||||
|
const float t_y = 0.5f + static_cast<float>(screen_y - contentsMargins().top()) *
|
||||||
|
(Core::kScreenBottomHeight - 1) / contentsRect().height();
|
||||||
|
if (t_x >= 0.5f && t_x < Core::kScreenBottomWidth && t_y >= 0.5f &&
|
||||||
|
t_y < Core::kScreenBottomHeight) {
|
||||||
|
|
||||||
|
return QPoint{static_cast<int>(t_x), static_cast<int>(t_y)};
|
||||||
|
}
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
void TouchScreenPreview::PositionDot(QLabel* const dot, const int device_x,
|
||||||
|
const int device_y) const {
|
||||||
|
dot->move(static_cast<int>(
|
||||||
|
static_cast<float>(device_x >= 0 ? device_x : dot->property(prop_x).toInt()) *
|
||||||
|
(contentsRect().width() - 1) / (Core::kScreenBottomWidth - 1) +
|
||||||
|
contentsMargins().left() - static_cast<float>(dot->width()) / 2 + 0.5f),
|
||||||
|
static_cast<int>(
|
||||||
|
static_cast<float>(device_y >= 0 ? device_y : dot->property(prop_y).toInt()) *
|
||||||
|
(contentsRect().height() - 1) / (Core::kScreenBottomHeight - 1) +
|
||||||
|
contentsMargins().top() - static_cast<float>(dot->height()) / 2 + 0.5f));
|
||||||
|
}
|
||||||
|
@ -7,12 +7,14 @@
|
|||||||
#include <functional>
|
#include <functional>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
#include <vector>
|
||||||
#include <QDialog>
|
#include <QDialog>
|
||||||
#include "core/settings.h"
|
#include "core/settings.h"
|
||||||
|
|
||||||
class QKeyEvent;
|
class QItemSelection;
|
||||||
class QModelIndex;
|
class QModelIndex;
|
||||||
class QStandardItemModel;
|
class QStandardItemModel;
|
||||||
|
class QStandardItem;
|
||||||
class QTimer;
|
class QTimer;
|
||||||
|
|
||||||
namespace Common {
|
namespace Common {
|
||||||
@ -34,33 +36,40 @@ class ConfigureTouchFromButton : public QDialog {
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
explicit ConfigureTouchFromButton(QWidget* parent,
|
explicit ConfigureTouchFromButton(QWidget* parent,
|
||||||
std::vector<Settings::TouchFromButtonMap> touch_maps,
|
const std::vector<Settings::TouchFromButtonMap>& touch_maps,
|
||||||
int default_index = 0);
|
const int default_index = 0);
|
||||||
~ConfigureTouchFromButton() override;
|
~ConfigureTouchFromButton() override;
|
||||||
|
|
||||||
const int GetSelectedIndex();
|
int GetSelectedIndex() const;
|
||||||
const std::vector<Settings::TouchFromButtonMap> GetMaps();
|
std::vector<Settings::TouchFromButtonMap> GetMaps() const;
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void ApplyConfiguration();
|
void ApplyConfiguration();
|
||||||
|
void NewBinding(const QPoint& pos);
|
||||||
|
void SetActiveBinding(const int dot_id);
|
||||||
|
void SetCoordinates(const int dot_id, const QPoint& pos);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void showEvent(QShowEvent* ev);
|
virtual void showEvent(QShowEvent* ev) override;
|
||||||
|
virtual void keyPressEvent(QKeyEvent* event) override;
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void NewMapping();
|
||||||
|
void DeleteMapping();
|
||||||
|
void RenameMapping();
|
||||||
|
void EditBinding(const QModelIndex& qi);
|
||||||
|
void DeleteBinding();
|
||||||
|
void OnBindingSelection(const QItemSelection& selected, const QItemSelection& deselected);
|
||||||
|
void OnBindingChanged(QStandardItem* item);
|
||||||
|
void OnBindingDeleted(const QModelIndex& parent, int first, int last);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void SetConfiguration();
|
void SetConfiguration();
|
||||||
void UpdateUiDisplay();
|
void UpdateUiDisplay();
|
||||||
void ConnectEvents();
|
void ConnectEvents();
|
||||||
void NewMapping();
|
void GetButtonInput(const int row_index, const bool is_new);
|
||||||
void DeleteMapping();
|
void SetPollingResult(const Common::ParamPackage& params, const bool cancel);
|
||||||
void RenameMapping();
|
|
||||||
void NewBinding();
|
|
||||||
void EditBinding(const QModelIndex& qi);
|
|
||||||
void DeleteBinding();
|
|
||||||
void GetButtonInput(int row_index, bool is_new);
|
|
||||||
void SetPollingResult(const Common::ParamPackage& params, bool cancel);
|
|
||||||
void SaveCurrentMapping();
|
void SaveCurrentMapping();
|
||||||
void keyPressEvent(QKeyEvent* event) override;
|
|
||||||
|
|
||||||
std::unique_ptr<Ui::ConfigureTouchFromButton> ui;
|
std::unique_ptr<Ui::ConfigureTouchFromButton> ui;
|
||||||
std::unique_ptr<QStandardItemModel> binding_list_model;
|
std::unique_ptr<QStandardItemModel> binding_list_model;
|
||||||
@ -71,4 +80,6 @@ private:
|
|||||||
std::unique_ptr<QTimer> poll_timer;
|
std::unique_ptr<QTimer> poll_timer;
|
||||||
std::vector<std::unique_ptr<InputCommon::Polling::DevicePoller>> device_pollers;
|
std::vector<std::unique_ptr<InputCommon::Polling::DevicePoller>> device_pollers;
|
||||||
std::optional<std::function<void(const Common::ParamPackage&, const bool)>> input_setter;
|
std::optional<std::function<void(const Common::ParamPackage&, const bool)>> input_setter;
|
||||||
|
|
||||||
|
static constexpr int data_role_dot = Qt::ItemDataRole::UserRole + 2;
|
||||||
};
|
};
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>500</width>
|
<width>500</width>
|
||||||
<height>450</height>
|
<height>500</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
@ -21,6 +21,9 @@
|
|||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Mapping:</string>
|
<string>Mapping:</string>
|
||||||
</property>
|
</property>
|
||||||
|
<property name="textFormat">
|
||||||
|
<enum>Qt::PlainText</enum>
|
||||||
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
@ -86,7 +89,11 @@
|
|||||||
<item>
|
<item>
|
||||||
<widget class="QLabel" name="label_2">
|
<widget class="QLabel" name="label_2">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Double-click to change a field.</string>
|
<string>Click the bottom area to add a point, then press a button to bind.
|
||||||
|
Drag points to change position, or double-click table cells to edit values.</string>
|
||||||
|
</property>
|
||||||
|
<property name="textFormat">
|
||||||
|
<enum>Qt::PlainText</enum>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
@ -103,17 +110,10 @@
|
|||||||
</property>
|
</property>
|
||||||
</spacer>
|
</spacer>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
|
||||||
<widget class="QPushButton" name="button_add_bind">
|
|
||||||
<property name="text">
|
|
||||||
<string>Add</string>
|
|
||||||
</property>
|
|
||||||
</widget>
|
|
||||||
</item>
|
|
||||||
<item>
|
<item>
|
||||||
<widget class="QPushButton" name="button_delete_bind">
|
<widget class="QPushButton" name="button_delete_bind">
|
||||||
<property name="text">
|
<property name="text">
|
||||||
<string>Delete</string>
|
<string>Delete point</string>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
@ -139,14 +139,76 @@
|
|||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QDialogButtonBox" name="buttonBox">
|
<widget class="TouchScreenPreview" name="bottom_screen">
|
||||||
<property name="standardButtons">
|
<property name="sizePolicy">
|
||||||
<set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
|
<sizepolicy hsizetype="Expanding" vsizetype="Expanding">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="minimumSize">
|
||||||
|
<size>
|
||||||
|
<width>160</width>
|
||||||
|
<height>120</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
<property name="baseSize">
|
||||||
|
<size>
|
||||||
|
<width>320</width>
|
||||||
|
<height>240</height>
|
||||||
|
</size>
|
||||||
|
</property>
|
||||||
|
<property name="cursor">
|
||||||
|
<cursorShape>CrossCursor</cursorShape>
|
||||||
|
</property>
|
||||||
|
<property name="mouseTracking">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
<property name="autoFillBackground">
|
||||||
|
<bool>true</bool>
|
||||||
|
</property>
|
||||||
|
<property name="frameShape">
|
||||||
|
<enum>QFrame::StyledPanel</enum>
|
||||||
|
</property>
|
||||||
|
<property name="frameShadow">
|
||||||
|
<enum>QFrame::Sunken</enum>
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
|
<item>
|
||||||
|
<layout class="QHBoxLayout" name="horizontalLayout_3">
|
||||||
|
<item>
|
||||||
|
<widget class="QLabel" name="coord_label">
|
||||||
|
<property name="sizePolicy">
|
||||||
|
<sizepolicy hsizetype="Expanding" vsizetype="Preferred">
|
||||||
|
<horstretch>0</horstretch>
|
||||||
|
<verstretch>0</verstretch>
|
||||||
|
</sizepolicy>
|
||||||
|
</property>
|
||||||
|
<property name="textFormat">
|
||||||
|
<enum>Qt::PlainText</enum>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
<item>
|
||||||
|
<widget class="QDialogButtonBox" name="buttonBox">
|
||||||
|
<property name="standardButtons">
|
||||||
|
<set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
|
||||||
|
</property>
|
||||||
|
</widget>
|
||||||
|
</item>
|
||||||
|
</layout>
|
||||||
|
</item>
|
||||||
</layout>
|
</layout>
|
||||||
</widget>
|
</widget>
|
||||||
|
<customwidgets>
|
||||||
|
<customwidget>
|
||||||
|
<class>TouchScreenPreview</class>
|
||||||
|
<extends>QFrame</extends>
|
||||||
|
<header>citra_qt/configuration/configure_touch_widget.h</header>
|
||||||
|
<container>1</container>
|
||||||
|
</customwidget>
|
||||||
|
</customwidgets>
|
||||||
<resources/>
|
<resources/>
|
||||||
<connections>
|
<connections>
|
||||||
<connection>
|
<connection>
|
||||||
|
61
src/citra_qt/configuration/configure_touch_widget.h
Normal file
61
src/citra_qt/configuration/configure_touch_widget.h
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
// Copyright 2020 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <optional>
|
||||||
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
#include <QFrame>
|
||||||
|
#include <QPointer>
|
||||||
|
|
||||||
|
class QLabel;
|
||||||
|
|
||||||
|
// Widget for representing touchscreen coordinates
|
||||||
|
class TouchScreenPreview : public QFrame {
|
||||||
|
Q_OBJECT
|
||||||
|
Q_PROPERTY(QColor dotHighlightColor MEMBER dot_highlight_color)
|
||||||
|
|
||||||
|
public:
|
||||||
|
TouchScreenPreview(QWidget* parent);
|
||||||
|
~TouchScreenPreview() override;
|
||||||
|
|
||||||
|
void SetCoordLabel(QLabel* const);
|
||||||
|
int AddDot(const int device_x, const int device_y);
|
||||||
|
void RemoveDot(const int id);
|
||||||
|
void HighlightDot(const int id, const bool active = true) const;
|
||||||
|
void MoveDot(const int id, const int device_x, const int device_y) const;
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void DotAdded(const QPoint& pos);
|
||||||
|
void DotSelected(const int dot_id);
|
||||||
|
void DotMoved(const int dot_id, const QPoint& pos);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual void resizeEvent(QResizeEvent*) override;
|
||||||
|
virtual void mouseMoveEvent(QMouseEvent*) override;
|
||||||
|
virtual void leaveEvent(QEvent*) override;
|
||||||
|
virtual void mousePressEvent(QMouseEvent*) override;
|
||||||
|
virtual bool eventFilter(QObject*, QEvent*) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::optional<QPoint> MapToDeviceCoords(const int screen_x, const int screen_y) const;
|
||||||
|
void PositionDot(QLabel* const dot, const int device_x = -1, const int device_y = -1) const;
|
||||||
|
|
||||||
|
bool ignore_resize = false;
|
||||||
|
QPointer<QLabel> coord_label;
|
||||||
|
|
||||||
|
std::vector<std::pair<int, QLabel*>> dots;
|
||||||
|
int max_dot_id = 0;
|
||||||
|
QColor dot_highlight_color;
|
||||||
|
static constexpr char prop_id[] = "dot_id";
|
||||||
|
static constexpr char prop_x[] = "device_x";
|
||||||
|
static constexpr char prop_y[] = "device_y";
|
||||||
|
|
||||||
|
struct {
|
||||||
|
bool active = false;
|
||||||
|
QPointer<QLabel> dot;
|
||||||
|
QPoint start_pos;
|
||||||
|
} drag_state;
|
||||||
|
};
|
Loading…
Reference in New Issue
Block a user