2015-01-07 05:26:32 -08:00
|
|
|
#define PSAPI_VERSION 1
|
2014-10-17 04:01:18 -07:00
|
|
|
#include <obs.h>
|
|
|
|
#include <util/dstr.h>
|
|
|
|
|
|
|
|
#include <windows.h>
|
|
|
|
#include <psapi.h>
|
|
|
|
#include "window-helpers.h"
|
2014-11-10 01:27:42 -08:00
|
|
|
#include "obfuscate.h"
|
2014-10-17 04:01:18 -07:00
|
|
|
|
|
|
|
static inline void encode_dstr(struct dstr *str)
|
|
|
|
{
|
|
|
|
dstr_replace(str, "#", "#22");
|
|
|
|
dstr_replace(str, ":", "#3A");
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline char *decode_str(const char *src)
|
|
|
|
{
|
|
|
|
struct dstr str = {0};
|
|
|
|
dstr_copy(&str, src);
|
|
|
|
dstr_replace(&str, "#3A", ":");
|
|
|
|
dstr_replace(&str, "#22", "#");
|
|
|
|
return str.array;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern void build_window_strings(const char *str,
|
|
|
|
char **class,
|
|
|
|
char **title,
|
|
|
|
char **exe)
|
|
|
|
{
|
|
|
|
char **strlist;
|
|
|
|
|
2014-11-10 01:18:28 -08:00
|
|
|
*class = NULL;
|
|
|
|
*title = NULL;
|
|
|
|
*exe = NULL;
|
|
|
|
|
2014-10-17 04:01:18 -07:00
|
|
|
if (!str) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
strlist = strlist_split(str, ':', true);
|
|
|
|
|
|
|
|
if (strlist && strlist[0] && strlist[1] && strlist[2]) {
|
|
|
|
*title = decode_str(strlist[0]);
|
|
|
|
*class = decode_str(strlist[1]);
|
|
|
|
*exe = decode_str(strlist[2]);
|
|
|
|
}
|
|
|
|
|
|
|
|
strlist_free(strlist);
|
|
|
|
}
|
|
|
|
|
2014-11-10 01:27:42 -08:00
|
|
|
static HMODULE kernel32(void)
|
|
|
|
{
|
|
|
|
static HMODULE kernel32_handle = NULL;
|
|
|
|
if (!kernel32_handle)
|
|
|
|
kernel32_handle = GetModuleHandleA("kernel32");
|
|
|
|
return kernel32_handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline HANDLE open_process(DWORD desired_access, bool inherit_handle,
|
|
|
|
DWORD process_id)
|
|
|
|
{
|
|
|
|
static HANDLE (WINAPI *open_process_proc)(DWORD, BOOL, DWORD) = NULL;
|
|
|
|
if (!open_process_proc)
|
|
|
|
open_process_proc = get_obfuscated_func(kernel32(),
|
|
|
|
"B}caZyah`~q", 0x2D5BEBAF6DDULL);
|
|
|
|
|
|
|
|
return open_process_proc(desired_access, inherit_handle, process_id);
|
|
|
|
}
|
|
|
|
|
2014-11-10 01:33:08 -08:00
|
|
|
bool get_window_exe(struct dstr *name, HWND window)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
|
|
|
wchar_t wname[MAX_PATH];
|
|
|
|
struct dstr temp = {0};
|
|
|
|
bool success = false;
|
|
|
|
HANDLE process = NULL;
|
|
|
|
char *slash;
|
|
|
|
DWORD id;
|
|
|
|
|
|
|
|
GetWindowThreadProcessId(window, &id);
|
2016-05-13 04:21:39 -07:00
|
|
|
if (id == GetCurrentProcessId())
|
|
|
|
return false;
|
2014-10-17 04:01:18 -07:00
|
|
|
|
2014-11-10 01:27:42 -08:00
|
|
|
process = open_process(PROCESS_QUERY_LIMITED_INFORMATION, false, id);
|
2014-10-17 04:01:18 -07:00
|
|
|
if (!process)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
if (!GetProcessImageFileNameW(process, wname, MAX_PATH))
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
dstr_from_wcs(&temp, wname);
|
|
|
|
slash = strrchr(temp.array, '\\');
|
|
|
|
if (!slash)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
dstr_copy(name, slash+1);
|
|
|
|
success = true;
|
|
|
|
|
|
|
|
fail:
|
|
|
|
if (!success)
|
|
|
|
dstr_copy(name, "unknown");
|
|
|
|
|
|
|
|
dstr_free(&temp);
|
|
|
|
CloseHandle(process);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-07-30 11:24:39 -07:00
|
|
|
void get_window_title(struct dstr *name, HWND hwnd)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
|
|
|
wchar_t *temp;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
len = GetWindowTextLengthW(hwnd);
|
|
|
|
if (!len)
|
|
|
|
return;
|
|
|
|
|
|
|
|
temp = malloc(sizeof(wchar_t) * (len+1));
|
2015-10-12 21:50:03 +02:00
|
|
|
if (GetWindowTextW(hwnd, temp, len+1))
|
|
|
|
dstr_from_wcs(name, temp);
|
2014-10-17 04:01:18 -07:00
|
|
|
free(temp);
|
|
|
|
}
|
|
|
|
|
2016-07-30 11:24:39 -07:00
|
|
|
void get_window_class(struct dstr *class, HWND hwnd)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
|
|
|
wchar_t temp[256];
|
|
|
|
|
|
|
|
temp[0] = 0;
|
2015-10-12 21:50:03 +02:00
|
|
|
if (GetClassNameW(hwnd, temp, sizeof(temp) / sizeof(wchar_t)))
|
|
|
|
dstr_from_wcs(class, temp);
|
2014-10-17 04:01:18 -07:00
|
|
|
}
|
|
|
|
|
2016-12-23 02:15:59 -08:00
|
|
|
/* not capturable or internal windows */
|
|
|
|
static const char *internal_microsoft_exes[] = {
|
|
|
|
"applicationframehost",
|
|
|
|
"shellexperiencehost",
|
|
|
|
"winstore.app",
|
|
|
|
"searchui",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool is_microsoft_internal_window_exe(const char *exe)
|
|
|
|
{
|
|
|
|
char cur_exe[MAX_PATH];
|
|
|
|
|
|
|
|
if (!exe)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (const char **vals = internal_microsoft_exes; *vals; vals++) {
|
|
|
|
strcpy(cur_exe, *vals);
|
|
|
|
strcat(cur_exe, ".exe");
|
|
|
|
|
|
|
|
if (strcmpi(cur_exe, exe) == 0)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-30 12:49:05 -07:00
|
|
|
static void add_window(obs_property_t *p, HWND hwnd, add_window_cb callback)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
|
|
|
struct dstr class = {0};
|
|
|
|
struct dstr title = {0};
|
|
|
|
struct dstr exe = {0};
|
|
|
|
struct dstr encoded = {0};
|
|
|
|
struct dstr desc = {0};
|
|
|
|
|
|
|
|
if (!get_window_exe(&exe, hwnd))
|
|
|
|
return;
|
2016-12-23 02:15:59 -08:00
|
|
|
if (is_microsoft_internal_window_exe(exe.array)) {
|
|
|
|
dstr_free(&exe);
|
|
|
|
return;
|
|
|
|
}
|
2014-10-17 04:01:18 -07:00
|
|
|
get_window_title(&title, hwnd);
|
|
|
|
get_window_class(&class, hwnd);
|
|
|
|
|
2016-07-30 12:49:05 -07:00
|
|
|
if (callback && !callback(title.array, class.array, exe.array)) {
|
|
|
|
dstr_free(&title);
|
|
|
|
dstr_free(&class);
|
|
|
|
dstr_free(&exe);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-10-17 04:01:18 -07:00
|
|
|
dstr_printf(&desc, "[%s]: %s", exe.array, title.array);
|
|
|
|
|
|
|
|
encode_dstr(&title);
|
|
|
|
encode_dstr(&class);
|
|
|
|
encode_dstr(&exe);
|
|
|
|
|
|
|
|
dstr_cat_dstr(&encoded, &title);
|
|
|
|
dstr_cat(&encoded, ":");
|
|
|
|
dstr_cat_dstr(&encoded, &class);
|
|
|
|
dstr_cat(&encoded, ":");
|
|
|
|
dstr_cat_dstr(&encoded, &exe);
|
|
|
|
|
|
|
|
obs_property_list_add_string(p, desc.array, encoded.array);
|
|
|
|
|
|
|
|
dstr_free(&encoded);
|
|
|
|
dstr_free(&desc);
|
|
|
|
dstr_free(&class);
|
|
|
|
dstr_free(&title);
|
|
|
|
dstr_free(&exe);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool check_window_valid(HWND window, enum window_search_mode mode)
|
|
|
|
{
|
|
|
|
DWORD styles, ex_styles;
|
|
|
|
RECT rect;
|
|
|
|
|
|
|
|
if (!IsWindowVisible(window) ||
|
|
|
|
(mode == EXCLUDE_MINIMIZED && IsIconic(window)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
GetClientRect(window, &rect);
|
|
|
|
styles = (DWORD)GetWindowLongPtr(window, GWL_STYLE);
|
|
|
|
ex_styles = (DWORD)GetWindowLongPtr(window, GWL_EXSTYLE);
|
|
|
|
|
|
|
|
if (ex_styles & WS_EX_TOOLWINDOW)
|
|
|
|
return false;
|
|
|
|
if (styles & WS_CHILD)
|
|
|
|
return false;
|
2014-11-10 01:28:59 -08:00
|
|
|
if (mode == EXCLUDE_MINIMIZED && (rect.bottom == 0 || rect.right == 0))
|
2014-10-17 04:01:18 -07:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-10-31 01:46:18 -07:00
|
|
|
bool is_uwp_window(HWND hwnd)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
2016-10-31 01:46:18 -07:00
|
|
|
wchar_t name[256];
|
|
|
|
|
|
|
|
name[0] = 0;
|
|
|
|
if (!GetClassNameW(hwnd, name, sizeof(name) / sizeof(wchar_t)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return wcscmp(name, L"ApplicationFrameWindow") == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
HWND get_uwp_actual_window(HWND parent)
|
|
|
|
{
|
|
|
|
DWORD parent_id = 0;
|
|
|
|
HWND child;
|
|
|
|
|
|
|
|
GetWindowThreadProcessId(parent, &parent_id);
|
2017-01-18 20:18:06 -08:00
|
|
|
child = FindWindowEx(parent, NULL, NULL, NULL);
|
2016-10-31 01:46:18 -07:00
|
|
|
|
|
|
|
while (child) {
|
|
|
|
DWORD child_id = 0;
|
|
|
|
GetWindowThreadProcessId(child, &child_id);
|
|
|
|
|
|
|
|
if (child_id != parent_id)
|
|
|
|
return child;
|
|
|
|
|
2017-01-18 20:18:06 -08:00
|
|
|
child = FindWindowEx(parent, child, NULL, NULL);
|
2016-10-31 01:46:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-02-21 00:37:50 -08:00
|
|
|
static HWND next_window(HWND window, enum window_search_mode mode,
|
|
|
|
HWND *parent, bool use_findwindowex)
|
2016-10-31 01:46:18 -07:00
|
|
|
{
|
|
|
|
if (*parent) {
|
|
|
|
window = *parent;
|
|
|
|
*parent = NULL;
|
|
|
|
}
|
|
|
|
|
2014-10-17 04:01:18 -07:00
|
|
|
while (true) {
|
2017-02-21 00:37:50 -08:00
|
|
|
if (use_findwindowex)
|
|
|
|
window = FindWindowEx(GetDesktopWindow(), window, NULL,
|
|
|
|
NULL);
|
|
|
|
else
|
|
|
|
window = GetNextWindow(window, GW_HWNDNEXT);
|
|
|
|
|
2014-10-17 04:01:18 -07:00
|
|
|
if (!window || check_window_valid(window, mode))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-10-31 01:46:18 -07:00
|
|
|
if (is_uwp_window(window)) {
|
|
|
|
HWND child = get_uwp_actual_window(window);
|
|
|
|
if (child) {
|
|
|
|
*parent = window;
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-17 04:01:18 -07:00
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
2017-02-21 00:37:50 -08:00
|
|
|
static HWND first_window(enum window_search_mode mode, HWND *parent,
|
|
|
|
bool *use_findwindowex)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
2017-01-18 20:18:06 -08:00
|
|
|
HWND window = FindWindowEx(GetDesktopWindow(), NULL, NULL, NULL);
|
2016-10-31 01:46:18 -07:00
|
|
|
|
2017-02-21 00:37:50 -08:00
|
|
|
if (!window) {
|
|
|
|
*use_findwindowex = false;
|
|
|
|
window = GetWindow(GetDesktopWindow(), GW_CHILD);
|
|
|
|
} else {
|
|
|
|
*use_findwindowex = true;
|
|
|
|
}
|
|
|
|
|
2016-10-31 01:46:18 -07:00
|
|
|
*parent = NULL;
|
|
|
|
|
2017-02-21 00:37:50 -08:00
|
|
|
if (!check_window_valid(window, mode)) {
|
|
|
|
window = next_window(window, mode, parent, *use_findwindowex);
|
|
|
|
|
|
|
|
if (!window && *use_findwindowex) {
|
|
|
|
*use_findwindowex = false;
|
|
|
|
|
|
|
|
window = GetWindow(GetDesktopWindow(), GW_CHILD);
|
|
|
|
if (!check_window_valid(window, mode))
|
|
|
|
window = next_window(window, mode, parent,
|
|
|
|
*use_findwindowex);
|
|
|
|
}
|
|
|
|
}
|
2016-10-31 01:46:18 -07:00
|
|
|
|
|
|
|
if (is_uwp_window(window)) {
|
|
|
|
HWND child = get_uwp_actual_window(window);
|
|
|
|
if (child) {
|
|
|
|
*parent = window;
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-17 04:01:18 -07:00
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
2016-07-30 12:49:05 -07:00
|
|
|
void fill_window_list(obs_property_t *p, enum window_search_mode mode,
|
|
|
|
add_window_cb callback)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
2016-10-31 01:46:18 -07:00
|
|
|
HWND parent;
|
2017-02-21 00:37:50 -08:00
|
|
|
bool use_findwindowex = false;
|
|
|
|
|
|
|
|
HWND window = first_window(mode, &parent, &use_findwindowex);
|
2014-10-17 04:01:18 -07:00
|
|
|
|
|
|
|
while (window) {
|
2016-07-30 12:49:05 -07:00
|
|
|
add_window(p, window, callback);
|
2017-02-21 00:37:50 -08:00
|
|
|
window = next_window(window, mode, &parent, use_findwindowex);
|
2014-10-17 04:01:18 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int window_rating(HWND window,
|
|
|
|
enum window_priority priority,
|
|
|
|
const char *class,
|
|
|
|
const char *title,
|
2016-12-16 02:04:06 -08:00
|
|
|
const char *exe,
|
|
|
|
bool uwp_window)
|
2014-10-17 04:01:18 -07:00
|
|
|
{
|
|
|
|
struct dstr cur_class = {0};
|
|
|
|
struct dstr cur_title = {0};
|
|
|
|
struct dstr cur_exe = {0};
|
2017-04-30 00:25:02 -07:00
|
|
|
int val = 0x7FFFFFFF;
|
2014-10-17 04:01:18 -07:00
|
|
|
|
|
|
|
if (!get_window_exe(&cur_exe, window))
|
2017-04-30 00:25:02 -07:00
|
|
|
return 0x7FFFFFFF;
|
2014-10-17 04:01:18 -07:00
|
|
|
get_window_title(&cur_title, window);
|
|
|
|
get_window_class(&cur_class, window);
|
|
|
|
|
2017-04-30 00:25:02 -07:00
|
|
|
bool class_matches = dstr_cmpi(&cur_class, class) == 0;
|
|
|
|
bool exe_matches = dstr_cmpi(&cur_exe, exe) == 0;
|
|
|
|
int title_val = abs(dstr_cmpi(&cur_title, title));
|
2014-10-17 04:01:18 -07:00
|
|
|
|
2017-04-30 00:25:02 -07:00
|
|
|
/* always match by name with UWP windows */
|
2016-12-16 02:04:06 -08:00
|
|
|
if (uwp_window) {
|
2017-04-30 00:25:02 -07:00
|
|
|
if (priority == WINDOW_PRIORITY_EXE && !exe_matches)
|
|
|
|
val = 0x7FFFFFFF;
|
|
|
|
else
|
|
|
|
val = title_val == 0 ? 0 : 0x7FFFFFFF;
|
|
|
|
|
|
|
|
} else if (priority == WINDOW_PRIORITY_CLASS) {
|
|
|
|
val = class_matches ? title_val : 0x7FFFFFFF;
|
|
|
|
if (val != 0x7FFFFFFF && !exe_matches)
|
|
|
|
val += 0x1000;
|
|
|
|
|
|
|
|
} else if (priority == WINDOW_PRIORITY_TITLE) {
|
|
|
|
val = title_val == 0 ? 0 : 0x7FFFFFFF;
|
|
|
|
|
|
|
|
} else if (priority == WINDOW_PRIORITY_EXE) {
|
|
|
|
val = exe_matches ? title_val : 0x7FFFFFFF;
|
2016-12-16 02:04:06 -08:00
|
|
|
}
|
2014-10-17 04:01:18 -07:00
|
|
|
|
|
|
|
dstr_free(&cur_class);
|
|
|
|
dstr_free(&cur_title);
|
|
|
|
dstr_free(&cur_exe);
|
|
|
|
|
2017-04-30 00:25:02 -07:00
|
|
|
return val;
|
2014-10-17 04:01:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
HWND find_window(enum window_search_mode mode,
|
|
|
|
enum window_priority priority,
|
|
|
|
const char *class,
|
|
|
|
const char *title,
|
|
|
|
const char *exe)
|
|
|
|
{
|
2016-10-31 01:46:18 -07:00
|
|
|
HWND parent;
|
2017-02-21 00:37:50 -08:00
|
|
|
bool use_findwindowex = false;
|
|
|
|
|
|
|
|
HWND window = first_window(mode, &parent, &use_findwindowex);
|
2014-10-17 04:01:18 -07:00
|
|
|
HWND best_window = NULL;
|
2017-04-30 00:25:02 -07:00
|
|
|
int best_rating = 0x7FFFFFFF;
|
2016-12-23 02:01:09 -08:00
|
|
|
|
|
|
|
if (!class)
|
|
|
|
return NULL;
|
|
|
|
|
2016-12-16 02:04:06 -08:00
|
|
|
bool uwp_window = strcmp(class, "Windows.UI.Core.CoreWindow") == 0;
|
2014-10-17 04:01:18 -07:00
|
|
|
|
|
|
|
while (window) {
|
2016-12-16 02:04:06 -08:00
|
|
|
int rating = window_rating(window, priority, class, title, exe,
|
|
|
|
uwp_window);
|
2017-04-30 00:25:02 -07:00
|
|
|
if (rating < best_rating) {
|
2014-10-17 04:01:18 -07:00
|
|
|
best_rating = rating;
|
|
|
|
best_window = window;
|
2017-04-30 00:25:02 -07:00
|
|
|
if (rating == 0)
|
|
|
|
break;
|
2014-10-17 04:01:18 -07:00
|
|
|
}
|
|
|
|
|
2017-02-21 00:37:50 -08:00
|
|
|
window = next_window(window, mode, &parent, use_findwindowex);
|
2014-10-17 04:01:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return best_window;
|
|
|
|
}
|