Merge pull request #9232 from AdmiralCurtiss/show-result-value-in-expression-editor

Qt/IOWindow: Show result value in expression editor.
This commit is contained in:
Léo Lam 2020-11-26 01:30:12 +01:00 committed by GitHub
commit 3891ac2682
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 148 additions and 64 deletions

View File

@ -93,9 +93,8 @@ QTextCharFormat GetCommentCharFormat()
} }
} // namespace } // namespace
ControlExpressionSyntaxHighlighter::ControlExpressionSyntaxHighlighter(QTextDocument* parent, ControlExpressionSyntaxHighlighter::ControlExpressionSyntaxHighlighter(QTextDocument* parent)
QLineEdit* result) : QSyntaxHighlighter(parent)
: QSyntaxHighlighter(parent), m_result_text(result)
{ {
} }
@ -168,18 +167,11 @@ void ControlExpressionSyntaxHighlighter::highlightBlock(const QString&)
} }
// This doesn't need to be run for every "block", but it works. // This doesn't need to be run for every "block", but it works.
if (ciface::ExpressionParser::ParseStatus::Successful != tokenize_status) if (ciface::ExpressionParser::ParseStatus::Successful == tokenize_status)
{
m_result_text->setText(tr("Invalid Token."));
}
else
{ {
ciface::ExpressionParser::RemoveInertTokens(&tokens); ciface::ExpressionParser::RemoveInertTokens(&tokens);
const auto parse_status = ciface::ExpressionParser::ParseTokens(tokens); const auto parse_status = ciface::ExpressionParser::ParseTokens(tokens);
m_result_text->setText(
QString::fromStdString(parse_status.description.value_or(_trans("Success."))));
if (ciface::ExpressionParser::ParseStatus::Successful != parse_status.status) if (ciface::ExpressionParser::ParseStatus::Successful != parse_status.status)
{ {
const auto token = *parse_status.token; const auto token = *parse_status.token;
@ -192,18 +184,33 @@ void ControlExpressionSyntaxHighlighter::highlightBlock(const QString&)
class InputStateDelegate : public QItemDelegate class InputStateDelegate : public QItemDelegate
{ {
public: public:
explicit InputStateDelegate(IOWindow* parent); explicit InputStateDelegate(IOWindow* parent, int column,
std::function<ControlState(int row)> state_evaluator);
void paint(QPainter* painter, const QStyleOptionViewItem& option, void paint(QPainter* painter, const QStyleOptionViewItem& option,
const QModelIndex& index) const override; const QModelIndex& index) const override;
private: private:
IOWindow* m_parent; 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;
}; };
IOWindow::IOWindow(MappingWidget* parent, ControllerEmu::EmulatedController* controller, IOWindow::IOWindow(MappingWidget* parent, ControllerEmu::EmulatedController* controller,
ControlReference* ref, IOWindow::Type type) ControlReference* ref, IOWindow::Type type)
: QDialog(parent), m_reference(ref), m_controller(controller), m_type(type) : QDialog(parent), m_reference(ref), m_original_expression(ref->GetExpression()),
m_controller(controller), m_type(type)
{ {
CreateMainLayout(); CreateMainLayout();
@ -233,16 +240,18 @@ void IOWindow::CreateMainLayout()
m_test_button = new QPushButton(tr("Test"), this); m_test_button = new QPushButton(tr("Test"), this);
m_button_box = new QDialogButtonBox(); m_button_box = new QDialogButtonBox();
m_clear_button = new QPushButton(tr("Clear")); m_clear_button = new QPushButton(tr("Clear"));
m_apply_button = new QPushButton(tr("Apply"));
m_range_slider = new QSlider(Qt::Horizontal); m_range_slider = new QSlider(Qt::Horizontal);
m_range_spinbox = new QSpinBox(); m_range_spinbox = new QSpinBox();
m_parse_text = new QLineEdit(); m_parse_text = new InputStateLineEdit([this] {
const auto lock = m_controller->GetStateLock();
return m_reference->GetState<ControlState>();
});
m_parse_text->setReadOnly(true); m_parse_text->setReadOnly(true);
m_expression_text = new QPlainTextEdit(); m_expression_text = new QPlainTextEdit();
m_expression_text->setFont(QFontDatabase::systemFont(QFontDatabase::FixedFont)); m_expression_text->setFont(QFontDatabase::systemFont(QFontDatabase::FixedFont));
new ControlExpressionSyntaxHighlighter(m_expression_text->document(), m_parse_text); new ControlExpressionSyntaxHighlighter(m_expression_text->document());
m_operators_combo = new QComboBox(); m_operators_combo = new QComboBox();
m_operators_combo->addItem(tr("Operators")); m_operators_combo->addItem(tr("Operators"));
@ -315,7 +324,10 @@ void IOWindow::CreateMainLayout()
m_option_list->setColumnWidth(1, 64); m_option_list->setColumnWidth(1, 64);
m_option_list->horizontalHeader()->setSectionResizeMode(1, QHeaderView::Fixed); m_option_list->horizontalHeader()->setSectionResizeMode(1, QHeaderView::Fixed);
m_option_list->setItemDelegate(new InputStateDelegate(this)); 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);
}));
} }
else else
{ {
@ -363,7 +375,6 @@ void IOWindow::CreateMainLayout()
// Button Box // Button Box
m_main_layout->addWidget(m_button_box); m_main_layout->addWidget(m_button_box);
m_button_box->addButton(m_clear_button, QDialogButtonBox::ActionRole); m_button_box->addButton(m_clear_button, QDialogButtonBox::ActionRole);
m_button_box->addButton(m_apply_button, QDialogButtonBox::ActionRole);
m_button_box->addButton(QDialogButtonBox::Ok); m_button_box->addButton(QDialogButtonBox::Ok);
setLayout(m_main_layout); setLayout(m_main_layout);
@ -372,6 +383,10 @@ void IOWindow::CreateMainLayout()
void IOWindow::ConfigChanged() void IOWindow::ConfigChanged()
{ {
const QSignalBlocker blocker(this); const QSignalBlocker blocker(this);
const auto lock = m_controller->GetStateLock();
// ensure m_parse_text is in the right state
UpdateExpression(m_reference->GetExpression(), UpdateMode::Force);
m_expression_text->setPlainText(QString::fromStdString(m_reference->GetExpression())); m_expression_text->setPlainText(QString::fromStdString(m_reference->GetExpression()));
m_expression_text->moveCursor(QTextCursor::End, QTextCursor::MoveAnchor); m_expression_text->moveCursor(QTextCursor::End, QTextCursor::MoveAnchor);
@ -387,6 +402,7 @@ void IOWindow::ConfigChanged()
void IOWindow::Update() void IOWindow::Update()
{ {
m_option_list->viewport()->update(); m_option_list->viewport()->update();
m_parse_text->update();
} }
void IOWindow::ConnectWidgets() void IOWindow::ConnectWidgets()
@ -401,10 +417,8 @@ void IOWindow::ConnectWidgets()
connect(m_range_spinbox, qOverload<int>(&QSpinBox::valueChanged), this, connect(m_range_spinbox, qOverload<int>(&QSpinBox::valueChanged), this,
&IOWindow::OnRangeChanged); &IOWindow::OnRangeChanged);
connect(m_expression_text, &QPlainTextEdit::textChanged, [this] { connect(m_expression_text, &QPlainTextEdit::textChanged,
m_apply_button->setText(m_apply_button->text().remove(QStringLiteral("*"))); [this] { UpdateExpression(m_expression_text->toPlainText().toStdString()); });
m_apply_button->setText(m_apply_button->text() + QStringLiteral("*"));
});
connect(m_operators_combo, qOverload<int>(&QComboBox::activated), [this](int index) { connect(m_operators_combo, qOverload<int>(&QComboBox::activated), [this](int index) {
if (0 == index) if (0 == index)
@ -423,6 +437,9 @@ void IOWindow::ConnectWidgets()
m_functions_combo->setCurrentIndex(0); m_functions_combo->setCurrentIndex(0);
}); });
// revert the expression when the window closes without using the OK button
connect(this, &IOWindow::finished, [this] { UpdateExpression(m_original_expression); });
} }
void IOWindow::AppendSelectedOption() void IOWindow::AppendSelectedOption()
@ -448,18 +465,18 @@ void IOWindow::OnDialogButtonPressed(QAbstractButton* button)
return; return;
} }
m_reference->SetExpression(m_expression_text->toPlainText().toStdString()); const auto lock = m_controller->GetStateLock();
m_controller->UpdateSingleControlReference(g_controller_interface, m_reference);
m_apply_button->setText(m_apply_button->text().remove(QStringLiteral("*"))); UpdateExpression(m_expression_text->toPlainText().toStdString());
m_original_expression = m_reference->GetExpression();
if (ciface::ExpressionParser::ParseStatus::SyntaxError == m_reference->GetParseStatus()) if (ciface::ExpressionParser::ParseStatus::SyntaxError == m_reference->GetParseStatus())
{ {
ModalMessageBox::warning(this, tr("Error"), tr("The expression contains a syntax error.")); ModalMessageBox::warning(this, tr("Error"), tr("The expression contains a syntax error."));
} }
if (button != m_apply_button) // must be the OK button
accept(); accept();
} }
void IOWindow::OnDetectButtonPressed() void IOWindow::OnDetectButtonPressed()
@ -532,8 +549,71 @@ void IOWindow::UpdateDeviceList()
QString::fromStdString(m_controller->GetDefaultDevice().ToString())); QString::fromStdString(m_controller->GetDefaultDevice().ToString()));
} }
InputStateDelegate::InputStateDelegate(IOWindow* parent) : QItemDelegate(parent), m_parent(parent) void IOWindow::UpdateExpression(std::string new_expression, UpdateMode mode)
{ {
const auto lock = m_controller->GetStateLock();
if (mode != UpdateMode::Force && new_expression == m_reference->GetExpression())
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)
{
m_parse_text->SetShouldPaintStateIndicator(false);
m_parse_text->setText(QString::fromStdString(*error));
}
else if (status == ciface::ExpressionParser::ParseStatus::EmptyExpression)
{
m_parse_text->SetShouldPaintStateIndicator(false);
m_parse_text->setText(QString());
}
else if (status != ciface::ExpressionParser::ParseStatus::Successful)
{
m_parse_text->SetShouldPaintStateIndicator(false);
m_parse_text->setText(tr("Invalid Expression."));
}
else
{
m_parse_text->SetShouldPaintStateIndicator(true);
m_parse_text->setText(QString());
}
}
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)
{
}
InputStateLineEdit::InputStateLineEdit(std::function<ControlState()> state_evaluator)
: m_state_evaluator(std::move(state_evaluator))
{
}
static void PaintStateIndicator(QPainter& painter, const QRect& region, ControlState state)
{
const QString state_string = QString::number(state, 'g', 4);
QRect meter_region = region;
meter_region.setWidth(region.width() * std::clamp(state, 0.0, 1.0));
// Create a temporary indicator object to retreive color constants.
MappingIndicator indicator;
// Normal text.
painter.setPen(indicator.GetTextColor());
painter.drawText(region, Qt::AlignCenter, state_string);
// Input state meter.
painter.fillRect(meter_region, indicator.GetAdjustedInputColor());
// Text on top of meter.
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, void InputStateDelegate::paint(QPainter* painter, const QStyleOptionViewItem& option,
@ -541,35 +621,26 @@ void InputStateDelegate::paint(QPainter* painter, const QStyleOptionViewItem& op
{ {
QItemDelegate::paint(painter, option, index); QItemDelegate::paint(painter, option, index);
// Don't do anything special for the first column. if (index.column() != m_column)
if (index.column() == 0)
return; return;
// Clamp off negative values but allow greater than one in the text display.
const auto state =
std::max(m_parent->GetSelectedDevice()->Inputs()[index.row()]->GetState(), 0.0);
const auto state_str = QString::number(state, 'g', 4);
QRect rect = option.rect;
rect.setWidth(rect.width() * std::clamp(state, 0.0, 1.0));
// Create a temporary indicator object to retreive color constants.
MappingIndicator indicator;
painter->save(); painter->save();
PaintStateIndicator(*painter, option.rect, m_state_evaluator(index.row()));
// Normal text.
painter->setPen(indicator.GetTextColor());
painter->drawText(option.rect, Qt::AlignCenter, state_str);
// Input state meter.
painter->fillRect(rect, indicator.GetAdjustedInputColor());
// Text on top of meter.
painter->setPen(indicator.GetAltTextColor());
painter->setClipping(true);
painter->setClipRect(rect);
painter->drawText(option.rect, Qt::AlignCenter, state_str);
painter->restore(); painter->restore();
} }
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());
}

View File

@ -4,6 +4,9 @@
#pragma once #pragma once
#include <memory>
#include <string>
#include <QDialog> #include <QDialog>
#include <QString> #include <QString>
#include <QSyntaxHighlighter> #include <QSyntaxHighlighter>
@ -30,17 +33,16 @@ namespace ControllerEmu
class EmulatedController; class EmulatedController;
} }
class InputStateLineEdit;
class ControlExpressionSyntaxHighlighter final : public QSyntaxHighlighter class ControlExpressionSyntaxHighlighter final : public QSyntaxHighlighter
{ {
Q_OBJECT Q_OBJECT
public: public:
ControlExpressionSyntaxHighlighter(QTextDocument* parent, QLineEdit* result); explicit ControlExpressionSyntaxHighlighter(QTextDocument* parent);
protected: protected:
void highlightBlock(const QString& text) final override; void highlightBlock(const QString& text) final override;
private:
QLineEdit* const m_result_text;
}; };
class IOWindow final : public QDialog class IOWindow final : public QDialog
@ -74,6 +76,14 @@ private:
void UpdateOptionList(); void UpdateOptionList();
void UpdateDeviceList(); void UpdateDeviceList();
enum class UpdateMode
{
Normal,
Force,
};
void UpdateExpression(std::string new_expression, UpdateMode mode = UpdateMode::Normal);
// Main Layout // Main Layout
QVBoxLayout* m_main_layout; QVBoxLayout* m_main_layout;
@ -100,14 +110,14 @@ private:
// Textarea // Textarea
QPlainTextEdit* m_expression_text; QPlainTextEdit* m_expression_text;
QLineEdit* m_parse_text; InputStateLineEdit* m_parse_text;
// Buttonbox // Buttonbox
QDialogButtonBox* m_button_box; QDialogButtonBox* m_button_box;
QPushButton* m_clear_button; QPushButton* m_clear_button;
QPushButton* m_apply_button;
ControlReference* m_reference; ControlReference* m_reference;
std::string m_original_expression;
ControllerEmu::EmulatedController* m_controller; ControllerEmu::EmulatedController* m_controller;
ciface::Core::DeviceQualifier m_devq; ciface::Core::DeviceQualifier m_devq;

