mirror of
https://github.com/hawkeye-stan/msfs-popout-panel-manager.git
synced 2025-01-27 23:02:44 +01:00
202 lines
8.4 KiB
C#
202 lines
8.4 KiB
C#
using MSFSPopoutPanelManager.DomainModel.Profile;
|
|
using MSFSPopoutPanelManager.DomainModel.Setting;
|
|
using System;
|
|
using System.Diagnostics;
|
|
using System.Linq;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
|
|
namespace MSFSPopoutPanelManager.WindowsAgent
|
|
{
|
|
public class TouchEventManager
|
|
{
|
|
private static IntPtr _hHook = IntPtr.Zero;
|
|
private static PInvoke.WindowsHookExProc callbackDelegate = HookCallBack;
|
|
private static bool _isTouchDown = false;
|
|
private static bool _isDragged = false;
|
|
private static int _refocusedTaskIndex = 0;
|
|
|
|
private const int PANEL_MENUBAR_HEIGHT = 31;
|
|
private const uint TOUCH_FLAG = 0xFF515700;
|
|
private const uint WM_MOUSEMOVE = 0x0200;
|
|
private const uint WM_LBUTTONDOWN = 0x0201;
|
|
private const uint WM_LBUTTONUP = 0x0202;
|
|
private const uint WM_RBUTTONDOWN = 0x0204;
|
|
private const uint WM_RBUTTONUP = 0x0205;
|
|
private const uint MOUSEEVENTF_LEFTDOWN = 0x0002;
|
|
private const uint MOUSEEVENTF_LEFTUP = 0x0004;
|
|
private const uint MOUSEEVENTF_MOVE = 0x0001;
|
|
|
|
public static UserProfile ActiveProfile { private get; set; }
|
|
|
|
public static ApplicationSetting ApplicationSetting { private get; set; }
|
|
|
|
public static void Hook()
|
|
{
|
|
Debug.WriteLine("Executing touch event manager mouse hook...");
|
|
|
|
Process curProcess = Process.GetCurrentProcess();
|
|
ProcessModule curModule = curProcess.MainModule;
|
|
var hookWindowPtr = PInvoke.GetModuleHandle(Process.GetCurrentProcess().MainModule.ModuleName);
|
|
|
|
_hHook = PInvoke.SetWindowsHookEx(HookType.WH_MOUSE_LL, callbackDelegate, hookWindowPtr, 0);
|
|
}
|
|
|
|
public static void UnHook()
|
|
{
|
|
if (_hHook != IntPtr.Zero)
|
|
{
|
|
Debug.WriteLine("Executing touch event manager mouse unhook...");
|
|
|
|
PInvoke.UnhookWindowsHookEx(_hHook);
|
|
_hHook = IntPtr.Zero;
|
|
}
|
|
}
|
|
|
|
public static bool IsHooked { get { return _hHook != IntPtr.Zero; } }
|
|
|
|
private static int HookCallBack(int code, IntPtr wParam, IntPtr lParam)
|
|
{
|
|
if (code != 0)
|
|
return PInvoke.CallNextHookEx(_hHook, code, wParam, lParam);
|
|
|
|
var info = (MSLLHOOKSTRUCT)Marshal.PtrToStructure(lParam, typeof(MSLLHOOKSTRUCT));
|
|
var extraInfo = (uint)info.dwExtraInfo;
|
|
var isTouched = (extraInfo & TOUCH_FLAG) == TOUCH_FLAG;
|
|
|
|
// Mouse Click
|
|
if (!isTouched)
|
|
{
|
|
switch ((uint)wParam)
|
|
{
|
|
case WM_LBUTTONDOWN:
|
|
case WM_LBUTTONUP:
|
|
if (_isTouchDown)
|
|
return 1;
|
|
break;
|
|
}
|
|
|
|
return PInvoke.CallNextHookEx(_hHook, code, wParam, lParam);
|
|
}
|
|
|
|
// If touch point is within pop out panel boundaries and have touch enabled
|
|
var panelConfig = ActiveProfile.PanelConfigs.FirstOrDefault(p => p.TouchEnabled &&
|
|
(info.pt.X > p.Left
|
|
&& info.pt.X < p.Left + p.Width
|
|
&& info.pt.Y > p.Top + (p.HideTitlebar ? 5 : PANEL_MENUBAR_HEIGHT)
|
|
&& info.pt.Y < p.Top + p.Height));
|
|
|
|
if (panelConfig == null)
|
|
return PInvoke.CallNextHookEx(_hHook, code, wParam, lParam);
|
|
|
|
switch ((uint)wParam)
|
|
{
|
|
case WM_RBUTTONDOWN:
|
|
case WM_RBUTTONUP:
|
|
return 1;
|
|
|
|
case WM_LBUTTONDOWN:
|
|
if (!_isTouchDown)
|
|
{
|
|
_refocusedTaskIndex++;
|
|
if (panelConfig.PanelType == PanelType.RefocusDisplay)
|
|
return 1;
|
|
|
|
_isTouchDown = true;
|
|
|
|
if (_isDragged)
|
|
return 1;
|
|
|
|
Task.Run(() =>
|
|
{
|
|
PInvoke.mouse_event(MOUSEEVENTF_LEFTUP, info.pt.X, info.pt.Y, 0, 0); // focus window
|
|
Thread.Sleep(ApplicationSetting.TouchSetting.TouchDownUpDelay + 25);
|
|
PInvoke.mouse_event(MOUSEEVENTF_LEFTDOWN, info.pt.X, info.pt.Y, 0, 0);
|
|
Thread.Sleep(ApplicationSetting.TouchSetting.TouchDownUpDelay + 50);
|
|
_isTouchDown = false;
|
|
});
|
|
}
|
|
return 1;
|
|
case WM_LBUTTONUP:
|
|
if (panelConfig.PanelType == PanelType.RefocusDisplay)
|
|
{
|
|
Task.Run(() =>
|
|
{
|
|
_isTouchDown = false;
|
|
_isDragged = false;
|
|
|
|
// Refocus game window
|
|
if (ApplicationSetting.RefocusSetting.RefocusGameWindow.IsEnabled && panelConfig.AutoGameRefocus)
|
|
{
|
|
var currentRefocusIndex = _refocusedTaskIndex;
|
|
|
|
Thread.Sleep(Convert.ToInt32(ApplicationSetting.RefocusSetting.RefocusGameWindow.Delay * 1000));
|
|
|
|
if (currentRefocusIndex == _refocusedTaskIndex)
|
|
{
|
|
var rect = WindowActionManager.GetWindowRectangle(WindowProcessManager.SimulatorProcess.Handle);
|
|
InputEmulationManager.LeftClick(rect.X + rect.Width / 2, rect.Y + rect.Height / 2);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
else
|
|
{
|
|
Task.Run(() =>
|
|
{
|
|
while (_isTouchDown) { }
|
|
|
|
if (_isDragged)
|
|
{
|
|
if (ApplicationSetting.TouchSetting.TouchDownUpDelay > 0)
|
|
Thread.Sleep(ApplicationSetting.TouchSetting.TouchDownUpDelay);
|
|
|
|
PInvoke.SetCursorPos(info.pt.X, info.pt.Y);
|
|
|
|
if (ApplicationSetting.TouchSetting.TouchDownUpDelay > 0)
|
|
Thread.Sleep(ApplicationSetting.TouchSetting.TouchDownUpDelay);
|
|
|
|
InputEmulationManager.LeftClickFast(info.pt.X, info.pt.Y);
|
|
|
|
_isDragged = false;
|
|
}
|
|
else
|
|
{
|
|
if (ApplicationSetting.TouchSetting.TouchDownUpDelay > 0)
|
|
Thread.Sleep(ApplicationSetting.TouchSetting.TouchDownUpDelay);
|
|
|
|
PInvoke.mouse_event(MOUSEEVENTF_LEFTUP, info.pt.X, info.pt.Y, 0, 0);
|
|
}
|
|
|
|
_isTouchDown = false;
|
|
|
|
// Refocus game window
|
|
if (ApplicationSetting.RefocusSetting.RefocusGameWindow.IsEnabled && panelConfig.AutoGameRefocus)
|
|
{
|
|
var currentRefocusIndex = _refocusedTaskIndex;
|
|
|
|
Thread.Sleep(Convert.ToInt32(ApplicationSetting.RefocusSetting.RefocusGameWindow.Delay * 1000));
|
|
|
|
if (currentRefocusIndex == _refocusedTaskIndex)
|
|
{
|
|
var rect = WindowActionManager.GetWindowRectangle(WindowProcessManager.SimulatorProcess.Handle);
|
|
PInvoke.SetCursorPos(rect.X + rect.Width / 2, rect.Y + rect.Height / 2);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
return 1;
|
|
case WM_MOUSEMOVE:
|
|
if (_isTouchDown)
|
|
{
|
|
_isDragged = true;
|
|
return 1;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return PInvoke.CallNextHookEx(_hHook, code, wParam, lParam);
|
|
}
|
|
}
|
|
}
|