2019-12-18 22:33:23 +01:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Runtime.InteropServices ;
using System.Security ;
using System.Text ;
2020-01-09 16:04:27 +01:00
using System.Diagnostics ; // StopWatch
using System.Threading ; // Sleep
2019-12-18 22:33:23 +01:00
using System.Threading.Tasks ;
using DS4Windows ;
namespace DS4WinWPF
{
[SuppressUnmanagedCodeSecurity]
public class AutoProfileChecker
{
private AutoProfileHolder profileHolder ;
private IntPtr prevForegroundWnd = IntPtr . Zero ;
private uint prevForegroundProcessID ;
private string prevForegroundProcessName = string . Empty ;
private string prevForegroundWndTitleName = string . Empty ;
private StringBuilder autoProfileCheckTextBuilder = new StringBuilder ( 1000 ) ;
private int autoProfileDebugLogLevel = 0 ;
private bool turnOffTemp ;
private AutoProfileEntity tempAutoProfile ;
private bool running ;
public int AutoProfileDebugLogLevel { get = > autoProfileDebugLogLevel ; set = > autoProfileDebugLogLevel = value ; }
public bool Running { get = > running ; set = > running = value ; }
public delegate void ChangeServiceHandler ( AutoProfileChecker sender , bool state ) ;
public event ChangeServiceHandler RequestServiceChange ;
public AutoProfileChecker ( AutoProfileHolder holder )
{
profileHolder = holder ;
}
public void Process ( )
{
string topProcessName , topWindowTitle ;
bool turnOffDS4WinApp = false ;
AutoProfileEntity matchedProfileEntity = null ;
if ( GetTopWindowName ( out topProcessName , out topWindowTitle ) )
{
// Find a profile match based on autoprofile program path and wnd title list.
// The same program may set different profiles for each of the controllers, so we need an array of newProfileName[controllerIdx] values.
for ( int i = 0 , pathsLen = profileHolder . AutoProfileColl . Count ; i < pathsLen ; i + + )
{
AutoProfileEntity tempEntity = profileHolder . AutoProfileColl [ i ] ;
if ( tempEntity . IsMatch ( topProcessName , topWindowTitle ) )
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. Rule#{i + 1} Path={tempEntity.path} Title={tempEntity.title}" , false ) ;
// Matching autoprofile rule found
turnOffDS4WinApp = tempEntity . Turnoff ;
matchedProfileEntity = tempEntity ;
break ;
}
}
if ( matchedProfileEntity ! = null )
{
2020-01-09 16:04:27 +01:00
bool forceLoadProfile = false ;
if ( ! turnOffDS4WinApp & & turnOffTemp )
{
// DS4Win was temporarily turned off by another auto-profile rule. Turn DS4Win on before trying to load a new profile because otherwise the new profile won't do anything.
// Force load the profile when DS4Win service afer waking up DS4Win service to make sure that the new profile will be active.
turnOffTemp = false ;
SetAndWaitServiceStatus ( true ) ;
forceLoadProfile = true ;
}
2019-12-18 22:33:23 +01:00
// Program match found. Check if the new profile is different than current profile of the controller. Load the new profile only if it is not already loaded.
for ( int j = 0 ; j < 4 ; j + + )
{
string tempname = matchedProfileEntity . ProfileNames [ j ] ;
if ( tempname ! = string . Empty & & tempname ! = "(none)" )
{
if ( ( Global . useTempProfile [ j ] & & tempname ! = Global . tempprofilename [ j ] ) | |
2020-01-09 16:04:27 +01:00
( ! Global . useTempProfile [ j ] & & tempname ! = Global . ProfilePath [ j ] ) | |
forceLoadProfile )
2019-12-18 22:33:23 +01:00
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. LoadProfile Controller {j + 1}={tempname}" , false ) ;
Global . LoadTempProfile ( j , tempname , true , Program . rootHub ) ; // j is controller index, i is filename
//if (LaunchProgram[j] != string.Empty) Process.Start(LaunchProgram[j]);
}
else
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. LoadProfile Controller {j + 1}={tempname} (already loaded)" , false ) ;
}
}
}
if ( turnOffDS4WinApp )
{
turnOffTemp = true ;
2020-01-09 16:04:27 +01:00
if ( App . rootHub . running )
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. Turning DS4Windows temporarily off" , false ) ;
SetAndWaitServiceStatus ( false ) ;
}
2019-12-18 22:33:23 +01:00
}
tempAutoProfile = matchedProfileEntity ;
}
else if ( tempAutoProfile ! = null )
{
2020-01-09 16:04:27 +01:00
if ( turnOffTemp & & DS4Windows . Global . AutoProfileRevertDefaultProfile )
{
turnOffTemp = false ;
if ( ! App . rootHub . running )
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. Turning DS4Windows on before reverting to default profile" , false ) ;
SetAndWaitServiceStatus ( true ) ;
}
}
2019-12-18 22:33:23 +01:00
tempAutoProfile = null ;
for ( int j = 0 ; j < 4 ; j + + )
{
if ( Global . useTempProfile [ j ] )
{
2019-12-29 00:34:41 +01:00
if ( DS4Windows . Global . AutoProfileRevertDefaultProfile )
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. Unknown process. Reverting to default profile. Controller {j + 1}={Global.ProfilePath[j]} (default)" , false ) ;
2019-12-18 22:33:23 +01:00
2019-12-29 00:34:41 +01:00
Global . LoadProfile ( j , false , Program . rootHub ) ;
}
else
{
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. Unknown process. Existing profile left as active. Controller {j + 1}={Global.tempprofilename[j]}" , false ) ;
}
2019-12-18 22:33:23 +01:00
}
}
}
}
}
private bool GetTopWindowName ( out string topProcessName , out string topWndTitleName )
{
IntPtr hWnd = GetForegroundWindow ( ) ;
if ( hWnd = = IntPtr . Zero )
{
// Top window unknown or cannot acquire a handle. Return FALSE and return unknown process and wndTitle values
prevForegroundWnd = IntPtr . Zero ;
prevForegroundProcessID = 0 ;
topProcessName = topWndTitleName = String . Empty ;
return false ;
}
//
// If this function was called from "auto-profile watcher timer" then check cached "previous hWnd handle". If the current hWnd is the same
// as during the previous check then return cached previous wnd and name values (ie. foreground app and window are assumed to be the same, so no need to re-query names).
// This should optimize the auto-profile timer check process and causes less burden to .NET GC collector because StringBuffer is not re-allocated every second.
//
// Note! hWnd handles may be re-cycled but not during the lifetime of the window. This "cache" optimization still works because when an old window is closed
// then foreground window changes to something else and the cached prevForgroundWnd variable is updated to store the new hWnd handle.
// It doesn't matter even when the previously cached handle is recycled by WinOS to represent some other window (it is no longer used as a cached value anyway).
//
if ( hWnd = = prevForegroundWnd )
{
// The active window is still the same. Return cached process and wndTitle values and FALSE to indicate caller that no changes since the last call of this method
topProcessName = prevForegroundProcessName ;
topWndTitleName = prevForegroundWndTitleName ;
return false ;
}
prevForegroundWnd = hWnd ;
IntPtr hProcess = IntPtr . Zero ;
uint lpdwProcessId = 0 ;
GetWindowThreadProcessId ( hWnd , out lpdwProcessId ) ;
if ( lpdwProcessId = = prevForegroundProcessID )
{
topProcessName = prevForegroundProcessName ;
}
else
{
prevForegroundProcessID = lpdwProcessId ;
hProcess = OpenProcess ( 0x0410 , false , lpdwProcessId ) ;
if ( hProcess ! = IntPtr . Zero ) GetModuleFileNameEx ( hProcess , IntPtr . Zero , autoProfileCheckTextBuilder , autoProfileCheckTextBuilder . Capacity ) ;
else autoProfileCheckTextBuilder . Clear ( ) ;
prevForegroundProcessName = topProcessName = autoProfileCheckTextBuilder . Replace ( '/' , '\\' ) . ToString ( ) . ToLower ( ) ;
}
GetWindowText ( hWnd , autoProfileCheckTextBuilder , autoProfileCheckTextBuilder . Capacity ) ;
prevForegroundWndTitleName = topWndTitleName = autoProfileCheckTextBuilder . ToString ( ) . ToLower ( ) ;
if ( hProcess ! = IntPtr . Zero ) CloseHandle ( hProcess ) ;
if ( autoProfileDebugLogLevel > 0 )
DS4Windows . AppLogger . LogToGui ( $"DEBUG: Auto-Profile. PID={lpdwProcessId} Path={topProcessName} | WND={hWnd} Title={topWndTitleName}" , false ) ;
return true ;
}
2020-01-09 16:04:27 +01:00
private void SetAndWaitServiceStatus ( bool serviceRunningStatus )
{
// Start or Stop the service only if it is not already in the requested state
if ( App . rootHub . running ! = serviceRunningStatus )
{
RequestServiceChange ? . Invoke ( this , serviceRunningStatus ) ;
// Wait until DS4Win app service is running or stopped (as requested by serviceRunningStatus value) or timeout.
// LoadProfile call fails if a new profile is loaded while DS4Win service is still in stopped state (ie the loaded temp profile doesn't do anything).
Stopwatch sw = new Stopwatch ( ) ;
sw . Start ( ) ;
while ( App . rootHub . running ! = serviceRunningStatus & & sw . Elapsed . TotalSeconds < 10 )
{
2020-01-10 00:20:46 +01:00
Thread . SpinWait ( 1000 ) ;
2020-01-09 16:04:27 +01:00
}
2020-01-10 00:20:46 +01:00
Thread . SpinWait ( 1000 ) ;
2020-01-09 16:04:27 +01:00
}
}
2019-12-18 22:33:23 +01:00
[DllImport("user32.dll")]
private static extern IntPtr GetForegroundWindow ( ) ;
[DllImport("user32.dll")]
private static extern uint GetWindowThreadProcessId ( IntPtr hWnd , out uint lpdwProcessId ) ;
[DllImport("kernel32.dll")]
private static extern IntPtr OpenProcess ( uint dwDesiredAccess , bool bInheritHandle , uint dwProcessId ) ;
[DllImport("kernel32.dll")]
private static extern bool CloseHandle ( IntPtr handle ) ;
[DllImport("psapi.dll")]
private static extern uint GetModuleFileNameEx ( IntPtr hWnd , IntPtr hModule , StringBuilder lpFileName , int nSize ) ;
[DllImport("user32.dll", CharSet = CharSet.Auto)]
private static extern int GetWindowText ( IntPtr hWnd , StringBuilder lpString , int nSize ) ;
}
}