| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | // Copyright 2017 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  | #include <QMenu>
 | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | #include <QMessageBox>
 | 
					
						
							|  |  |  | #include <QStandardItemModel>
 | 
					
						
							| 
									
										
										
										
											2021-04-14 16:07:40 -07:00
										 |  |  | #include "common/settings.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | #include "ui_configure_hotkeys.h"
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  | #include "yuzu/configuration/config.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | #include "yuzu/configuration/configure_hotkeys.h"
 | 
					
						
							|  |  |  | #include "yuzu/hotkeys.h"
 | 
					
						
							|  |  |  | #include "yuzu/util/sequence_dialog/sequence_dialog.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ConfigureHotkeys::ConfigureHotkeys(QWidget* parent) | 
					
						
							|  |  |  |     : QWidget(parent), ui(std::make_unique<Ui::ConfigureHotkeys>()) { | 
					
						
							|  |  |  |     ui->setupUi(this); | 
					
						
							|  |  |  |     setFocusPolicy(Qt::ClickFocus); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     model = new QStandardItemModel(this); | 
					
						
							|  |  |  |     model->setColumnCount(3); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     connect(ui->hotkey_list, &QTreeView::doubleClicked, this, &ConfigureHotkeys::Configure); | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |     connect(ui->hotkey_list, &QTreeView::customContextMenuRequested, this, | 
					
						
							|  |  |  |             &ConfigureHotkeys::PopupContextMenu); | 
					
						
							|  |  |  |     ui->hotkey_list->setContextMenuPolicy(Qt::CustomContextMenu); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  |     ui->hotkey_list->setModel(model); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO(Kloen): Make context configurable as well (hiding the column for now)
 | 
					
						
							|  |  |  |     ui->hotkey_list->hideColumn(2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ui->hotkey_list->setColumnWidth(0, 200); | 
					
						
							|  |  |  |     ui->hotkey_list->resizeColumnToContents(1); | 
					
						
							| 
									
										
										
										
											2019-06-05 18:39:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |     connect(ui->button_restore_defaults, &QPushButton::clicked, this, | 
					
						
							|  |  |  |             &ConfigureHotkeys::RestoreDefaults); | 
					
						
							|  |  |  |     connect(ui->button_clear_all, &QPushButton::clicked, this, &ConfigureHotkeys::ClearAll); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-05 18:39:46 -04:00
										 |  |  |     RetranslateUI(); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ConfigureHotkeys::~ConfigureHotkeys() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ConfigureHotkeys::Populate(const HotkeyRegistry& registry) { | 
					
						
							|  |  |  |     for (const auto& group : registry.hotkey_groups) { | 
					
						
							|  |  |  |         auto* parent_item = new QStandardItem(group.first); | 
					
						
							|  |  |  |         parent_item->setEditable(false); | 
					
						
							|  |  |  |         for (const auto& hotkey : group.second) { | 
					
						
							|  |  |  |             auto* action = new QStandardItem(hotkey.first); | 
					
						
							|  |  |  |             auto* keyseq = | 
					
						
							|  |  |  |                 new QStandardItem(hotkey.second.keyseq.toString(QKeySequence::NativeText)); | 
					
						
							|  |  |  |             action->setEditable(false); | 
					
						
							|  |  |  |             keyseq->setEditable(false); | 
					
						
							|  |  |  |             parent_item->appendRow({action, keyseq}); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         model->appendRow(parent_item); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ui->hotkey_list->expandAll(); | 
					
						
							| 
									
										
										
										
											2020-01-12 22:46:28 +01:00
										 |  |  |     ui->hotkey_list->resizeColumnToContents(0); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-05 18:39:46 -04:00
										 |  |  | void ConfigureHotkeys::changeEvent(QEvent* event) { | 
					
						
							|  |  |  |     if (event->type() == QEvent::LanguageChange) { | 
					
						
							|  |  |  |         RetranslateUI(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QWidget::changeEvent(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ConfigureHotkeys::RetranslateUI() { | 
					
						
							|  |  |  |     ui->retranslateUi(this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     model->setHorizontalHeaderLabels({tr("Action"), tr("Hotkey"), tr("Context")}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | void ConfigureHotkeys::Configure(QModelIndex index) { | 
					
						
							| 
									
										
										
										
											2019-04-09 19:47:18 -04:00
										 |  |  |     if (!index.parent().isValid()) { | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-04-09 19:47:18 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     index = index.sibling(index.row(), 1); | 
					
						
							| 
									
										
										
										
											2019-04-09 19:50:14 -04:00
										 |  |  |     const auto previous_key = model->data(index); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-09 20:06:45 -04:00
										 |  |  |     SequenceDialog hotkey_dialog{this}; | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-09 19:50:59 -04:00
										 |  |  |     const int return_code = hotkey_dialog.exec(); | 
					
						
							|  |  |  |     const auto key_sequence = hotkey_dialog.GetSequence(); | 
					
						
							| 
									
										
										
										
											2019-04-09 19:50:14 -04:00
										 |  |  |     if (return_code == QDialog::Rejected || key_sequence.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-04-09 19:50:14 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |     const auto [key_sequence_used, used_action] = IsUsedKey(key_sequence); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |     if (key_sequence_used && key_sequence != QKeySequence(previous_key.toString())) { | 
					
						
							|  |  |  |         QMessageBox::warning( | 
					
						
							|  |  |  |             this, tr("Conflicting Key Sequence"), | 
					
						
							|  |  |  |             tr("The entered key sequence is already assigned to: %1").arg(used_action)); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         model->setData(index, key_sequence.toString(QKeySequence::NativeText)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  | std::pair<bool, QString> ConfigureHotkeys::IsUsedKey(QKeySequence key_sequence) const { | 
					
						
							|  |  |  |     for (int r = 0; r < model->rowCount(); ++r) { | 
					
						
							| 
									
										
										
										
											2019-05-25 04:03:15 -04:00
										 |  |  |         const QStandardItem* const parent = model->item(r, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |         for (int r2 = 0; r2 < parent->rowCount(); ++r2) { | 
					
						
							| 
									
										
										
										
											2019-05-25 04:03:15 -04:00
										 |  |  |             const QStandardItem* const key_seq_item = parent->child(r2, 1); | 
					
						
							|  |  |  |             const auto key_seq_str = key_seq_item->text(); | 
					
						
							|  |  |  |             const auto key_seq = QKeySequence::fromString(key_seq_str, QKeySequence::NativeText); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (key_sequence == key_seq) { | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |                 return std::make_pair(true, parent->child(r2, 0)->text()); | 
					
						
							| 
									
										
										
										
											2019-05-25 04:03:15 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |     return std::make_pair(false, QString()); | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-26 00:39:23 -04:00
										 |  |  | void ConfigureHotkeys::ApplyConfiguration(HotkeyRegistry& registry) { | 
					
						
							| 
									
										
										
										
											2019-02-16 16:19:29 +01:00
										 |  |  |     for (int key_id = 0; key_id < model->rowCount(); key_id++) { | 
					
						
							|  |  |  |         const QStandardItem* parent = model->item(key_id, 0); | 
					
						
							|  |  |  |         for (int key_column_id = 0; key_column_id < parent->rowCount(); key_column_id++) { | 
					
						
							|  |  |  |             const QStandardItem* action = parent->child(key_column_id, 0); | 
					
						
							|  |  |  |             const QStandardItem* keyseq = parent->child(key_column_id, 1); | 
					
						
							|  |  |  |             for (auto& [group, sub_actions] : registry.hotkey_groups) { | 
					
						
							|  |  |  |                 if (group != parent->text()) | 
					
						
							|  |  |  |                     continue; | 
					
						
							|  |  |  |                 for (auto& [action_name, hotkey] : sub_actions) { | 
					
						
							|  |  |  |                     if (action_name != action->text()) | 
					
						
							|  |  |  |                         continue; | 
					
						
							|  |  |  |                     hotkey.keyseq = QKeySequence(keyseq->text()); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     registry.SaveHotkeys(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void ConfigureHotkeys::RestoreDefaults() { | 
					
						
							|  |  |  |     for (int r = 0; r < model->rowCount(); ++r) { | 
					
						
							|  |  |  |         const QStandardItem* parent = model->item(r, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int r2 = 0; r2 < parent->rowCount(); ++r2) { | 
					
						
							|  |  |  |             model->item(r, 0)->child(r2, 1)->setText(Config::default_hotkeys[r2].shortcut.first); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ConfigureHotkeys::ClearAll() { | 
					
						
							|  |  |  |     for (int r = 0; r < model->rowCount(); ++r) { | 
					
						
							|  |  |  |         const QStandardItem* parent = model->item(r, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int r2 = 0; r2 < parent->rowCount(); ++r2) { | 
					
						
							| 
									
										
										
										
											2020-08-16 08:13:23 -04:00
										 |  |  |             model->item(r, 0)->child(r2, 1)->setText(QString{}); | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ConfigureHotkeys::PopupContextMenu(const QPoint& menu_location) { | 
					
						
							|  |  |  |     QModelIndex index = ui->hotkey_list->indexAt(menu_location); | 
					
						
							|  |  |  |     if (!index.parent().isValid()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto selected = index.sibling(index.row(), 1); | 
					
						
							|  |  |  |     QMenu context_menu; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QAction* restore_default = context_menu.addAction(tr("Restore Default")); | 
					
						
							|  |  |  |     QAction* clear = context_menu.addAction(tr("Clear")); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     connect(restore_default, &QAction::triggered, [this, selected] { | 
					
						
							|  |  |  |         const QKeySequence& default_key_sequence = QKeySequence::fromString( | 
					
						
							|  |  |  |             Config::default_hotkeys[selected.row()].shortcut.first, QKeySequence::NativeText); | 
					
						
							|  |  |  |         const auto [key_sequence_used, used_action] = IsUsedKey(default_key_sequence); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (key_sequence_used && | 
					
						
							|  |  |  |             default_key_sequence != QKeySequence(model->data(selected).toString())) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             QMessageBox::warning( | 
					
						
							|  |  |  |                 this, tr("Conflicting Key Sequence"), | 
					
						
							|  |  |  |                 tr("The default key sequence is already assigned to: %1").arg(used_action)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             model->setData(selected, default_key_sequence.toString(QKeySequence::NativeText)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-08-16 08:13:23 -04:00
										 |  |  |     connect(clear, &QAction::triggered, [this, selected] { model->setData(selected, QString{}); }); | 
					
						
							| 
									
										
										
										
											2020-04-23 13:31:16 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     context_menu.exec(ui->hotkey_list->viewport()->mapToGlobal(menu_location)); | 
					
						
							|  |  |  | } |