View File

@ -50,12 +50,13 @@ std::string ControlReference::GetExpression() const
return m_expression; return m_expression;
} }
void ControlReference::SetExpression(std::string expr) std::optional<std::string> ControlReference::SetExpression(std::string expr)
{ {
m_expression = std::move(expr); m_expression = std::move(expr);
auto parse_result = ParseExpression(m_expression); auto parse_result = ParseExpression(m_expression);
m_parse_status = parse_result.status; m_parse_status = parse_result.status;
m_parsed_expression = std::move(parse_result.expr); m_parsed_expression = std::move(parse_result.expr);
return parse_result.description;
} }
ControlReference::ControlReference() : range(1), m_parsed_expression(nullptr) ControlReference::ControlReference() : range(1), m_parsed_expression(nullptr)

View File

@ -38,7 +38,9 @@ public:
ciface::ExpressionParser::ParseStatus GetParseStatus() const; ciface::ExpressionParser::ParseStatus GetParseStatus() const;
void UpdateReference(ciface::ExpressionParser::ControlEnvironment& env); void UpdateReference(ciface::ExpressionParser::ControlEnvironment& env);
std::string GetExpression() const; std::string GetExpression() const;
void SetExpression(std::string expr);
// Returns a human-readable error description when the given expression is invalid.
std::optional<std::string> SetExpression(std::string expr);
ControlState range; ControlState range;