#ifndef __MULTIMON_H #define __MULTIMON_H #ifdef _WIN32 //============================================================================= // // MULTIMON // stub module that "stubs" multiple monitor APIs on pre-Memphis Win32 OSes // // By using this header your code will work unchanged on Win95, // you will get back correct values from GetSystemMetrics() for new metrics // and the new APIs will act like only one display is present. // // exactly one source must include this with COMPILE_MULTIMON_STUBS defined // //============================================================================= #ifdef __cplusplus extern "C" { /* Assume C declarations for C++ */ #endif /* __cplusplus */ // // if we are building on Win95/NT4 headers we need to declare this stuff ourselves // #ifndef SM_CMONITORS #define SM_XVIRTUALSCREEN 76 #define SM_YVIRTUALSCREEN 77 #define SM_CXVIRTUALSCREEN 78 #define SM_CYVIRTUALSCREEN 79 #define SM_CMONITORS 80 #define SM_SAMEDISPLAYFORMAT 81 DECLARE_HANDLE(HMONITOR); #define MONITOR_DEFAULTTONULL 0x00000000 #define MONITOR_DEFAULTTOPRIMARY 0x00000001 #define MONITOR_DEFAULTTONEAREST 0x00000002 #define MONITORINFOF_PRIMARY 0x00000001 typedef struct tagMONITORINFO { DWORD cbSize; RECT rcMonitor; RECT rcWork; DWORD dwFlags; } MONITORINFO, *LPMONITORINFO; #define CCHDEVICENAME 32 #ifdef __cplusplus typedef struct tagMONITORINFOEX : public tagMONITORINFO { TCHAR szDevice[CCHDEVICENAME]; } MONITORINFOEX, *LPMONITORINFOEX; #else typedef struct { MONITORINFO; TCHAR szDevice[CCHDEVICENAME]; } MONITORINFOEX, *LPMONITORINFOEX; #endif typedef BOOL (CALLBACK* MONITORENUMPROC)(HMONITOR, HDC, LPRECT, LPARAM); #endif // SM_CMONITORS #ifndef DISPLAY_DEVICE_ATTACHED_TO_DESKTOP typedef struct { DWORD cb; CHAR DeviceName[32]; CHAR DeviceString[128]; DWORD StateFlags; } DISPLAY_DEVICE; #define DISPLAY_DEVICE_ATTACHED_TO_DESKTOP 0x00000001 #define DISPLAY_DEVICE_MULTI_DRIVER 0x00000002 #define DISPLAY_DEVICE_PRIMARY_DEVICE 0x00000004 #define DISPLAY_DEVICE_MIRRORING_DRIVER 0x00000008 #endif #define DISPLAY_DEVICE_VGA 0x00000010 #ifndef ENUM_CURRENT_SETTINGS #define ENUM_CURRENT_SETTINGS ((DWORD)-1) #define ENUM_REGISTRY_SETTINGS ((DWORD)-2) #endif #undef GetMonitorInfo #undef GetSystemMetrics #undef MonitorFromWindow #undef MonitorFromRect #undef MonitorFromPoint #undef EnumDisplayMonitors #undef EnumDisplayDevices // // define this to compile the stubs // otherwise you get the declarations // #ifdef COMPILE_MULTIMON_STUBS //--------------------------------------------------------------------------- // // Implement the API stubs. // //--------------------------------------------------------------------------- int (WINAPI* g_pfnGetSystemMetrics)(int); HMONITOR (WINAPI* g_pfnMonitorFromWindow)(HWND, BOOL); HMONITOR (WINAPI* g_pfnMonitorFromRect)(LPCRECT, BOOL); HMONITOR (WINAPI* g_pfnMonitorFromPoint)(POINT, BOOL); BOOL (WINAPI* g_pfnGetMonitorInfo)(HMONITOR, LPMONITORINFO); BOOL (WINAPI* g_pfnEnumDisplayMonitors)(HDC, LPCRECT, MONITORENUMPROC, LPARAM); BOOL (WINAPI *g_pfnEnumDisplayDevices)(LPVOID, int, DISPLAY_DEVICE *, DWORD); BOOL InitMultipleMonitorStubs(void) { HMODULE hUser32; static BOOL fInitDone; if (fInitDone) { return g_pfnGetMonitorInfo != NULL; } if ((hUser32 = GetModuleHandle(TEXT("USER32"))) && (*(FARPROC*)&g_pfnGetSystemMetrics = GetProcAddress(hUser32,"GetSystemMetrics")) && (*(FARPROC*)&g_pfnMonitorFromWindow = GetProcAddress(hUser32,"MonitorFromWindow")) && (*(FARPROC*)&g_pfnMonitorFromRect = GetProcAddress(hUser32,"MonitorFromRect")) && (*(FARPROC*)&g_pfnMonitorFromPoint = GetProcAddress(hUser32,"MonitorFromPoint")) && (*(FARPROC*)&g_pfnEnumDisplayMonitors = GetProcAddress(hUser32,"EnumDisplayMonitors")) && #ifdef UNICODE (*(FARPROC*)&g_pfnGetMonitorInfo = GetProcAddress(hUser32,"GetMonitorInfoW")) && (*(FARPROC*)&g_pfnEnumDisplayDevices = GetProcAddress(hUser32,"EnumDisplayDevicesW")) && #else (*(FARPROC*)&g_pfnGetMonitorInfo = GetProcAddress(hUser32,"GetMonitorInfoA")) && (*(FARPROC*)&g_pfnEnumDisplayDevices = GetProcAddress(hUser32,"EnumDisplayDevicesA")) && #endif (GetSystemMetrics(SM_CXVIRTUALSCREEN) >= GetSystemMetrics(SM_CXSCREEN)) && (GetSystemMetrics(SM_CYVIRTUALSCREEN) >= GetSystemMetrics(SM_CYSCREEN)) ) { fInitDone = TRUE; return TRUE; } else { g_pfnGetSystemMetrics = NULL; g_pfnMonitorFromWindow = NULL; g_pfnMonitorFromRect = NULL; g_pfnMonitorFromPoint = NULL; g_pfnGetMonitorInfo = NULL; g_pfnEnumDisplayMonitors = NULL; g_pfnEnumDisplayDevices = NULL; fInitDone = TRUE; return FALSE; } } //--------------------------------------------------------------------------- // // "stubbed" implementations of Monitor APIs that work with the primary // display // //--------------------------------------------------------------------------- int WINAPI xGetSystemMetrics(int nIndex) { if (InitMultipleMonitorStubs()) return g_pfnGetSystemMetrics(nIndex); switch (nIndex) { case SM_CMONITORS: case SM_SAMEDISPLAYFORMAT: return 1; case SM_XVIRTUALSCREEN: case SM_YVIRTUALSCREEN: return 0; case SM_CXVIRTUALSCREEN: nIndex = SM_CXSCREEN; break; case SM_CYVIRTUALSCREEN: nIndex = SM_CYSCREEN; break; } return GetSystemMetrics(nIndex); } #define xPRIMARY_MONITOR ((HMONITOR)0x42) HMONITOR WINAPI xMonitorFromRect(LPCRECT lprcScreenCoords, UINT uFlags) { if (InitMultipleMonitorStubs()) return g_pfnMonitorFromRect(lprcScreenCoords, uFlags); if ((uFlags & (MONITOR_DEFAULTTOPRIMARY | MONITOR_DEFAULTTONEAREST)) || ((lprcScreenCoords->right > 0) && (lprcScreenCoords->bottom > 0) && (lprcScreenCoords->left < GetSystemMetrics(SM_CXSCREEN)) && (lprcScreenCoords->top < GetSystemMetrics(SM_CYSCREEN)))) { return xPRIMARY_MONITOR; } return NULL; } HMONITOR WINAPI xMonitorFromWindow(HWND hWnd, UINT uFlags) { RECT rc; if (InitMultipleMonitorStubs()) return g_pfnMonitorFromWindow(hWnd, uFlags); if (uFlags & (MONITOR_DEFAULTTOPRIMARY | MONITOR_DEFAULTTONEAREST)) return xPRIMARY_MONITOR; if (GetWindowRect(hWnd, &rc)) return xMonitorFromRect(&rc, uFlags); return NULL; } HMONITOR WINAPI xMonitorFromPoint(POINT ptScreenCoords, UINT uFlags) { if (InitMultipleMonitorStubs()) return g_pfnMonitorFromPoint(ptScreenCoords, uFlags); if ((uFlags & (MONITOR_DEFAULTTOPRIMARY | MONITOR_DEFAULTTONEAREST)) || ((ptScreenCoords.x >= 0) && (ptScreenCoords.x < GetSystemMetrics(SM_CXSCREEN)) && (ptScreenCoords.y >= 0) && (ptScreenCoords.y < GetSystemMetrics(SM_CYSCREEN)))) { return xPRIMARY_MONITOR; } return NULL; } BOOL WINAPI xGetMonitorInfo(HMONITOR hMonitor, LPMONITORINFO lpMonitorInfo) { RECT rcWork; if (InitMultipleMonitorStubs()) return g_pfnGetMonitorInfo(hMonitor, lpMonitorInfo); if ((hMonitor == xPRIMARY_MONITOR) && lpMonitorInfo && (lpMonitorInfo->cbSize >= sizeof(MONITORINFO)) && SystemParametersInfo(SPI_GETWORKAREA, 0, &rcWork, 0)) { lpMonitorInfo->rcMonitor.left = 0; lpMonitorInfo->rcMonitor.top = 0; lpMonitorInfo->rcMonitor.right = GetSystemMetrics(SM_CXSCREEN); lpMonitorInfo->rcMonitor.bottom = GetSystemMetrics(SM_CYSCREEN); lpMonitorInfo->rcWork = rcWork; lpMonitorInfo->dwFlags = MONITORINFOF_PRIMARY; if (lpMonitorInfo->cbSize >= sizeof(MONITORINFOEX)) lstrcpy(((MONITORINFOEX*)lpMonitorInfo)->szDevice, TEXT("DISPLAY")); return TRUE; } return FALSE; } BOOL WINAPI xEnumDisplayMonitors(HDC hdc, LPCRECT lprcIntersect, MONITORENUMPROC lpfnEnumProc, LPARAM lData) { RECT rcCallback, rcLimit; if (InitMultipleMonitorStubs()) return g_pfnEnumDisplayMonitors(hdc, lprcIntersect, lpfnEnumProc, lData); if (!lpfnEnumProc) return FALSE; rcLimit.left = 0; rcLimit.top = 0; rcLimit.right = GetSystemMetrics(SM_CXSCREEN); rcLimit.bottom = GetSystemMetrics(SM_CYSCREEN); if (hdc) { RECT rcClip; HWND hWnd; if ((hWnd = WindowFromDC(hdc)) == NULL) return FALSE; switch (GetClipBox(hdc, &rcClip)) { default: MapWindowPoints(NULL, hWnd, (LPPOINT)&rcLimit, 2); if (IntersectRect(&rcCallback, &rcClip, &rcLimit)) break; //fall thru case NULLREGION: return TRUE; case ERROR: return FALSE; } rcLimit = rcCallback; } if (!lprcIntersect || IntersectRect(&rcCallback, lprcIntersect, &rcLimit)) { lpfnEnumProc(xPRIMARY_MONITOR, hdc, &rcCallback, lData); } return TRUE; } BOOL WINAPI xEnumDisplayDevices(LPVOID lpReserved, int iDeviceNum, DISPLAY_DEVICE * pDisplayDevice, DWORD dwFlags) { if (InitMultipleMonitorStubs()) return g_pfnEnumDisplayDevices(lpReserved, iDeviceNum, pDisplayDevice, dwFlags); return FALSE; } #undef xPRIMARY_MONITOR #undef COMPILE_MULTIMON_STUBS #else // COMPILE_MULTIMON_STUBS extern int WINAPI xGetSystemMetrics(int); extern HMONITOR WINAPI xMonitorFromWindow(HWND, UINT); extern HMONITOR WINAPI xMonitorFromRect(LPCRECT, UINT); extern HMONITOR WINAPI xMonitorFromPoint(POINT, UINT); extern BOOL WINAPI xGetMonitorInfo(HMONITOR, LPMONITORINFO); extern BOOL WINAPI xEnumDisplayMonitors(HDC, LPCRECT, MONITORENUMPROC, LPARAM); extern BOOL WINAPI xEnumDisplayDevices(LPVOID, int, DISPLAY_DEVICE *, DWORD); #endif // COMPILE_MULTIMON_STUBS // // build defines that replace the regular APIs with our versions // #define GetSystemMetrics xGetSystemMetrics #define MonitorFromWindow xMonitorFromWindow #define MonitorFromRect xMonitorFromRect #define MonitorFromPoint xMonitorFromPoint #define GetMonitorInfo xGetMonitorInfo #define EnumDisplayMonitors xEnumDisplayMonitors #define EnumDisplayDevices xEnumDisplayDevices #ifdef __cplusplus } #endif // __cplusplus #endif // WIN32 #endif // __MULTIMON_H