@EDIT:我发现这似乎是 Windows 8 RC 的问题,因为我尝试使用 Windows 7 和 VS 2012,包括经典视图和 Aero 视图,它工作正常。感谢@Werner Henze 和@Ven Boigt 的反馈
编辑 2:原来这是 Windows 中的一个错误,因为它是测试版,它在较新的版本中得到了修复,所以我不必再担心这个了。无论如何感谢您的反馈。
我曾经执行以下操作来创建一个 800*600 客户区的窗口:
dwStyle = WS_OVERLAPPED | WS_THICKFRAME | WS_BORDER | WS_DLGFRAME | WS_SYSMENU | WS_MINIMIZEBOX;
hWindowHandle = CreateWindow( L"CGFramework", wszWndCaption,
pWindowData->dwStyle, pWindowData->nPositionX, pWindowData->nPositionY,
800 + GetSystemMetrics( SM_CXSIZEFRAME )*2,
600 + GetSystemMetrics( SM_CYSIZEFRAME ) *2
+ GetSystemMetrics( SM_CYCAPTION ),
0, 0, hInstance, 0
);
然后当用GetClientRect查询客户端矩形时,我以前得到800*600,但现在我将我的Visual Studio 2008项目升级到VS2012,现在GetClientRect()函数返回792*592。
最重要的是,正在创建的窗口的实际大小是 804*629,我认为这是没有原因的,因为可调整大小的框架(来自 WS_THICKFRAME)明显大于每边 2 个像素。
我认为这是 Windows 8 的 Aero 行为问题,但后来我意识到这仅发生在我的 VS2012 版本中,而不是我的 VS2008 版本中。如果我在 Aero 或经典风格上运行它没有区别,该行为仅适用于 VS2012 构建。为什么?我可以对我的 VS2012 项目配置进行更改以解决这种可怕的行为吗?
我尝试更改项目配置的“DPI Awareness”设置,但这没有任何区别。我还在同一个配置页面中删除了清单的使用,但在结果窗口中仍然看不到任何变化。
这是我的测试代码:
#include <Windows.h>
#include <cstdio>
LRESULT CALLBACK MainWndProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam )
{
switch( uMsg )
{
// WM_DESTROY is sent when the window is being destroyed.
case WM_DESTROY:
PostQuitMessage(0);
return 0;
default:
return DefWindowProc( hWnd, uMsg, wParam, lParam );
}
}
int WINAPI WinMain( HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, INT nShowCmd )
{
WNDCLASS wc;
wc.style = NULL; // CS_HREDRAW | CS_VREDRAW;
wc.lpfnWndProc = MainWndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hInstance;
wc.hIcon = LoadIcon(0, IDI_APPLICATION);
wc.hCursor = LoadCursor(0, IDC_ARROW);
wc.hbrBackground = CreateSolidBrush(GetSysColor(COLOR_3DFACE));
wc.lpszMenuName = 0;
wc.lpszClassName = L"CGFramework";
DWORD dwStyle = WS_OVERLAPPED | WS_THICKFRAME | WS_BORDER |
WS_DLGFRAME | WS_SYSMENU | WS_MINIMIZEBOX; // WS_DLGFRAME |
if( !RegisterClass(&wc) )
{
MessageBox(0, L"RegisterClass FAILED", 0, 0);
return E_FAIL;
}
RECT r;
//r.left = 100;
//r.top = 100;
//r.right = 800;
//r.bottom = 600;
////-----------------------------
r.left = 100;
r.top = 100;
r.right = 900;
r.bottom = 700;
////-----------------------------
//r.left = 100;
//r.top = 100;
//r.right = 800+GetSystemMetrics( SM_CXFRAME )*2;
//r.bottom = 600+GetSystemMetrics( SM_CYFRAME )*2+GetSystemMetrics( SM_CYCAPTION );
BOOL result = AdjustWindowRect( &r, dwStyle, FALSE );
HWND hWindowHandle = CreateWindow( L"CGFramework", L"testWindow", dwStyle,
r.left, r.top, r.right-r.left, r.bottom-r.top,
// r.left, r.top, r.right, r.bottom,
0, 0, hInstance, 0 );
if( 0 == hWindowHandle )
{
MessageBox(0, L"CreateWindow FAILED", 0, 0);
UnregisterClass( wc.lpszClassName, hInstance );
return 0;
}
char buffer[512]; // for outing test message
GetClientRect( hWindowHandle, &r );
sprintf( &buffer[0], "left=%i, top=%i, right=%i, bottom=%i", r.left, r.top, r.right, r.bottom );
MessageBoxA(0, buffer, 0, 0); // print rect values before ShowWindow
ShowWindow( hWindowHandle, SW_SHOW );
GetClientRect( hWindowHandle, &r );
sprintf( &buffer[0], "left=%i, top=%i, right=%i, bottom=%i", r.left, r.top, r.right, r.bottom );
MessageBoxA(0, buffer, 0, 0); // print rect values after ShowWindow
// main window loop
MSG msg;
ZeroMemory( &msg, sizeof( MSG ) );
while( msg.message != WM_QUIT )
{
while( PeekMessage( &msg, 0, 0, 0, PM_REMOVE ) )
{
TranslateMessage( &msg );
DispatchMessage( &msg );
}
if( GetAsyncKeyState( VK_ESCAPE ) )
DestroyWindow( hWindowHandle );
}
UnregisterClass( wc.lpszClassName, hInstance );
return 0;
}
请在回答之前仔细查看代码,因为您最终可能会回答我已经尝试过的问题(我发布这个问题是因为我已经浪费了所有的选择)。
我正在使用 Windows 8 RC(MS windows 6.2.8400)和 VS2012 RC(11.0.50706.0 QRELRC 2012 年 7 月)来获得这种奇怪的行为,并且没有任何一个好的答案能够解决这个问题。在做出任何假设之前,请务必阅读它们并测试我的代码,因为这段代码已经在许多方面进行了测试,但存在细微的差异,最终没有提供任何改进。