2017-06-13 17:16:41 +02:00
|
|
|
// Copyright 2017 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2+
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-07-07 00:40:15 +02:00
|
|
|
#include "DolphinQt/Config/Mapping/IOWindow.h"
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
#include <optional>
|
2017-06-13 17:16:41 +02:00
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#include <QComboBox>
|
|
|
|
#include <QDialogButtonBox>
|
|
|
|
#include <QGroupBox>
|
|
|
|
#include <QHBoxLayout>
|
2019-01-06 17:15:00 -06:00
|
|
|
#include <QHeaderView>
|
|
|
|
#include <QItemDelegate>
|
2017-06-13 17:16:41 +02:00
|
|
|
#include <QLabel>
|
2019-03-02 14:47:26 -06:00
|
|
|
#include <QLineEdit>
|
2019-01-06 17:15:00 -06:00
|
|
|
#include <QPainter>
|
2017-06-13 17:16:41 +02:00
|
|
|
#include <QPlainTextEdit>
|
|
|
|
#include <QPushButton>
|
|
|
|
#include <QSlider>
|
|
|
|
#include <QSpinBox>
|
2019-01-06 17:15:00 -06:00
|
|
|
#include <QTableWidget>
|
2017-06-13 17:16:41 +02:00
|
|
|
#include <QVBoxLayout>
|
|
|
|
|
|
|
|
#include "Core/Core.h"
|
2018-05-28 03:48:04 +02:00
|
|
|
|
2018-07-07 00:40:15 +02:00
|
|
|
#include "DolphinQt/Config/Mapping/MappingCommon.h"
|
2019-01-06 17:15:00 -06:00
|
|
|
#include "DolphinQt/Config/Mapping/MappingIndicator.h"
|
|
|
|
#include "DolphinQt/Config/Mapping/MappingWidget.h"
|
2018-07-07 00:40:15 +02:00
|
|
|
#include "DolphinQt/Config/Mapping/MappingWindow.h"
|
|
|
|
#include "DolphinQt/QtUtils/BlockUserInputFilter.h"
|
2019-01-06 17:15:00 -06:00
|
|
|
#include "DolphinQt/QtUtils/ModalMessageBox.h"
|
2018-05-28 03:48:04 +02:00
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
#include "InputCommon/ControlReference/ControlReference.h"
|
2019-03-02 10:10:26 -06:00
|
|
|
#include "InputCommon/ControlReference/ExpressionParser.h"
|
2017-06-13 17:16:41 +02:00
|
|
|
#include "InputCommon/ControllerEmu/ControllerEmu.h"
|
|
|
|
#include "InputCommon/ControllerInterface/ControllerInterface.h"
|
|
|
|
|
|
|
|
constexpr int SLIDER_TICK_COUNT = 100;
|
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
namespace
|
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
// TODO: Make sure these functions return colors that will be visible in the current theme.
|
2019-03-02 10:10:26 -06:00
|
|
|
QTextCharFormat GetSpecialCharFormat()
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
|
|
|
format.setFontWeight(QFont::Weight::Bold);
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCharFormat GetLiteralCharFormat()
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
|
|
|
format.setForeground(QBrush{Qt::darkMagenta});
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCharFormat GetInvalidCharFormat()
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
|
|
|
format.setUnderlineStyle(QTextCharFormat::WaveUnderline);
|
|
|
|
format.setUnderlineColor(Qt::darkRed);
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCharFormat GetControlCharFormat()
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
|
|
|
format.setForeground(QBrush{Qt::darkGreen});
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCharFormat GetVariableCharFormat()
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
2019-10-12 11:41:02 -05:00
|
|
|
format.setForeground(QBrush{Qt::darkYellow});
|
2019-03-02 10:10:26 -06:00
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
2019-04-04 17:35:49 -05:00
|
|
|
QTextCharFormat GetBarewordCharFormat()
|
2019-03-02 10:10:26 -06:00
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
2019-10-12 11:41:02 -05:00
|
|
|
format.setForeground(QBrush{Qt::darkBlue});
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextCharFormat GetCommentCharFormat()
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
|
|
|
format.setForeground(QBrush{Qt::darkGray});
|
2019-03-02 10:10:26 -06:00
|
|
|
return format;
|
|
|
|
}
|
2019-03-02 14:47:26 -06:00
|
|
|
} // namespace
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2020-11-08 01:10:26 +01:00
|
|
|
ControlExpressionSyntaxHighlighter::ControlExpressionSyntaxHighlighter(QTextDocument* parent)
|
|
|
|
: QSyntaxHighlighter(parent)
|
2019-03-02 10:10:26 -06:00
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
}
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2021-02-03 20:11:38 -08:00
|
|
|
void QComboBoxWithMouseWheelDisabled::wheelEvent(QWheelEvent* event)
|
|
|
|
{
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
|
2019-10-12 11:41:02 -05:00
|
|
|
void ControlExpressionSyntaxHighlighter::highlightBlock(const QString&)
|
2019-03-02 14:47:26 -06:00
|
|
|
{
|
|
|
|
// TODO: This is going to result in improper highlighting with non-ascii characters:
|
2019-10-12 11:41:02 -05:00
|
|
|
ciface::ExpressionParser::Lexer lexer(document()->toPlainText().toStdString());
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
std::vector<ciface::ExpressionParser::Token> tokens;
|
|
|
|
const auto tokenize_status = lexer.Tokenize(tokens);
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
using ciface::ExpressionParser::TokenType;
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2019-10-12 11:41:02 -05:00
|
|
|
const auto set_block_format = [this](int start, int count, const QTextCharFormat& format) {
|
|
|
|
if (start + count <= currentBlock().position() ||
|
|
|
|
start >= currentBlock().position() + currentBlock().length())
|
|
|
|
{
|
|
|
|
// This range is not within the current block.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int block_start = start - currentBlock().position();
|
|
|
|
|
|
|
|
if (block_start < 0)
|
|
|
|
{
|
|
|
|
count += block_start;
|
|
|
|
block_start = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
setFormat(block_start, count, format);
|
|
|
|
};
|
|
|
|
|
2019-03-02 14:47:26 -06:00
|
|
|
for (auto& token : tokens)
|
|
|
|
{
|
|
|
|
std::optional<QTextCharFormat> char_format;
|
|
|
|
|
|
|
|
switch (token.type)
|
2019-03-02 10:10:26 -06:00
|
|
|
{
|
2019-03-02 14:47:26 -06:00
|
|
|
case TokenType::TOK_INVALID:
|
|
|
|
char_format = GetInvalidCharFormat();
|
|
|
|
break;
|
|
|
|
case TokenType::TOK_LPAREN:
|
|
|
|
case TokenType::TOK_RPAREN:
|
|
|
|
case TokenType::TOK_COMMA:
|
|
|
|
char_format = GetSpecialCharFormat();
|
|
|
|
break;
|
|
|
|
case TokenType::TOK_LITERAL:
|
|
|
|
char_format = GetLiteralCharFormat();
|
|
|
|
break;
|
|
|
|
case TokenType::TOK_CONTROL:
|
|
|
|
char_format = GetControlCharFormat();
|
|
|
|
break;
|
2019-04-04 17:35:49 -05:00
|
|
|
case TokenType::TOK_BAREWORD:
|
|
|
|
char_format = GetBarewordCharFormat();
|
2019-03-02 14:47:26 -06:00
|
|
|
break;
|
|
|
|
case TokenType::TOK_VARIABLE:
|
|
|
|
char_format = GetVariableCharFormat();
|
|
|
|
break;
|
2019-10-12 11:41:02 -05:00
|
|
|
case TokenType::TOK_COMMENT:
|
|
|
|
char_format = GetCommentCharFormat();
|
|
|
|
break;
|
2019-03-02 14:47:26 -06:00
|
|
|
default:
|
|
|
|
if (token.IsBinaryOperator())
|
2019-10-12 11:41:02 -05:00
|
|
|
char_format = GetSpecialCharFormat();
|
2019-03-02 14:47:26 -06:00
|
|
|
break;
|
2019-03-02 10:10:26 -06:00
|
|
|
}
|
2019-03-02 14:47:26 -06:00
|
|
|
|
|
|
|
if (char_format.has_value())
|
2019-10-12 11:41:02 -05:00
|
|
|
set_block_format(int(token.string_position), int(token.string_length), *char_format);
|
2019-03-02 10:10:26 -06:00
|
|
|
}
|
|
|
|
|
2019-10-12 11:41:02 -05:00
|
|
|
// This doesn't need to be run for every "block", but it works.
|
2020-11-08 01:10:26 +01:00
|
|
|
if (ciface::ExpressionParser::ParseStatus::Successful == tokenize_status)
|
2019-03-02 14:47:26 -06:00
|
|
|
{
|
2019-10-12 11:41:02 -05:00
|
|
|
ciface::ExpressionParser::RemoveInertTokens(&tokens);
|
2019-03-02 14:47:26 -06:00
|
|
|
const auto parse_status = ciface::ExpressionParser::ParseTokens(tokens);
|
|
|
|
|
|
|
|
if (ciface::ExpressionParser::ParseStatus::Successful != parse_status.status)
|
|
|
|
{
|
|
|
|
const auto token = *parse_status.token;
|
2019-10-12 11:41:02 -05:00
|
|
|
set_block_format(int(token.string_position), int(token.string_length),
|
|
|
|
GetInvalidCharFormat());
|
2019-03-02 14:47:26 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
class InputStateDelegate : public QItemDelegate
|
|
|
|
{
|
|
|
|
public:
|
2020-11-22 03:33:36 +01:00
|
|
|
explicit InputStateDelegate(IOWindow* parent, int column,
|
|
|
|
std::function<ControlState(int row)> state_evaluator);
|
2019-01-06 17:15:00 -06:00
|
|
|
|
|
|
|
void paint(QPainter* painter, const QStyleOptionViewItem& option,
|
|
|
|
const QModelIndex& index) const override;
|
|
|
|
|
|
|
|
private:
|
2020-11-22 03:33:36 +01:00
|
|
|
std::function<ControlState(int row)> m_state_evaluator;
|
|
|
|
int m_column;
|
|
|
|
};
|
|
|
|
|
|
|
|
class InputStateLineEdit : public QLineEdit
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit InputStateLineEdit(std::function<ControlState()> state_evaluator);
|
|
|
|
void SetShouldPaintStateIndicator(bool value);
|
|
|
|
void paintEvent(QPaintEvent* event) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::function<ControlState()> m_state_evaluator;
|
|
|
|
bool m_should_paint_state_indicator;
|
2019-01-06 17:15:00 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
IOWindow::IOWindow(MappingWidget* parent, ControllerEmu::EmulatedController* controller,
|
2017-06-13 17:16:41 +02:00
|
|
|
ControlReference* ref, IOWindow::Type type)
|
2020-11-08 01:10:26 +01:00
|
|
|
: QDialog(parent), m_reference(ref), m_original_expression(ref->GetExpression()),
|
|
|
|
m_controller(controller), m_type(type)
|
2017-06-13 17:16:41 +02:00
|
|
|
{
|
|
|
|
CreateMainLayout();
|
2018-05-05 02:29:16 +02:00
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
connect(parent, &MappingWidget::Update, this, &IOWindow::Update);
|
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
setWindowTitle(type == IOWindow::Type::Input ? tr("Configure Input") : tr("Configure Output"));
|
2018-05-05 02:29:16 +02:00
|
|
|
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-03-14 20:27:49 -05:00
|
|
|
ConfigChanged();
|
2019-01-27 10:35:29 -06:00
|
|
|
|
|
|
|
ConnectWidgets();
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
std::shared_ptr<ciface::Core::Device> IOWindow::GetSelectedDevice()
|
|
|
|
{
|
|
|
|
return m_selected_device;
|
|
|
|
}
|
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
void IOWindow::CreateMainLayout()
|
|
|
|
{
|
|
|
|
m_main_layout = new QVBoxLayout();
|
|
|
|
|
|
|
|
m_devices_combo = new QComboBox();
|
2019-01-06 17:15:00 -06:00
|
|
|
m_option_list = new QTableWidget();
|
2017-06-13 17:16:41 +02:00
|
|
|
m_select_button = new QPushButton(tr("Select"));
|
2019-10-18 12:26:03 -05:00
|
|
|
m_detect_button = new QPushButton(tr("Detect"), this);
|
|
|
|
m_test_button = new QPushButton(tr("Test"), this);
|
2017-06-13 17:16:41 +02:00
|
|
|
m_button_box = new QDialogButtonBox();
|
|
|
|
m_clear_button = new QPushButton(tr("Clear"));
|
|
|
|
m_range_slider = new QSlider(Qt::Horizontal);
|
|
|
|
m_range_spinbox = new QSpinBox();
|
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
m_parse_text = new InputStateLineEdit([this] {
|
|
|
|
const auto lock = m_controller->GetStateLock();
|
|
|
|
return m_reference->GetState<ControlState>();
|
|
|
|
});
|
2019-03-02 14:47:26 -06:00
|
|
|
m_parse_text->setReadOnly(true);
|
|
|
|
|
2019-03-02 10:10:26 -06:00
|
|
|
m_expression_text = new QPlainTextEdit();
|
|
|
|
m_expression_text->setFont(QFontDatabase::systemFont(QFontDatabase::FixedFont));
|
2020-11-08 01:10:26 +01:00
|
|
|
new ControlExpressionSyntaxHighlighter(m_expression_text->document());
|
2019-03-02 10:10:26 -06:00
|
|
|
|
2021-02-03 20:11:38 -08:00
|
|
|
m_operators_combo = new QComboBoxWithMouseWheelDisabled();
|
2019-01-27 10:35:29 -06:00
|
|
|
m_operators_combo->addItem(tr("Operators"));
|
|
|
|
m_operators_combo->insertSeparator(1);
|
|
|
|
if (m_type == Type::Input)
|
|
|
|
{
|
|
|
|
m_operators_combo->addItem(tr("! Not"));
|
|
|
|
m_operators_combo->addItem(tr("* Multiply"));
|
|
|
|
m_operators_combo->addItem(tr("/ Divide"));
|
|
|
|
m_operators_combo->addItem(tr("% Modulo"));
|
|
|
|
m_operators_combo->addItem(tr("+ Add"));
|
|
|
|
m_operators_combo->addItem(tr("- Subtract"));
|
|
|
|
m_operators_combo->addItem(tr("> Greater-than"));
|
|
|
|
m_operators_combo->addItem(tr("< Less-than"));
|
|
|
|
m_operators_combo->addItem(tr("& And"));
|
2019-10-20 09:51:19 -05:00
|
|
|
m_operators_combo->addItem(tr("^ Xor"));
|
2019-01-27 10:35:29 -06:00
|
|
|
}
|
|
|
|
m_operators_combo->addItem(tr("| Or"));
|
|
|
|
if (m_type == Type::Input)
|
|
|
|
{
|
|
|
|
m_operators_combo->addItem(tr(", Comma"));
|
|
|
|
}
|
|
|
|
|
2021-02-03 20:11:38 -08:00
|
|
|
m_functions_combo = new QComboBoxWithMouseWheelDisabled(this);
|
2019-01-27 10:35:29 -06:00
|
|
|
m_functions_combo->addItem(tr("Functions"));
|
|
|
|
m_functions_combo->insertSeparator(1);
|
2019-04-04 17:35:49 -05:00
|
|
|
m_functions_combo->addItem(QStringLiteral("if"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("timer"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("toggle"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("deadzone"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("smooth"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("hold"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("tap"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("relative"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("pulse"));
|
2020-08-02 21:42:18 +02:00
|
|
|
m_functions_combo->addItem(QStringLiteral("sin"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("cos"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("tan"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("asin"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("acos"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("atan"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("atan2"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("sqrt"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("pow"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("min"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("max"));
|
|
|
|
m_functions_combo->addItem(QStringLiteral("clamp"));
|
2019-01-27 10:35:29 -06:00
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
// Devices
|
|
|
|
m_main_layout->addWidget(m_devices_combo);
|
|
|
|
|
|
|
|
// Range
|
|
|
|
auto* range_hbox = new QHBoxLayout();
|
|
|
|
range_hbox->addWidget(new QLabel(tr("Range")));
|
|
|
|
range_hbox->addWidget(m_range_slider);
|
|
|
|
range_hbox->addWidget(m_range_spinbox);
|
|
|
|
m_range_slider->setMinimum(-500);
|
|
|
|
m_range_slider->setMaximum(500);
|
|
|
|
m_range_spinbox->setMinimum(-500);
|
|
|
|
m_range_spinbox->setMaximum(500);
|
2018-05-08 17:54:47 -06:00
|
|
|
m_main_layout->addLayout(range_hbox);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
// Options (Buttons, Outputs) and action buttons
|
|
|
|
|
2020-02-06 20:34:27 -06:00
|
|
|
m_option_list->setTabKeyNavigation(false);
|
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
if (m_type == Type::Input)
|
|
|
|
{
|
|
|
|
m_option_list->setColumnCount(2);
|
|
|
|
m_option_list->setColumnWidth(1, 64);
|
|
|
|
m_option_list->horizontalHeader()->setSectionResizeMode(1, QHeaderView::Fixed);
|
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
m_option_list->setItemDelegate(new InputStateDelegate(this, 1, [&](int row) {
|
|
|
|
// Clamp off negative values but allow greater than one in the text display.
|
|
|
|
return std::max(GetSelectedDevice()->Inputs()[row]->GetState(), 0.0);
|
|
|
|
}));
|
2019-01-06 17:15:00 -06:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_option_list->setColumnCount(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_option_list->horizontalHeader()->hide();
|
|
|
|
m_option_list->horizontalHeader()->setSectionResizeMode(0, QHeaderView::Stretch);
|
|
|
|
m_option_list->verticalHeader()->hide();
|
|
|
|
m_option_list->verticalHeader()->setDefaultSectionSize(
|
|
|
|
m_option_list->verticalHeader()->minimumSectionSize());
|
|
|
|
m_option_list->setEditTriggers(QAbstractItemView::NoEditTriggers);
|
|
|
|
m_option_list->setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
|
|
m_option_list->setSelectionMode(QAbstractItemView::SingleSelection);
|
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
auto* hbox = new QHBoxLayout();
|
|
|
|
auto* button_vbox = new QVBoxLayout();
|
|
|
|
hbox->addWidget(m_option_list, 8);
|
|
|
|
hbox->addLayout(button_vbox, 1);
|
|
|
|
|
|
|
|
button_vbox->addWidget(m_select_button);
|
2019-11-08 17:45:43 -06:00
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
if (m_type == Type::Input)
|
|
|
|
{
|
2019-11-08 17:45:43 -06:00
|
|
|
m_test_button->hide();
|
|
|
|
button_vbox->addWidget(m_detect_button);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_detect_button->hide();
|
|
|
|
button_vbox->addWidget(m_test_button);
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
|
2019-11-08 17:45:43 -06:00
|
|
|
button_vbox->addWidget(m_operators_combo);
|
|
|
|
|
|
|
|
if (m_type == Type::Input)
|
|
|
|
button_vbox->addWidget(m_functions_combo);
|
|
|
|
else
|
|
|
|
m_functions_combo->hide();
|
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
m_main_layout->addLayout(hbox, 2);
|
|
|
|
m_main_layout->addWidget(m_expression_text, 1);
|
2019-03-02 14:47:26 -06:00
|
|
|
m_main_layout->addWidget(m_parse_text);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
|
|
|
// Button Box
|
|
|
|
m_main_layout->addWidget(m_button_box);
|
|
|
|
m_button_box->addButton(m_clear_button, QDialogButtonBox::ActionRole);
|
|
|
|
m_button_box->addButton(QDialogButtonBox::Ok);
|
|
|
|
|
|
|
|
setLayout(m_main_layout);
|
|
|
|
}
|
|
|
|
|
2019-03-14 20:27:49 -05:00
|
|
|
void IOWindow::ConfigChanged()
|
2017-06-13 17:16:41 +02:00
|
|
|
{
|
2019-04-27 10:51:57 -05:00
|
|
|
const QSignalBlocker blocker(this);
|
2020-11-22 03:33:36 +01:00
|
|
|
const auto lock = m_controller->GetStateLock();
|
|
|
|
|
|
|
|
// ensure m_parse_text is in the right state
|
|
|
|
UpdateExpression(m_reference->GetExpression(), UpdateMode::Force);
|
2019-03-14 20:27:49 -05:00
|
|
|
|
2017-06-07 19:02:16 -07:00
|
|
|
m_expression_text->setPlainText(QString::fromStdString(m_reference->GetExpression()));
|
2018-09-03 18:03:26 +02:00
|
|
|
m_expression_text->moveCursor(QTextCursor::End, QTextCursor::MoveAnchor);
|
2017-06-13 17:16:41 +02:00
|
|
|
m_range_spinbox->setValue(m_reference->range * SLIDER_TICK_COUNT);
|
|
|
|
m_range_slider->setValue(m_reference->range * SLIDER_TICK_COUNT);
|
|
|
|
|
2017-11-04 15:29:15 -07:00
|
|
|
m_devq = m_controller->GetDefaultDevice();
|
2017-06-13 17:16:41 +02:00
|
|
|
|
|
|
|
UpdateDeviceList();
|
|
|
|
UpdateOptionList();
|
|
|
|
}
|
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
void IOWindow::Update()
|
|
|
|
{
|
|
|
|
m_option_list->viewport()->update();
|
2020-11-22 03:33:36 +01:00
|
|
|
m_parse_text->update();
|
2019-01-06 17:15:00 -06:00
|
|
|
}
|
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
void IOWindow::ConnectWidgets()
|
|
|
|
{
|
2019-01-27 10:35:29 -06:00
|
|
|
connect(m_select_button, &QPushButton::clicked, [this] { AppendSelectedOption(); });
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-02-27 18:10:18 -06:00
|
|
|
connect(m_detect_button, &QPushButton::clicked, this, &IOWindow::OnDetectButtonPressed);
|
|
|
|
connect(m_test_button, &QPushButton::clicked, this, &IOWindow::OnTestButtonPressed);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
|
|
|
connect(m_button_box, &QDialogButtonBox::clicked, this, &IOWindow::OnDialogButtonPressed);
|
|
|
|
connect(m_devices_combo, &QComboBox::currentTextChanged, this, &IOWindow::OnDeviceChanged);
|
2019-07-30 09:35:46 -04:00
|
|
|
connect(m_range_spinbox, qOverload<int>(&QSpinBox::valueChanged), this,
|
2017-06-13 17:16:41 +02:00
|
|
|
&IOWindow::OnRangeChanged);
|
2019-01-27 10:35:29 -06:00
|
|
|
|
2020-11-22 04:00:28 +01:00
|
|
|
connect(m_expression_text, &QPlainTextEdit::textChanged,
|
|
|
|
[this] { UpdateExpression(m_expression_text->toPlainText().toStdString()); });
|
2019-01-27 10:35:29 -06:00
|
|
|
|
2019-07-30 09:35:46 -04:00
|
|
|
connect(m_operators_combo, qOverload<int>(&QComboBox::activated), [this](int index) {
|
2019-01-27 10:35:29 -06:00
|
|
|
if (0 == index)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_expression_text->insertPlainText(m_operators_combo->currentText().left(1));
|
|
|
|
|
|
|
|
m_operators_combo->setCurrentIndex(0);
|
|
|
|
});
|
|
|
|
|
2019-07-30 09:35:46 -04:00
|
|
|
connect(m_functions_combo, qOverload<int>(&QComboBox::activated), [this](int index) {
|
2019-01-27 10:35:29 -06:00
|
|
|
if (0 == index)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_expression_text->insertPlainText(m_functions_combo->currentText() + QStringLiteral("()"));
|
|
|
|
|
|
|
|
m_functions_combo->setCurrentIndex(0);
|
|
|
|
});
|
2020-11-08 01:10:26 +01:00
|
|
|
|
|
|
|
// revert the expression when the window closes without using the OK button
|
|
|
|
connect(this, &IOWindow::finished, [this] { UpdateExpression(m_original_expression); });
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
|
2019-01-27 10:35:29 -06:00
|
|
|
void IOWindow::AppendSelectedOption()
|
2017-06-13 17:16:41 +02:00
|
|
|
{
|
|
|
|
if (m_option_list->currentItem() == nullptr)
|
|
|
|
return;
|
|
|
|
|
2019-01-27 10:35:29 -06:00
|
|
|
m_expression_text->insertPlainText(MappingCommon::GetExpressionForControl(
|
|
|
|
m_option_list->currentItem()->text(), m_devq, m_controller->GetDefaultDevice()));
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void IOWindow::OnDeviceChanged(const QString& device)
|
|
|
|
{
|
|
|
|
m_devq.FromString(device.toStdString());
|
|
|
|
UpdateOptionList();
|
|
|
|
}
|
|
|
|
|
|
|
|
void IOWindow::OnDialogButtonPressed(QAbstractButton* button)
|
|
|
|
{
|
|
|
|
if (button == m_clear_button)
|
|
|
|
{
|
|
|
|
m_expression_text->clear();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-11-08 01:10:26 +01:00
|
|
|
const auto lock = m_controller->GetStateLock();
|
|
|
|
|
|
|
|
UpdateExpression(m_expression_text->toPlainText().toStdString());
|
|
|
|
m_original_expression = m_reference->GetExpression();
|
2019-01-27 10:35:29 -06:00
|
|
|
|
|
|
|
if (ciface::ExpressionParser::ParseStatus::SyntaxError == m_reference->GetParseStatus())
|
|
|
|
{
|
2019-01-06 17:15:00 -06:00
|
|
|
ModalMessageBox::warning(this, tr("Error"), tr("The expression contains a syntax error."));
|
2019-01-27 10:35:29 -06:00
|
|
|
}
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2020-11-22 04:00:28 +01:00
|
|
|
// must be the OK button
|
|
|
|
accept();
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void IOWindow::OnDetectButtonPressed()
|
|
|
|
{
|
2019-02-27 18:10:18 -06:00
|
|
|
const auto expression =
|
|
|
|
MappingCommon::DetectExpression(m_detect_button, g_controller_interface, {m_devq.ToString()},
|
|
|
|
m_devq, MappingCommon::Quote::Off);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-02-27 18:10:18 -06:00
|
|
|
if (expression.isEmpty())
|
|
|
|
return;
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-02-27 18:10:18 -06:00
|
|
|
const auto list = m_option_list->findItems(expression, Qt::MatchFixedString);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-02-27 18:10:18 -06:00
|
|
|
if (!list.empty())
|
|
|
|
m_option_list->setCurrentItem(list[0]);
|
|
|
|
}
|
2017-06-26 23:23:22 -07:00
|
|
|
|
2019-02-27 18:10:18 -06:00
|
|
|
void IOWindow::OnTestButtonPressed()
|
|
|
|
{
|
|
|
|
MappingCommon::TestOutput(m_test_button, static_cast<OutputReference*>(m_reference));
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void IOWindow::OnRangeChanged(int value)
|
|
|
|
{
|
|
|
|
m_reference->range = static_cast<double>(value) / SLIDER_TICK_COUNT;
|
|
|
|
m_range_spinbox->setValue(m_reference->range * SLIDER_TICK_COUNT);
|
|
|
|
m_range_slider->setValue(m_reference->range * SLIDER_TICK_COUNT);
|
|
|
|
}
|
|
|
|
|
|
|
|
void IOWindow::UpdateOptionList()
|
|
|
|
{
|
2019-01-06 17:15:00 -06:00
|
|
|
m_selected_device = g_controller_interface.FindDevice(m_devq);
|
|
|
|
m_option_list->setRowCount(0);
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-01-06 17:15:00 -06:00
|
|
|
if (m_selected_device == nullptr)
|
2018-05-10 21:12:19 +02:00
|
|
|
return;
|
|
|
|
|
2017-06-13 17:16:41 +02:00
|
|
|
if (m_reference->IsInput())
|
|
|
|
{
|
2019-01-06 17:15:00 -06:00
|
|
|
int row = 0;
|
|
|
|
for (const auto* input : m_selected_device->Inputs())
|
2017-06-13 17:16:41 +02:00
|
|
|
{
|
2019-01-06 17:15:00 -06:00
|
|
|
m_option_list->insertRow(row);
|
|
|
|
m_option_list->setItem(row, 0,
|
|
|
|
new QTableWidgetItem(QString::fromStdString(input->GetName())));
|
|
|
|
++row;
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-06 17:15:00 -06:00
|
|
|
int row = 0;
|
|
|
|
for (const auto* output : m_selected_device->Outputs())
|
2017-06-13 17:16:41 +02:00
|
|
|
{
|
2019-01-06 17:15:00 -06:00
|
|
|
m_option_list->insertRow(row);
|
|
|
|
m_option_list->setItem(row, 0,
|
|
|
|
new QTableWidgetItem(QString::fromStdString(output->GetName())));
|
|
|
|
++row;
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void IOWindow::UpdateDeviceList()
|
|
|
|
{
|
|
|
|
m_devices_combo->clear();
|
|
|
|
|
2019-03-14 20:27:49 -05:00
|
|
|
for (const auto& name : g_controller_interface.GetAllDeviceStrings())
|
|
|
|
m_devices_combo->addItem(QString::fromStdString(name));
|
2017-06-13 17:16:41 +02:00
|
|
|
|
2019-03-14 20:27:49 -05:00
|
|
|
m_devices_combo->setCurrentText(
|
|
|
|
QString::fromStdString(m_controller->GetDefaultDevice().ToString()));
|
2017-06-13 17:16:41 +02:00
|
|
|
}
|
2019-01-06 17:15:00 -06:00
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
void IOWindow::UpdateExpression(std::string new_expression, UpdateMode mode)
|
2020-11-08 01:10:26 +01:00
|
|
|
{
|
|
|
|
const auto lock = m_controller->GetStateLock();
|
2020-11-22 03:33:36 +01:00
|
|
|
if (mode != UpdateMode::Force && new_expression == m_reference->GetExpression())
|
2020-11-08 01:10:26 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
const auto error = m_reference->SetExpression(std::move(new_expression));
|
|
|
|
const auto status = m_reference->GetParseStatus();
|
|
|
|
m_controller->UpdateSingleControlReference(g_controller_interface, m_reference);
|
|
|
|
|
|
|
|
if (error)
|
2020-11-22 03:33:36 +01:00
|
|
|
{
|
|
|
|
m_parse_text->SetShouldPaintStateIndicator(false);
|
2020-11-08 01:10:26 +01:00
|
|
|
m_parse_text->setText(QString::fromStdString(*error));
|
2020-11-22 03:33:36 +01:00
|
|
|
}
|
2020-11-08 01:10:26 +01:00
|
|
|
else if (status == ciface::ExpressionParser::ParseStatus::EmptyExpression)
|
2020-11-22 03:33:36 +01:00
|
|
|
{
|
|
|
|
m_parse_text->SetShouldPaintStateIndicator(false);
|
2020-11-08 01:10:26 +01:00
|
|
|
m_parse_text->setText(QString());
|
2020-11-22 03:33:36 +01:00
|
|
|
}
|
2020-11-08 01:10:26 +01:00
|
|
|
else if (status != ciface::ExpressionParser::ParseStatus::Successful)
|
2020-11-22 03:33:36 +01:00
|
|
|
{
|
|
|
|
m_parse_text->SetShouldPaintStateIndicator(false);
|
2020-11-08 01:10:26 +01:00
|
|
|
m_parse_text->setText(tr("Invalid Expression."));
|
2020-11-22 03:33:36 +01:00
|
|
|
}
|
2020-11-08 01:10:26 +01:00
|
|
|
else
|
2020-11-22 03:33:36 +01:00
|
|
|
{
|
|
|
|
m_parse_text->SetShouldPaintStateIndicator(true);
|
|
|
|
m_parse_text->setText(QString());
|
|
|
|
}
|
2020-11-08 01:10:26 +01:00
|
|
|
}
|
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
InputStateDelegate::InputStateDelegate(IOWindow* parent, int column,
|
|
|
|
std::function<ControlState(int row)> state_evaluator)
|
|
|
|
: QItemDelegate(parent), m_state_evaluator(std::move(state_evaluator)), m_column(column)
|
2019-01-06 17:15:00 -06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
InputStateLineEdit::InputStateLineEdit(std::function<ControlState()> state_evaluator)
|
|
|
|
: m_state_evaluator(std::move(state_evaluator))
|
2019-01-06 17:15:00 -06:00
|
|
|
{
|
2020-11-22 03:33:36 +01:00
|
|
|
}
|
2019-01-06 17:15:00 -06:00
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
static void PaintStateIndicator(QPainter& painter, const QRect& region, ControlState state)
|
|
|
|
{
|
|
|
|
const QString state_string = QString::number(state, 'g', 4);
|
2019-01-06 17:15:00 -06:00
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
QRect meter_region = region;
|
|
|
|
meter_region.setWidth(region.width() * std::clamp(state, 0.0, 1.0));
|
2019-01-06 17:15:00 -06:00
|
|
|
|
|
|
|
// Create a temporary indicator object to retreive color constants.
|
2020-02-23 13:34:05 -06:00
|
|
|
MappingIndicator indicator;
|
2019-01-06 17:15:00 -06:00
|
|
|
|
|
|
|
// Normal text.
|
2020-11-22 03:33:36 +01:00
|
|
|
painter.setPen(indicator.GetTextColor());
|
|
|
|
painter.drawText(region, Qt::AlignCenter, state_string);
|
2019-01-06 17:15:00 -06:00
|
|
|
|
|
|
|
// Input state meter.
|
2020-11-22 03:33:36 +01:00
|
|
|
painter.fillRect(meter_region, indicator.GetAdjustedInputColor());
|
2019-01-06 17:15:00 -06:00
|
|
|
|
|
|
|
// Text on top of meter.
|
2020-11-22 03:33:36 +01:00
|
|
|
painter.setPen(indicator.GetAltTextColor());
|
|
|
|
painter.setClipping(true);
|
|
|
|
painter.setClipRect(meter_region);
|
|
|
|
painter.drawText(region, Qt::AlignCenter, state_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
void InputStateDelegate::paint(QPainter* painter, const QStyleOptionViewItem& option,
|
|
|
|
const QModelIndex& index) const
|
|
|
|
{
|
|
|
|
QItemDelegate::paint(painter, option, index);
|
|
|
|
|
|
|
|
if (index.column() != m_column)
|
|
|
|
return;
|
2019-01-06 17:15:00 -06:00
|
|
|
|
2020-11-22 03:33:36 +01:00
|
|
|
painter->save();
|
|
|
|
PaintStateIndicator(*painter, option.rect, m_state_evaluator(index.row()));
|
2019-01-06 17:15:00 -06:00
|
|
|
painter->restore();
|
|
|
|
}
|
2020-11-22 03:33:36 +01:00
|
|
|
|
|
|
|
void InputStateLineEdit::SetShouldPaintStateIndicator(bool value)
|
|
|
|
{
|
|
|
|
m_should_paint_state_indicator = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InputStateLineEdit::paintEvent(QPaintEvent* event)
|
|
|
|
{
|
|
|
|
QLineEdit::paintEvent(event);
|
|
|
|
|
|
|
|
if (!m_should_paint_state_indicator)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QPainter painter(this);
|
|
|
|
PaintStateIndicator(painter, this->rect(), m_state_evaluator());
|
|
|
|
}
|