mirror of
https://github.com/valinet/ExplorerPatcher.git
synced 2024-11-30 18:24:36 +01:00
ca8ce137d8
As a way to showcase the `ep_extra` loader I wrote in my previous commit, I have developed this module which enables the use of Windows 7's Alt-Tab. Mind you, the implementation might not be complete and 100% ironed out, but rather a functional preview on what it can look like. Grab a copy of `AltTab.dll` from Windows 7 (I have tested against version 6.1.7600.16385) and place it in `C:\Windows`. Then, reload `explorer` with the `ep_extra` loader and this module in `C:\Windows` and when you "alt-tab", you should see the Windows 7 window switcher. This is not a reproduction - it executes the original code from Microsoft's DLL, only slightly patched to work properly in with the updated APIs in the newer Windows versions.
120 lines
4.7 KiB
C
120 lines
4.7 KiB
C
#include <initguid.h>
|
|
#include <Windows.h>
|
|
#include "../libs/libvalinet/valinet/hooking/iatpatch.h"
|
|
#include "../libs/sws/SimpleWindowSwitcher/sws_WindowHelpers.h"
|
|
#pragma comment(lib, "Uxtheme.lib")
|
|
|
|
HMODULE hModule = NULL;
|
|
HMODULE hAltTab = NULL;
|
|
IOleCommandTarget* pAltTabSSO = NULL;
|
|
|
|
DEFINE_GUID(CLSID_AltTabSSO,
|
|
0xA1607060, 0x5D4C, 0x467A, 0xB7, 0x11, 0x2B, 0x59, 0xA6, 0xF2, 0x59, 0x57);
|
|
|
|
HRESULT AltTab_DwmpActivateLivePreview(int s, HWND hWnd, int c, int d) {
|
|
return S_OK;
|
|
}
|
|
|
|
int AltTab_LoadStringW(HINSTANCE hInstance, UINT uID, LPWSTR lpBuffer, int cchBufferMax) {
|
|
if (uID == 0x3E8) {
|
|
swprintf_s(lpBuffer, cchBufferMax, L"AltTab"); return 6;
|
|
}
|
|
else if (uID == 0x3EA) {
|
|
if (cchBufferMax < MAX_PATH) return 0;
|
|
sws_WindowHelpers_GetDesktopText(lpBuffer);
|
|
int len = wcslen(lpBuffer);
|
|
for (int i = 0; i < len; ++i) if (lpBuffer[i] == L'&') lpBuffer[i] = L'\u200E';
|
|
return len;
|
|
}
|
|
return LoadStringW(hInstance, uID, lpBuffer, cchBufferMax);
|
|
}
|
|
|
|
HTHEME AltTab_OpenThemeData(HWND hwnd, LPCWSTR pszClassList) {
|
|
if (!wcscmp(pszClassList, L"AltTab")) return OpenThemeData(hwnd, L"WINDOW");
|
|
return OpenThemeData(hwnd, pszClassList);
|
|
}
|
|
|
|
HRESULT AltTab_DrawThemeTextEx(HTHEME hTheme, HDC hdc, int iPartId, int iStateId, LPCWSTR pszText, int cchText, DWORD dwTextFlags, LPRECT pRect, const DTTOPTS* pOptions) {
|
|
HRESULT hr = S_OK;
|
|
HTHEME hTheme2 = OpenThemeData(NULL, L"TEXTSTYLE");
|
|
if (hTheme2) hr = DrawThemeTextEx(hTheme2, hdc, iPartId + 1, iStateId, pszText, cchText, dwTextFlags, pRect, pOptions);
|
|
if (hTheme2) CloseThemeData(hTheme2);
|
|
return hr;
|
|
}
|
|
|
|
BOOL AltTab_IsWindowEnabled(HWND hWnd) {
|
|
if (!IsWindowEnabled(hWnd)) return FALSE;
|
|
BOOL isCloaked;
|
|
DwmGetWindowAttribute(hWnd, DWMWA_CLOAKED, &isCloaked, sizeof(BOOL));
|
|
if (isCloaked) return FALSE;
|
|
if (sws_IsShellFrameWindow(hWnd) && !_sws_GhostWindowFromHungWindow(hWnd)) return TRUE;
|
|
if (_sws_IsShellManagedWindow(hWnd) && !sws_WindowHelpers_ShouldTreatShellManagedWindowAsNotShellManaged(hWnd)) return FALSE;
|
|
if (sws_WindowHelpers_IsWindowShellManagedByExplorerPatcher(hWnd)) return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
HRESULT AltTab_DwmExtendFrameIntoClientArea(HWND hWnd, const MARGINS* pMarInset) {
|
|
HRESULT hr = DwmExtendFrameIntoClientArea(hWnd, pMarInset);
|
|
sws_WindowHelpers_SetMicaMaterialForThisWindow(hWnd, TRUE);
|
|
return hr;
|
|
}
|
|
|
|
BOOL AltTab_PostMessageW(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) {
|
|
if (hWnd == FindWindowW(L"Shell_TrayWnd", NULL) && uMsg == 0x5B7 && wParam == 0 && lParam == 0) {
|
|
return PostMessageW(hWnd, WM_COMMAND, 407, 0);
|
|
}
|
|
return PostMessageW(hWnd, uMsg, wParam, lParam);
|
|
}
|
|
|
|
__declspec(dllexport) void clean() {
|
|
if (pAltTabSSO) pAltTabSSO->lpVtbl->Release(pAltTabSSO);
|
|
if (hAltTab) sws_WindowHelpers_Clear();
|
|
}
|
|
|
|
__declspec(dllexport) int setup() {
|
|
hAltTab = LoadLibraryW(L"AltTab.dll");
|
|
if (hAltTab) {
|
|
sws_WindowHelpers_Initialize();
|
|
VnPatchIAT(hAltTab, "dwmapi.dll", "DwmExtendFrameIntoClientArea", AltTab_DwmExtendFrameIntoClientArea);
|
|
VnPatchIAT(hAltTab, "dwmapi.dll", (LPCSTR)113, AltTab_DwmpActivateLivePreview);
|
|
VnPatchIAT(hAltTab, "user32.dll", "PostMessageW", AltTab_PostMessageW);
|
|
VnPatchIAT(hAltTab, "user32.dll", "LoadStringW", AltTab_LoadStringW);
|
|
VnPatchIAT(hAltTab, "user32.dll", "IsWindowEnabled", AltTab_IsWindowEnabled);
|
|
VnPatchDelayIAT(hAltTab, "uxtheme.dll", "OpenThemeData", AltTab_OpenThemeData);
|
|
VnPatchDelayIAT(hAltTab, "uxtheme.dll", "DrawThemeTextEx", AltTab_DrawThemeTextEx);
|
|
HRESULT(*pDllGetClassObject)(REFCLSID, REFIID, LPVOID) = GetProcAddress(hAltTab, "DllGetClassObject");
|
|
IClassFactory* pFactory = NULL;
|
|
if (pDllGetClassObject && SUCCEEDED(pDllGetClassObject(&CLSID_AltTabSSO, &IID_IClassFactory, &pFactory)) && pFactory) {
|
|
if (SUCCEEDED(pFactory->lpVtbl->CreateInstance(pFactory, NULL, &IID_IOleCommandTarget, &pAltTabSSO)) && pAltTabSSO) {
|
|
if (SUCCEEDED(pAltTabSSO->lpVtbl->Exec(pAltTabSSO, &CGID_ShellServiceObject, 2, 0, NULL, NULL))) {
|
|
printf(">>> Using Windows 7 AltTab\n");
|
|
}
|
|
}
|
|
pFactory->lpVtbl->Release(pFactory);
|
|
}
|
|
FreeLibrary(hAltTab);
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
BOOL WINAPI DllMain(
|
|
_In_ HINSTANCE hinstDLL,
|
|
_In_ DWORD fdwReason,
|
|
_In_ LPVOID lpvReserved
|
|
) {
|
|
switch (fdwReason)
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
DisableThreadLibraryCalls(hinstDLL);
|
|
hModule = hinstDLL;
|
|
break;
|
|
case DLL_THREAD_ATTACH:
|
|
break;
|
|
case DLL_THREAD_DETACH:
|
|
break;
|
|
case DLL_PROCESS_DETACH:
|
|
break;
|
|
}
|
|
return TRUE;
|
|
} |