mirror of
https://github.com/cemu-project/DS4Windows.git
synced 2025-01-25 06:21:17 +01:00
544 lines
22 KiB
C#
544 lines
22 KiB
C#
using System;
|
|
|
|
namespace DS4Windows
|
|
{
|
|
public class Mouse : ITouchpadBehaviour
|
|
{
|
|
protected DateTime pastTime, firstTap, TimeofEnd;
|
|
protected Touch firstTouch, secondTouch;
|
|
private DS4State s = new DS4State();
|
|
protected int deviceNum;
|
|
private DS4Device dev = null;
|
|
private readonly MouseCursor cursor;
|
|
private readonly MouseWheel wheel;
|
|
private bool tappedOnce = false, secondtouchbegin = false;
|
|
public bool swipeLeft, swipeRight, swipeUp, swipeDown;
|
|
public bool priorSwipeLeft, priorSwipeRight, priorSwipeUp, priorSwipeDown;
|
|
public byte swipeLeftB, swipeRightB, swipeUpB, swipeDownB, swipedB;
|
|
public byte priorSwipeLeftB, priorSwipeRightB, priorSwipeUpB, priorSwipeDownB, priorSwipedB;
|
|
public bool slideleft, slideright;
|
|
public bool priorSlideLeft, priorSlideright;
|
|
// touch area stuff
|
|
public bool leftDown, rightDown, upperDown, multiDown;
|
|
public bool priorLeftDown, priorRightDown, priorUpperDown, priorMultiDown;
|
|
protected DS4Controls pushed = DS4Controls.None;
|
|
protected Mapping.Click clicked = Mapping.Click.None;
|
|
public int CursorGyroDead { get => cursor.GyroCursorDeadZone; set => cursor.GyroCursorDeadZone = value; }
|
|
|
|
internal const int TRACKBALL_INIT_FICTION = 10;
|
|
internal const int TRACKBALL_MASS = 45;
|
|
internal const double TRACKBALL_RADIUS = 0.0245;
|
|
|
|
private double TRACKBALL_INERTIA = 2.0 * (TRACKBALL_MASS * TRACKBALL_RADIUS * TRACKBALL_RADIUS) / 5.0;
|
|
private double TRACKBALL_SCALE = 0.004;
|
|
private const int TRACKBALL_BUFFER_LEN = 8;
|
|
private double[] trackballXBuffer = new double[TRACKBALL_BUFFER_LEN];
|
|
private double[] trackballYBuffer = new double[TRACKBALL_BUFFER_LEN];
|
|
private int trackballBufferTail = 0;
|
|
private int trackballBufferHead = 0;
|
|
private double trackballAccel = 0.0;
|
|
private double trackballXVel = 0.0;
|
|
private double trackballYVel = 0.0;
|
|
private bool trackballActive = false;
|
|
private double trackballDXRemain = 0.0;
|
|
private double trackballDYRemain = 0.0;
|
|
|
|
public Mouse(int deviceID, DS4Device d)
|
|
{
|
|
deviceNum = deviceID;
|
|
dev = d;
|
|
cursor = new MouseCursor(deviceNum);
|
|
wheel = new MouseWheel(deviceNum);
|
|
trackballAccel = TRACKBALL_RADIUS * TRACKBALL_INIT_FICTION / TRACKBALL_INERTIA;
|
|
firstTouch = new Touch(0, 0, 0, null);
|
|
}
|
|
|
|
public void ResetTrackAccel(double friction)
|
|
{
|
|
trackballAccel = TRACKBALL_RADIUS * friction / TRACKBALL_INERTIA;
|
|
}
|
|
|
|
public void ResetToggleGyroM()
|
|
{
|
|
currentToggleGyroM = false;
|
|
}
|
|
|
|
bool triggeractivated = false;
|
|
bool previousTriggerActivated = false;
|
|
bool useReverseRatchet = false;
|
|
bool toggleGyroMouse = true;
|
|
public bool ToggleGyroMouse { get => toggleGyroMouse;
|
|
set { toggleGyroMouse = value; ResetToggleGyroM(); } }
|
|
bool currentToggleGyroM = false;
|
|
|
|
public virtual void sixaxisMoved(DS4SixAxis sender, SixAxisEventArgs arg)
|
|
{
|
|
if (Global.isUsingSAforMouse(deviceNum) && Global.getGyroSensitivity(deviceNum) > 0)
|
|
{
|
|
s = dev.getCurrentStateRef();
|
|
|
|
useReverseRatchet = Global.getGyroTriggerTurns(deviceNum);
|
|
int i = 0;
|
|
string[] ss = Global.getSATriggers(deviceNum).Split(',');
|
|
bool andCond = Global.getSATriggerCond(deviceNum);
|
|
triggeractivated = andCond ? true : false;
|
|
if (!string.IsNullOrEmpty(ss[0]))
|
|
{
|
|
string s = string.Empty;
|
|
for (int index = 0, arlen = ss.Length; index < arlen; index++)
|
|
{
|
|
s = ss[index];
|
|
if (andCond && !(int.TryParse(s, out i) && getDS4ControlsByName(i)))
|
|
{
|
|
triggeractivated = false;
|
|
break;
|
|
}
|
|
else if (!andCond && int.TryParse(s, out i) && getDS4ControlsByName(i))
|
|
{
|
|
triggeractivated = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (toggleGyroMouse)
|
|
{
|
|
if (triggeractivated && triggeractivated != previousTriggerActivated)
|
|
{
|
|
currentToggleGyroM = !currentToggleGyroM;
|
|
}
|
|
|
|
previousTriggerActivated = triggeractivated;
|
|
triggeractivated = currentToggleGyroM;
|
|
}
|
|
else
|
|
{
|
|
previousTriggerActivated = triggeractivated;
|
|
}
|
|
|
|
if (useReverseRatchet && triggeractivated)
|
|
cursor.sixaxisMoved(arg);
|
|
else if (!useReverseRatchet && !triggeractivated)
|
|
cursor.sixaxisMoved(arg);
|
|
else
|
|
cursor.mouseRemainderReset();
|
|
}
|
|
}
|
|
|
|
private bool getDS4ControlsByName(int key)
|
|
{
|
|
switch (key)
|
|
{
|
|
case -1: return true;
|
|
case 0: return s.Cross;
|
|
case 1: return s.Circle;
|
|
case 2: return s.Square;
|
|
case 3: return s.Triangle;
|
|
case 4: return s.L1;
|
|
case 5: return s.L2 > 128;
|
|
case 6: return s.R1;
|
|
case 7: return s.R2 > 128;
|
|
case 8: return s.DpadUp;
|
|
case 9: return s.DpadDown;
|
|
case 10: return s.DpadLeft;
|
|
case 11: return s.DpadRight;
|
|
case 12: return s.L3;
|
|
case 13: return s.R3;
|
|
case 14: return s.Touch1Finger;
|
|
case 15: return s.Touch2Fingers;
|
|
case 16: return s.Options;
|
|
case 17: return s.Share;
|
|
case 18: return s.PS;
|
|
default: break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private bool tempBool = false;
|
|
public virtual void touchesMoved(DS4Touchpad sender, TouchpadEventArgs arg)
|
|
{
|
|
s = dev.getCurrentStateRef();
|
|
|
|
if (Global.getUseTPforControls(deviceNum) == false)
|
|
{
|
|
int[] disArray = Global.getTouchDisInvertTriggers(deviceNum);
|
|
tempBool = true;
|
|
for (int i = 0, arlen = disArray.Length; tempBool && i < arlen; i++)
|
|
{
|
|
if (getDS4ControlsByName(disArray[i]) == false)
|
|
tempBool = false;
|
|
}
|
|
|
|
if (Global.getTrackballMode(deviceNum))
|
|
{
|
|
int iIndex = trackballBufferTail;
|
|
trackballXBuffer[iIndex] = (arg.touches[0].deltaX * TRACKBALL_SCALE) / dev.getCurrentStateRef().elapsedTime;
|
|
trackballYBuffer[iIndex] = (arg.touches[0].deltaY * TRACKBALL_SCALE) / dev.getCurrentStateRef().elapsedTime;
|
|
trackballBufferTail = (iIndex + 1) % TRACKBALL_BUFFER_LEN;
|
|
if (trackballBufferHead == trackballBufferTail)
|
|
trackballBufferHead = (trackballBufferHead + 1) % TRACKBALL_BUFFER_LEN;
|
|
}
|
|
|
|
cursor.touchesMoved(arg, dragging || dragging2, tempBool);
|
|
wheel.touchesMoved(arg, dragging || dragging2);
|
|
}
|
|
else
|
|
{
|
|
if (!(swipeUp || swipeDown || swipeLeft || swipeRight) && arg.touches.Length == 1)
|
|
{
|
|
if (arg.touches[0].hwX - firstTouch.hwX > 400) swipeRight = true;
|
|
if (arg.touches[0].hwX - firstTouch.hwX < -400) swipeLeft = true;
|
|
if (arg.touches[0].hwY - firstTouch.hwY > 300) swipeDown = true;
|
|
if (arg.touches[0].hwY - firstTouch.hwY < -300) swipeUp = true;
|
|
}
|
|
|
|
swipeUpB = (byte)Math.Min(255, Math.Max(0, (firstTouch.hwY - arg.touches[0].hwY) * 1.5f));
|
|
swipeDownB = (byte)Math.Min(255, Math.Max(0, (arg.touches[0].hwY - firstTouch.hwY) * 1.5f));
|
|
swipeLeftB = (byte)Math.Min(255, Math.Max(0, firstTouch.hwX - arg.touches[0].hwX));
|
|
swipeRightB = (byte)Math.Min(255, Math.Max(0, arg.touches[0].hwX - firstTouch.hwX));
|
|
}
|
|
|
|
if (Math.Abs(firstTouch.hwY - arg.touches[0].hwY) < 50 && arg.touches.Length == 2)
|
|
{
|
|
if (arg.touches[0].hwX - firstTouch.hwX > 200 && !slideleft)
|
|
slideright = true;
|
|
else if (firstTouch.hwX - arg.touches[0].hwX > 200 && !slideright)
|
|
slideleft = true;
|
|
}
|
|
|
|
synthesizeMouseButtons();
|
|
}
|
|
|
|
public virtual void touchesBegan(DS4Touchpad sender, TouchpadEventArgs arg)
|
|
{
|
|
if (!Global.UseTPforControls[deviceNum])
|
|
{
|
|
Array.Clear(trackballXBuffer, 0, TRACKBALL_BUFFER_LEN);
|
|
Array.Clear(trackballXBuffer, 0, TRACKBALL_BUFFER_LEN);
|
|
trackballXVel = 0.0;
|
|
trackballYVel = 0.0;
|
|
trackballActive = false;
|
|
trackballBufferTail = 0;
|
|
trackballBufferHead = 0;
|
|
trackballDXRemain = 0.0;
|
|
trackballDYRemain = 0.0;
|
|
|
|
cursor.touchesBegan(arg);
|
|
wheel.touchesBegan(arg);
|
|
}
|
|
|
|
pastTime = arg.timeStamp;
|
|
firstTouch.populate(arg.touches[0].hwX, arg.touches[0].hwY, arg.touches[0].touchID,
|
|
arg.touches[0].previousTouch);
|
|
|
|
if (Global.getDoubleTap(deviceNum))
|
|
{
|
|
DateTime test = arg.timeStamp;
|
|
if (test <= (firstTap + TimeSpan.FromMilliseconds((double)Global.TapSensitivity[deviceNum] * 1.5)) && !arg.touchButtonPressed)
|
|
secondtouchbegin = true;
|
|
}
|
|
|
|
s = dev.getCurrentStateRef();
|
|
synthesizeMouseButtons();
|
|
}
|
|
|
|
public virtual void touchesEnded(DS4Touchpad sender, TouchpadEventArgs arg)
|
|
{
|
|
s = dev.getCurrentStateRef();
|
|
slideright = slideleft = false;
|
|
swipeUp = swipeDown = swipeLeft = swipeRight = false;
|
|
swipeUpB = swipeDownB = swipeLeftB = swipeRightB = 0;
|
|
byte tapSensitivity = Global.getTapSensitivity(deviceNum);
|
|
if (tapSensitivity != 0 && !Global.getUseTPforControls(deviceNum))
|
|
{
|
|
if (secondtouchbegin)
|
|
{
|
|
tappedOnce = false;
|
|
secondtouchbegin = false;
|
|
}
|
|
|
|
DateTime test = arg.timeStamp;
|
|
if (test <= (pastTime + TimeSpan.FromMilliseconds((double)tapSensitivity * 2)) && !arg.touchButtonPressed && !tappedOnce)
|
|
{
|
|
if (Math.Abs(firstTouch.hwX - arg.touches[0].hwX) < 10 && Math.Abs(firstTouch.hwY - arg.touches[0].hwY) < 10)
|
|
{
|
|
if (Global.getDoubleTap(deviceNum))
|
|
{
|
|
tappedOnce = true;
|
|
firstTap = arg.timeStamp;
|
|
TimeofEnd = DateTime.Now; //since arg can't be used in synthesizeMouseButtons
|
|
}
|
|
else
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Left); //this way no delay if disabled
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (Global.getUseTPforControls(deviceNum) == false)
|
|
{
|
|
int[] disArray = Global.getTouchDisInvertTriggers(deviceNum);
|
|
tempBool = true;
|
|
for (int i = 0, arlen = disArray.Length; tempBool && i < arlen; i++)
|
|
{
|
|
if (getDS4ControlsByName(disArray[i]) == false)
|
|
tempBool = false;
|
|
}
|
|
|
|
if (Global.getTrackballMode(deviceNum))
|
|
{
|
|
if (!trackballActive)
|
|
{
|
|
double currentWeight = 1.0;
|
|
double finalWeight = 0.0;
|
|
double x_out = 0.0, y_out = 0.0;
|
|
int idx = -1;
|
|
for (int i = 0; i < TRACKBALL_BUFFER_LEN && idx != trackballBufferHead; i++)
|
|
{
|
|
idx = (trackballBufferTail - i - 1 + TRACKBALL_BUFFER_LEN) % TRACKBALL_BUFFER_LEN;
|
|
x_out += trackballXBuffer[idx] * currentWeight;
|
|
y_out += trackballYBuffer[idx] * currentWeight;
|
|
finalWeight += currentWeight;
|
|
currentWeight *= 1.0;
|
|
}
|
|
|
|
x_out /= finalWeight;
|
|
trackballXVel = x_out;
|
|
y_out /= finalWeight;
|
|
trackballYVel = y_out;
|
|
|
|
trackballActive = true;
|
|
}
|
|
|
|
double tempAngle = Math.Atan2(-trackballYVel, trackballXVel);
|
|
double normX = Math.Abs(Math.Cos(tempAngle));
|
|
double normY = Math.Abs(Math.Sin(tempAngle));
|
|
int signX = Math.Sign(trackballXVel);
|
|
int signY = Math.Sign(trackballYVel);
|
|
|
|
double trackXvDecay = Math.Min(Math.Abs(trackballXVel), trackballAccel * s.elapsedTime * normX);
|
|
double trackYvDecay = Math.Min(Math.Abs(trackballYVel), trackballAccel * s.elapsedTime * normY);
|
|
double xVNew = trackballXVel - (trackXvDecay * signX);
|
|
double yVNew = trackballYVel - (trackYvDecay * signY);
|
|
double xMotion = (xVNew * s.elapsedTime) / TRACKBALL_SCALE;
|
|
double yMotion = (yVNew * s.elapsedTime) / TRACKBALL_SCALE;
|
|
if (xMotion != 0.0)
|
|
{
|
|
xMotion += trackballDXRemain;
|
|
}
|
|
else
|
|
{
|
|
trackballDXRemain = 0.0;
|
|
}
|
|
|
|
int dx = (int)xMotion;
|
|
trackballDXRemain = xMotion - dx;
|
|
|
|
if (yMotion != 0.0)
|
|
{
|
|
yMotion += trackballDYRemain;
|
|
}
|
|
else
|
|
{
|
|
trackballDYRemain = 0.0;
|
|
}
|
|
|
|
int dy = (int)yMotion;
|
|
trackballDYRemain = yMotion - dy;
|
|
|
|
trackballXVel = xVNew;
|
|
trackballYVel = yVNew;
|
|
|
|
if (dx == 0 && dy == 0)
|
|
{
|
|
trackballActive = false;
|
|
}
|
|
else
|
|
{
|
|
cursor.TouchMoveCursor(dx, dy, tempBool);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
synthesizeMouseButtons();
|
|
}
|
|
|
|
private bool isLeft(Touch t)
|
|
{
|
|
return t.hwX < 1920 * 2 / 5;
|
|
}
|
|
|
|
private bool isRight(Touch t)
|
|
{
|
|
return t.hwX >= 1920 * 2 / 5;
|
|
}
|
|
|
|
public virtual void touchUnchanged(DS4Touchpad sender, EventArgs unused)
|
|
{
|
|
s = dev.getCurrentStateRef();
|
|
|
|
if (trackballActive)
|
|
{
|
|
if (Global.getUseTPforControls(deviceNum) == false)
|
|
{
|
|
int[] disArray = Global.getTouchDisInvertTriggers(deviceNum);
|
|
tempBool = true;
|
|
for (int i = 0, arlen = disArray.Length; tempBool && i < arlen; i++)
|
|
{
|
|
if (getDS4ControlsByName(disArray[i]) == false)
|
|
tempBool = false;
|
|
}
|
|
|
|
double tempAngle = Math.Atan2(-trackballYVel, trackballXVel);
|
|
double normX = Math.Abs(Math.Cos(tempAngle));
|
|
double normY = Math.Abs(Math.Sin(tempAngle));
|
|
int signX = Math.Sign(trackballXVel);
|
|
int signY = Math.Sign(trackballYVel);
|
|
double trackXvDecay = Math.Min(Math.Abs(trackballXVel), trackballAccel * s.elapsedTime * normX);
|
|
double trackYvDecay = Math.Min(Math.Abs(trackballYVel), trackballAccel * s.elapsedTime * normY);
|
|
double xVNew = trackballXVel - (trackXvDecay * signX);
|
|
double yVNew = trackballYVel - (trackYvDecay * signY);
|
|
double xMotion = (xVNew * s.elapsedTime) / TRACKBALL_SCALE;
|
|
double yMotion = (yVNew * s.elapsedTime) / TRACKBALL_SCALE;
|
|
if (xMotion != 0.0)
|
|
{
|
|
xMotion += trackballDXRemain;
|
|
}
|
|
else
|
|
{
|
|
trackballDXRemain = 0.0;
|
|
}
|
|
|
|
int dx = (int)xMotion;
|
|
trackballDXRemain = xMotion - dx;
|
|
|
|
if (yMotion != 0.0)
|
|
{
|
|
yMotion += trackballDYRemain;
|
|
}
|
|
else
|
|
{
|
|
trackballDYRemain = 0.0;
|
|
}
|
|
|
|
int dy = (int)yMotion;
|
|
trackballDYRemain = yMotion - dy;
|
|
|
|
trackballXVel = xVNew;
|
|
trackballYVel = yVNew;
|
|
|
|
if (dx == 0 && dy == 0)
|
|
{
|
|
trackballActive = false;
|
|
}
|
|
else
|
|
{
|
|
cursor.TouchMoveCursor(dx, dy, tempBool);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (s.TouchButton)
|
|
synthesizeMouseButtons();
|
|
}
|
|
|
|
public bool dragging, dragging2;
|
|
|
|
private void synthesizeMouseButtons()
|
|
{
|
|
if (Global.GetDS4Action(deviceNum, DS4Controls.TouchLeft, false) == null && leftDown)
|
|
{
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Left);
|
|
dragging2 = true;
|
|
}
|
|
else
|
|
{
|
|
dragging2 = false;
|
|
}
|
|
|
|
if (Global.GetDS4Action(deviceNum, DS4Controls.TouchUpper, false) == null && upperDown)
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Middle);
|
|
|
|
if (Global.GetDS4Action(deviceNum, DS4Controls.TouchRight, false) == null && rightDown)
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Left);
|
|
|
|
if (Global.GetDS4Action(deviceNum, DS4Controls.TouchMulti, false) == null && multiDown)
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Right);
|
|
|
|
if (!Global.UseTPforControls[deviceNum])
|
|
{
|
|
if (tappedOnce)
|
|
{
|
|
DateTime tester = DateTime.Now;
|
|
if (tester > (TimeofEnd + TimeSpan.FromMilliseconds((double)(Global.TapSensitivity[deviceNum]) * 1.5)))
|
|
{
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Left);
|
|
tappedOnce = false;
|
|
}
|
|
//if it fails the method resets, and tries again with a new tester value (gives tap a delay so tap and hold can work)
|
|
}
|
|
if (secondtouchbegin) //if tap and hold (also works as double tap)
|
|
{
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Left);
|
|
dragging = true;
|
|
}
|
|
else
|
|
{
|
|
dragging = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
public virtual void touchButtonUp(DS4Touchpad sender, TouchpadEventArgs arg)
|
|
{
|
|
pushed = DS4Controls.None;
|
|
upperDown = leftDown = rightDown = multiDown = false;
|
|
dev.setRumble(0, 0);
|
|
s = dev.getCurrentStateRef();
|
|
if (s.Touch1 || s.Touch2)
|
|
synthesizeMouseButtons();
|
|
}
|
|
|
|
public virtual void touchButtonDown(DS4Touchpad sender, TouchpadEventArgs arg)
|
|
{
|
|
if (arg.touches == null)
|
|
upperDown = true;
|
|
else if (arg.touches.Length > 1)
|
|
multiDown = true;
|
|
else
|
|
{
|
|
if ((Global.LowerRCOn[deviceNum] && arg.touches[0].hwX > (1920 * 3) / 4 && arg.touches[0].hwY > (960 * 3) / 4))
|
|
Mapping.MapClick(deviceNum, Mapping.Click.Right);
|
|
|
|
if (isLeft(arg.touches[0]))
|
|
leftDown = true;
|
|
else if (isRight(arg.touches[0]))
|
|
rightDown = true;
|
|
}
|
|
|
|
s = dev.getCurrentStateRef();
|
|
synthesizeMouseButtons();
|
|
}
|
|
|
|
public void populatePriorButtonStates()
|
|
{
|
|
priorUpperDown = upperDown;
|
|
priorLeftDown = leftDown;
|
|
priorRightDown = rightDown;
|
|
priorMultiDown = multiDown;
|
|
|
|
priorSwipeLeft = swipeLeft; priorSwipeRight = swipeRight;
|
|
priorSwipeUp = swipeUp; priorSwipeDown = swipeDown;
|
|
priorSwipeLeftB = swipeLeftB; priorSwipeRightB = swipeRightB; priorSwipeUpB = swipeUpB;
|
|
priorSwipeDownB = swipeDownB; priorSwipedB = swipedB;
|
|
}
|
|
|
|
public DS4State getDS4State()
|
|
{
|
|
return s;
|
|
}
|
|
}
|
|
}
|