Merge pull request #7992 from artemist/centering
ControllerEmu: Add support for setting the center of a ReshapableInput
This commit is contained in:
commit
e2c769a9c5
|
@ -64,6 +64,11 @@ QColor MappingIndicator::GetAdjustedInputColor() const
|
||||||
return Qt::red;
|
return Qt::red;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QColor MappingIndicator::GetCenterColor() const
|
||||||
|
{
|
||||||
|
return Qt::blue;
|
||||||
|
}
|
||||||
|
|
||||||
QColor MappingIndicator::GetDeadZoneColor() const
|
QColor MappingIndicator::GetDeadZoneColor() const
|
||||||
{
|
{
|
||||||
return palette().shadow().color();
|
return palette().shadow().color();
|
||||||
|
@ -115,7 +120,8 @@ namespace
|
||||||
{
|
{
|
||||||
// Constructs a polygon by querying a radius at varying angles:
|
// Constructs a polygon by querying a radius at varying angles:
|
||||||
template <typename F>
|
template <typename F>
|
||||||
QPolygonF GetPolygonFromRadiusGetter(F&& radius_getter, double scale)
|
QPolygonF GetPolygonFromRadiusGetter(F&& radius_getter, double scale,
|
||||||
|
Common::DVec2 center = {0.0, 0.0})
|
||||||
{
|
{
|
||||||
// A multiple of 8 (octagon) and enough points to be visibly pleasing:
|
// A multiple of 8 (octagon) and enough points to be visibly pleasing:
|
||||||
constexpr int shape_point_count = 32;
|
constexpr int shape_point_count = 32;
|
||||||
|
@ -127,7 +133,8 @@ QPolygonF GetPolygonFromRadiusGetter(F&& radius_getter, double scale)
|
||||||
const double angle = MathUtil::TAU * p / shape.size();
|
const double angle = MathUtil::TAU * p / shape.size();
|
||||||
const double radius = radius_getter(angle) * scale;
|
const double radius = radius_getter(angle) * scale;
|
||||||
|
|
||||||
point = {std::cos(angle) * radius, std::sin(angle) * radius};
|
point = {std::cos(angle) * radius + center.x * scale,
|
||||||
|
std::sin(angle) * radius + center.y * scale};
|
||||||
++p;
|
++p;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,9 +174,10 @@ bool IsCalibrationDataSensible(const ControllerEmu::ReshapableInput::Calibration
|
||||||
// Used to test for a miscalibrated stick so the user can be informed.
|
// Used to test for a miscalibrated stick so the user can be informed.
|
||||||
bool IsPointOutsideCalibration(Common::DVec2 point, ControllerEmu::ReshapableInput& input)
|
bool IsPointOutsideCalibration(Common::DVec2 point, ControllerEmu::ReshapableInput& input)
|
||||||
{
|
{
|
||||||
const double current_radius = point.Length();
|
const auto center = input.GetCenter();
|
||||||
const double input_radius =
|
const double current_radius = (point - center).Length();
|
||||||
input.GetInputRadiusAtAngle(std::atan2(point.y, point.x) + MathUtil::TAU);
|
const double input_radius = input.GetInputRadiusAtAngle(
|
||||||
|
std::atan2(point.y - center.y, point.x - center.x) + MathUtil::TAU);
|
||||||
|
|
||||||
constexpr double ALLOWED_ERROR = 1.3;
|
constexpr double ALLOWED_ERROR = 1.3;
|
||||||
|
|
||||||
|
@ -180,6 +188,8 @@ bool IsPointOutsideCalibration(Common::DVec2 point, ControllerEmu::ReshapableInp
|
||||||
|
|
||||||
void MappingIndicator::DrawCursor(ControllerEmu::Cursor& cursor)
|
void MappingIndicator::DrawCursor(ControllerEmu::Cursor& cursor)
|
||||||
{
|
{
|
||||||
|
const auto center = cursor.GetCenter();
|
||||||
|
|
||||||
const QColor tv_brush_color = CURSOR_TV_COLOR;
|
const QColor tv_brush_color = CURSOR_TV_COLOR;
|
||||||
const QColor tv_pen_color = tv_brush_color.darker(125);
|
const QColor tv_pen_color = tv_brush_color.darker(125);
|
||||||
|
|
||||||
|
@ -224,13 +234,21 @@ void MappingIndicator::DrawCursor(ControllerEmu::Cursor& cursor)
|
||||||
p.setPen(GetDeadZonePen());
|
p.setPen(GetDeadZonePen());
|
||||||
p.setBrush(GetDeadZoneBrush());
|
p.setBrush(GetDeadZoneBrush());
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[&cursor](double ang) { return cursor.GetDeadzoneRadiusAtAngle(ang); }, scale));
|
[&cursor](double ang) { return cursor.GetDeadzoneRadiusAtAngle(ang); }, scale, center));
|
||||||
|
|
||||||
// Input shape.
|
// Input shape.
|
||||||
p.setPen(GetInputShapePen());
|
p.setPen(GetInputShapePen());
|
||||||
p.setBrush(Qt::NoBrush);
|
p.setBrush(Qt::NoBrush);
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[&cursor](double ang) { return cursor.GetInputRadiusAtAngle(ang); }, scale));
|
[&cursor](double ang) { return cursor.GetInputRadiusAtAngle(ang); }, scale, center));
|
||||||
|
|
||||||
|
// Center.
|
||||||
|
if (center.x || center.y)
|
||||||
|
{
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
p.setBrush(GetCenterColor());
|
||||||
|
p.drawEllipse(QPointF{center.x, center.y} * scale, INPUT_DOT_RADIUS, INPUT_DOT_RADIUS);
|
||||||
|
}
|
||||||
|
|
||||||
// Raw stick position.
|
// Raw stick position.
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
|
@ -253,6 +271,8 @@ void MappingIndicator::DrawReshapableInput(ControllerEmu::ReshapableInput& stick
|
||||||
const bool is_c_stick = m_group->name == "C-Stick";
|
const bool is_c_stick = m_group->name == "C-Stick";
|
||||||
const bool is_tilt = m_group->name == "Tilt";
|
const bool is_tilt = m_group->name == "Tilt";
|
||||||
|
|
||||||
|
const auto center = stick.GetCenter();
|
||||||
|
|
||||||
QColor gate_brush_color = GetGateColor();
|
QColor gate_brush_color = GetGateColor();
|
||||||
|
|
||||||
if (is_c_stick)
|
if (is_c_stick)
|
||||||
|
@ -312,13 +332,21 @@ void MappingIndicator::DrawReshapableInput(ControllerEmu::ReshapableInput& stick
|
||||||
p.setPen(GetDeadZonePen());
|
p.setPen(GetDeadZonePen());
|
||||||
p.setBrush(GetDeadZoneBrush());
|
p.setBrush(GetDeadZoneBrush());
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[&stick](double ang) { return stick.GetDeadzoneRadiusAtAngle(ang); }, scale));
|
[&stick](double ang) { return stick.GetDeadzoneRadiusAtAngle(ang); }, scale, center));
|
||||||
|
|
||||||
// Input shape.
|
// Input shape.
|
||||||
p.setPen(GetInputShapePen());
|
p.setPen(GetInputShapePen());
|
||||||
p.setBrush(Qt::NoBrush);
|
p.setBrush(Qt::NoBrush);
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[&stick](double ang) { return stick.GetInputRadiusAtAngle(ang); }, scale));
|
[&stick](double ang) { return stick.GetInputRadiusAtAngle(ang); }, scale, center));
|
||||||
|
|
||||||
|
// Center.
|
||||||
|
if (center.x || center.y)
|
||||||
|
{
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
p.setBrush(GetCenterColor());
|
||||||
|
p.drawEllipse(QPointF{center.x, center.y} * scale, INPUT_DOT_RADIUS, INPUT_DOT_RADIUS);
|
||||||
|
}
|
||||||
|
|
||||||
// Raw stick position.
|
// Raw stick position.
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
|
@ -431,6 +459,8 @@ void MappingIndicator::DrawMixedTriggers()
|
||||||
|
|
||||||
void MappingIndicator::DrawForce(ControllerEmu::Force& force)
|
void MappingIndicator::DrawForce(ControllerEmu::Force& force)
|
||||||
{
|
{
|
||||||
|
const auto center = force.GetCenter();
|
||||||
|
|
||||||
const QColor gate_brush_color = SWING_GATE_COLOR;
|
const QColor gate_brush_color = SWING_GATE_COLOR;
|
||||||
const QColor gate_pen_color = gate_brush_color.darker(125);
|
const QColor gate_pen_color = gate_brush_color.darker(125);
|
||||||
|
|
||||||
|
@ -505,13 +535,21 @@ void MappingIndicator::DrawForce(ControllerEmu::Force& force)
|
||||||
p.setPen(GetDeadZoneColor());
|
p.setPen(GetDeadZoneColor());
|
||||||
p.setBrush(GetDeadZoneBrush());
|
p.setBrush(GetDeadZoneBrush());
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[&force](double ang) { return force.GetDeadzoneRadiusAtAngle(ang); }, scale));
|
[&force](double ang) { return force.GetDeadzoneRadiusAtAngle(ang); }, scale, center));
|
||||||
|
|
||||||
// Input shape.
|
// Input shape.
|
||||||
p.setPen(GetInputShapePen());
|
p.setPen(GetInputShapePen());
|
||||||
p.setBrush(Qt::NoBrush);
|
p.setBrush(Qt::NoBrush);
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[&force](double ang) { return force.GetInputRadiusAtAngle(ang); }, scale));
|
[&force](double ang) { return force.GetInputRadiusAtAngle(ang); }, scale, center));
|
||||||
|
|
||||||
|
// Center.
|
||||||
|
if (center.x || center.y)
|
||||||
|
{
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
p.setBrush(GetCenterColor());
|
||||||
|
p.drawEllipse(QPointF{center.x, center.y} * scale, INPUT_DOT_RADIUS, INPUT_DOT_RADIUS);
|
||||||
|
}
|
||||||
|
|
||||||
// Raw stick position.
|
// Raw stick position.
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
|
@ -647,13 +685,22 @@ void MappingIndicator::DrawCalibration(QPainter& p, Common::DVec2 point)
|
||||||
{
|
{
|
||||||
// Bounding box size:
|
// Bounding box size:
|
||||||
const double scale = GetScale();
|
const double scale = GetScale();
|
||||||
|
const auto center = m_calibration_widget->GetCenter();
|
||||||
|
|
||||||
// Input shape.
|
// Input shape.
|
||||||
p.setPen(GetInputShapePen());
|
p.setPen(GetInputShapePen());
|
||||||
p.setBrush(Qt::NoBrush);
|
p.setBrush(Qt::NoBrush);
|
||||||
p.drawPolygon(GetPolygonFromRadiusGetter(
|
p.drawPolygon(GetPolygonFromRadiusGetter(
|
||||||
[this](double angle) { return m_calibration_widget->GetCalibrationRadiusAtAngle(angle); },
|
[this](double angle) { return m_calibration_widget->GetCalibrationRadiusAtAngle(angle); },
|
||||||
scale));
|
scale, center));
|
||||||
|
|
||||||
|
// Center.
|
||||||
|
if (center.x || center.y)
|
||||||
|
{
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
p.setBrush(GetCenterColor());
|
||||||
|
p.drawEllipse(QPointF{center.x, center.y} * scale, INPUT_DOT_RADIUS, INPUT_DOT_RADIUS);
|
||||||
|
}
|
||||||
|
|
||||||
// Stick position.
|
// Stick position.
|
||||||
p.setPen(Qt::NoPen);
|
p.setPen(Qt::NoPen);
|
||||||
|
@ -706,20 +753,33 @@ CalibrationWidget::CalibrationWidget(ControllerEmu::ReshapableInput& input,
|
||||||
void CalibrationWidget::SetupActions()
|
void CalibrationWidget::SetupActions()
|
||||||
{
|
{
|
||||||
const auto calibrate_action = new QAction(tr("Calibrate"), this);
|
const auto calibrate_action = new QAction(tr("Calibrate"), this);
|
||||||
|
const auto center_action = new QAction(tr("Center and Calibrate"), this);
|
||||||
const auto reset_action = new QAction(tr("Reset"), this);
|
const auto reset_action = new QAction(tr("Reset"), this);
|
||||||
|
|
||||||
connect(calibrate_action, &QAction::triggered, [this]() { StartCalibration(); });
|
connect(calibrate_action, &QAction::triggered, [this]() {
|
||||||
connect(reset_action, &QAction::triggered, [this]() { m_input.SetCalibrationToDefault(); });
|
StartCalibration();
|
||||||
|
m_input.SetCenter({0, 0});
|
||||||
|
});
|
||||||
|
connect(center_action, &QAction::triggered, [this]() {
|
||||||
|
StartCalibration();
|
||||||
|
m_is_centering = true;
|
||||||
|
});
|
||||||
|
connect(reset_action, &QAction::triggered, [this]() {
|
||||||
|
m_input.SetCalibrationToDefault();
|
||||||
|
m_input.SetCenter({0, 0});
|
||||||
|
});
|
||||||
|
|
||||||
for (auto* action : actions())
|
for (auto* action : actions())
|
||||||
removeAction(action);
|
removeAction(action);
|
||||||
|
|
||||||
addAction(calibrate_action);
|
addAction(calibrate_action);
|
||||||
|
addAction(center_action);
|
||||||
addAction(reset_action);
|
addAction(reset_action);
|
||||||
setDefaultAction(calibrate_action);
|
setDefaultAction(calibrate_action);
|
||||||
|
|
||||||
m_completion_action = new QAction(tr("Finish Calibration"), this);
|
m_completion_action = new QAction(tr("Finish Calibration"), this);
|
||||||
connect(m_completion_action, &QAction::triggered, [this]() {
|
connect(m_completion_action, &QAction::triggered, [this]() {
|
||||||
|
m_input.SetCenter(m_new_center);
|
||||||
m_input.SetCalibrationData(std::move(m_calibration_data));
|
m_input.SetCalibrationData(std::move(m_calibration_data));
|
||||||
m_informative_timer->stop();
|
m_informative_timer->stop();
|
||||||
SetupActions();
|
SetupActions();
|
||||||
|
@ -730,6 +790,8 @@ void CalibrationWidget::StartCalibration()
|
||||||
{
|
{
|
||||||
m_calibration_data.assign(m_input.CALIBRATION_SAMPLE_COUNT, 0.0);
|
m_calibration_data.assign(m_input.CALIBRATION_SAMPLE_COUNT, 0.0);
|
||||||
|
|
||||||
|
m_new_center = {0, 0};
|
||||||
|
|
||||||
// Cancel calibration.
|
// Cancel calibration.
|
||||||
const auto cancel_action = new QAction(tr("Cancel Calibration"), this);
|
const auto cancel_action = new QAction(tr("Cancel Calibration"), this);
|
||||||
connect(cancel_action, &QAction::triggered, [this]() {
|
connect(cancel_action, &QAction::triggered, [this]() {
|
||||||
|
@ -754,9 +816,14 @@ void CalibrationWidget::Update(Common::DVec2 point)
|
||||||
QFont f = parentWidget()->font();
|
QFont f = parentWidget()->font();
|
||||||
QPalette p = parentWidget()->palette();
|
QPalette p = parentWidget()->palette();
|
||||||
|
|
||||||
if (IsCalibrating())
|
if (m_is_centering)
|
||||||
{
|
{
|
||||||
m_input.UpdateCalibrationData(m_calibration_data, point);
|
m_new_center = point;
|
||||||
|
m_is_centering = false;
|
||||||
|
}
|
||||||
|
else if (IsCalibrating())
|
||||||
|
{
|
||||||
|
m_input.UpdateCalibrationData(m_calibration_data, point - m_new_center);
|
||||||
|
|
||||||
if (IsCalibrationDataSensible(m_calibration_data))
|
if (IsCalibrationDataSensible(m_calibration_data))
|
||||||
{
|
{
|
||||||
|
@ -786,3 +853,8 @@ double CalibrationWidget::GetCalibrationRadiusAtAngle(double angle) const
|
||||||
{
|
{
|
||||||
return m_input.GetCalibrationDataRadiusAtAngle(m_calibration_data, angle);
|
return m_input.GetCalibrationDataRadiusAtAngle(m_calibration_data, angle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Common::DVec2 CalibrationWidget::GetCenter() const
|
||||||
|
{
|
||||||
|
return m_new_center;
|
||||||
|
}
|
||||||
|
|
|
@ -40,6 +40,7 @@ protected:
|
||||||
QBrush GetBBoxBrush() const;
|
QBrush GetBBoxBrush() const;
|
||||||
QColor GetRawInputColor() const;
|
QColor GetRawInputColor() const;
|
||||||
QPen GetInputShapePen() const;
|
QPen GetInputShapePen() const;
|
||||||
|
QColor GetCenterColor() const;
|
||||||
QColor GetAdjustedInputColor() const;
|
QColor GetAdjustedInputColor() const;
|
||||||
QColor GetDeadZoneColor() const;
|
QColor GetDeadZoneColor() const;
|
||||||
QPen GetDeadZonePen() const;
|
QPen GetDeadZonePen() const;
|
||||||
|
@ -90,6 +91,8 @@ public:
|
||||||
|
|
||||||
double GetCalibrationRadiusAtAngle(double angle) const;
|
double GetCalibrationRadiusAtAngle(double angle) const;
|
||||||
|
|
||||||
|
Common::DVec2 GetCenter() const;
|
||||||
|
|
||||||
bool IsCalibrating() const;
|
bool IsCalibrating() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -101,4 +104,7 @@ private:
|
||||||
QAction* m_completion_action;
|
QAction* m_completion_action;
|
||||||
ControllerEmu::ReshapableInput::CalibrationData m_calibration_data;
|
ControllerEmu::ReshapableInput::CalibrationData m_calibration_data;
|
||||||
QTimer* m_informative_timer;
|
QTimer* m_informative_timer;
|
||||||
|
|
||||||
|
bool m_is_centering = false;
|
||||||
|
Common::DVec2 m_new_center;
|
||||||
};
|
};
|
||||||
|
|
|
@ -20,6 +20,9 @@ constexpr auto CALIBRATION_CONFIG_NAME = "Calibration";
|
||||||
constexpr auto CALIBRATION_DEFAULT_VALUE = 1.0;
|
constexpr auto CALIBRATION_DEFAULT_VALUE = 1.0;
|
||||||
constexpr auto CALIBRATION_CONFIG_SCALE = 100;
|
constexpr auto CALIBRATION_CONFIG_SCALE = 100;
|
||||||
|
|
||||||
|
constexpr auto CENTER_CONFIG_NAME = "Center";
|
||||||
|
constexpr auto CENTER_CONFIG_SCALE = 100;
|
||||||
|
|
||||||
// Calculate distance to intersection of a ray with a line defined by two points.
|
// Calculate distance to intersection of a ray with a line defined by two points.
|
||||||
double GetRayLineIntersection(Common::DVec2 ray, Common::DVec2 point1, Common::DVec2 point2)
|
double GetRayLineIntersection(Common::DVec2 ray, Common::DVec2 point1, Common::DVec2 point2)
|
||||||
{
|
{
|
||||||
|
@ -214,6 +217,16 @@ void ReshapableInput::SetCalibrationData(CalibrationData data)
|
||||||
m_calibration = std::move(data);
|
m_calibration = std::move(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const ReshapableInput::ReshapeData& ReshapableInput::GetCenter() const
|
||||||
|
{
|
||||||
|
return m_center;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReshapableInput::SetCenter(ReshapableInput::ReshapeData center)
|
||||||
|
{
|
||||||
|
m_center = center;
|
||||||
|
}
|
||||||
|
|
||||||
void ReshapableInput::LoadConfig(IniFile::Section* section, const std::string& default_device,
|
void ReshapableInput::LoadConfig(IniFile::Section* section, const std::string& default_device,
|
||||||
const std::string& base_name)
|
const std::string& base_name)
|
||||||
{
|
{
|
||||||
|
@ -232,6 +245,20 @@ void ReshapableInput::LoadConfig(IniFile::Section* section, const std::string& d
|
||||||
if (TryParse(*(it++), &sample))
|
if (TryParse(*(it++), &sample))
|
||||||
sample /= CALIBRATION_CONFIG_SCALE;
|
sample /= CALIBRATION_CONFIG_SCALE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
section->Get(group + CENTER_CONFIG_NAME, &load_str, "");
|
||||||
|
const auto center_data = SplitString(load_str, ' ');
|
||||||
|
|
||||||
|
m_center = Common::DVec2();
|
||||||
|
|
||||||
|
if (center_data.size() == 2)
|
||||||
|
{
|
||||||
|
if (TryParse(center_data[0], &m_center.x))
|
||||||
|
m_center.x /= CENTER_CONFIG_SCALE;
|
||||||
|
|
||||||
|
if (TryParse(center_data[1], &m_center.y))
|
||||||
|
m_center.y /= CENTER_CONFIG_SCALE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ReshapableInput::SaveConfig(IniFile::Section* section, const std::string& default_device,
|
void ReshapableInput::SaveConfig(IniFile::Section* section, const std::string& default_device,
|
||||||
|
@ -245,11 +272,19 @@ void ReshapableInput::SaveConfig(IniFile::Section* section, const std::string& d
|
||||||
m_calibration.begin(), m_calibration.end(), save_data.begin(),
|
m_calibration.begin(), m_calibration.end(), save_data.begin(),
|
||||||
[](ControlState val) { return StringFromFormat("%.2f", val * CALIBRATION_CONFIG_SCALE); });
|
[](ControlState val) { return StringFromFormat("%.2f", val * CALIBRATION_CONFIG_SCALE); });
|
||||||
section->Set(group + CALIBRATION_CONFIG_NAME, JoinStrings(save_data, " "), "");
|
section->Set(group + CALIBRATION_CONFIG_NAME, JoinStrings(save_data, " "), "");
|
||||||
|
|
||||||
|
const auto center_data = StringFromFormat("%.2f %.2f", m_center.x * CENTER_CONFIG_SCALE,
|
||||||
|
m_center.y * CENTER_CONFIG_SCALE);
|
||||||
|
|
||||||
|
section->Set(group + CENTER_CONFIG_NAME, center_data, "");
|
||||||
}
|
}
|
||||||
|
|
||||||
ReshapableInput::ReshapeData ReshapableInput::Reshape(ControlState x, ControlState y,
|
ReshapableInput::ReshapeData ReshapableInput::Reshape(ControlState x, ControlState y,
|
||||||
ControlState modifier)
|
ControlState modifier)
|
||||||
{
|
{
|
||||||
|
x -= m_center.x;
|
||||||
|
y -= m_center.y;
|
||||||
|
|
||||||
// TODO: make the AtAngle functions work with negative angles:
|
// TODO: make the AtAngle functions work with negative angles:
|
||||||
const ControlState angle = std::atan2(y, x) + MathUtil::TAU;
|
const ControlState angle = std::atan2(y, x) + MathUtil::TAU;
|
||||||
|
|
||||||
|
|
|
@ -97,6 +97,9 @@ public:
|
||||||
const CalibrationData& GetCalibrationData() const;
|
const CalibrationData& GetCalibrationData() const;
|
||||||
void SetCalibrationData(CalibrationData data);
|
void SetCalibrationData(CalibrationData data);
|
||||||
|
|
||||||
|
const ReshapeData& GetCenter() const;
|
||||||
|
void SetCenter(ReshapeData center);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
ReshapeData Reshape(ControlState x, ControlState y, ControlState modifier = 0.0);
|
ReshapeData Reshape(ControlState x, ControlState y, ControlState modifier = 0.0);
|
||||||
|
|
||||||
|
@ -106,6 +109,7 @@ private:
|
||||||
|
|
||||||
CalibrationData m_calibration;
|
CalibrationData m_calibration;
|
||||||
SettingValue<double> m_deadzone_setting;
|
SettingValue<double> m_deadzone_setting;
|
||||||
|
ReshapeData m_center;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace ControllerEmu
|
} // namespace ControllerEmu
|
||||||
|
|
Loading…
Reference in New Issue