ControllerEmu: Always convert to polar coordinates and back

Most users will have something in the radius or deadzone fields, so
don't bother filtering out 'extra' work. This also lets us clean up
the modifier implementation.
This commit is contained in:
Jasper St. Pierre 2014-07-11 08:09:39 -04:00
parent f2d4f10fc9
commit c29d5ff989
1 changed files with 29 additions and 35 deletions

View File

@ -140,24 +140,21 @@ public:
xx = (fabsf(xx)>deadzone) * sign(xx) * (m + deadzone/2); xx = (fabsf(xx)>deadzone) * sign(xx) * (m + deadzone/2);
} }
if (radius != 1 || deadzone) ControlState ang = atan2(yy, xx);
{ ControlState ang_sin = sin(ang);
ControlState ang = atan2(yy, xx); ControlState ang_cos = cos(ang);
ControlState ang_sin = sin(ang);
ControlState ang_cos = cos(ang);
ControlState dist = sqrt(xx*xx + yy*yy); ControlState dist = sqrt(xx*xx + yy*yy);
// dead zone code // dead zone code
dist = std::max(0.0f, dist - deadzone); dist = std::max(0.0f, dist - deadzone);
dist /= (1 - deadzone); dist /= (1 - deadzone);
// radius // radius
dist *= radius; dist *= radius;
yy = std::max(-1.0f, std::min(1.0f, ang_sin * dist)); yy = std::max(-1.0f, std::min(1.0f, ang_sin * dist));
xx = std::max(-1.0f, std::min(1.0f, ang_cos * dist)); xx = std::max(-1.0f, std::min(1.0f, ang_cos * dist));
}
*y = C(yy * range + base); *y = C(yy * range + base);
*x = C(xx * range + base); *x = C(xx * range + base);
@ -292,34 +289,31 @@ public:
} }
// deadzone / circle stick code // deadzone / circle stick code
if (deadzone || circle) // this section might be all wrong, but its working good enough, I think
{
// this section might be all wrong, but its working good enough, I think
ControlState ang = atan2(yy, xx); ControlState ang = atan2(yy, xx);
ControlState ang_sin = sin(ang); ControlState ang_sin = sin(ang);
ControlState ang_cos = cos(ang); ControlState ang_cos = cos(ang);
// the amt a full square stick would have at current angle // the amt a full square stick would have at current angle
ControlState square_full = std::min(ang_sin ? 1/fabsf(ang_sin) : 2, ang_cos ? 1/fabsf(ang_cos) : 2); ControlState square_full = std::min(ang_sin ? 1/fabsf(ang_sin) : 2, ang_cos ? 1/fabsf(ang_cos) : 2);
// the amt a full stick would have that was (user setting circular) at current angle // the amt a full stick would have that was (user setting circular) at current angle
// I think this is more like a pointed circle rather than a rounded square like it should be // I think this is more like a pointed circle rather than a rounded square like it should be
ControlState stick_full = (square_full * (1 - circle)) + (circle); ControlState stick_full = (square_full * (1 - circle)) + (circle);
ControlState dist = sqrt(xx*xx + yy*yy); ControlState dist = sqrt(xx*xx + yy*yy);
// dead zone code // dead zone code
dist = std::max(0.0f, dist - deadzone * stick_full); dist = std::max(0.0f, dist - deadzone * stick_full);
dist /= (1 - deadzone); dist /= (1 - deadzone);
// circle stick code // circle stick code
ControlState amt = dist / stick_full; ControlState amt = dist / stick_full;
dist += (square_full - 1) * amt * circle; dist += (square_full - 1) * amt * circle;
yy = std::max(-1.0f, std::min(1.0f, ang_sin * dist)); yy = std::max(-1.0f, std::min(1.0f, ang_sin * dist));
xx = std::max(-1.0f, std::min(1.0f, ang_cos * dist)); xx = std::max(-1.0f, std::min(1.0f, ang_cos * dist));
}
// this is kinda silly here // this is kinda silly here
// gui being open will make this happen 2x as fast, o well // gui being open will make this happen 2x as fast, o well