per*_*age 3 c++ mfc highdpi visual-studio
我正在使用 CToolbar 开发一个旧的 C++ MFC 应用程序,用于使用 Visual Studio 2017 编译其工具栏。
当应用程序在清单设置(项目/属性/配置属性/清单工具/输入和输出/DPI 感知)中按预期在运行时设置为“高 DPI 感知”时,工具栏似乎随着主监视器 DPI 缩放因子的开启而缩放Windows 10 大约是菜单栏文本的一半高。
在 Windows 7 上,这种缩放似乎不会发生。工具栏在 100% DPI 时与菜单栏文本的高度大致相同,并且在不同的 DPI 设置下保持相同的像素高度。使用或不使用 XP 风格的 DPI 缩放没有区别。
Windows 7 上的底层工具栏公共控件是否根本无法实现高 DPI 缩放?有没有一种有效的方法来调整这一点并使 Windows 7 和 Windows 10 工具栏的大小更加一致?
我可以在运行使用 Visual Studio 2017 的清单中使用高 DPI 感知重新编译的 DOCKTOOL MFC 示例时重现相同的问题。
https://github.com/Microsoft/VCSamples/tree/master/VC2010Samples/MFC/general/docktool
// code excerpts from DOCKTOOL below:
class CMainFrame : public CFrameWnd
{
protected: // create from serialization only
CMainFrame();
DECLARE_DYNCREATE(CMainFrame)
CToolBar m_wndMainBar;
// Generated message map functions
protected:
//{{AFX_MSG(CMainFrame)
afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);
//}}AFX_MSG
DECLARE_MESSAGE_MAP()
};
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
return -1;
//....
if (!m_wndMainBar.Create(this, WS_CHILD | WS_VISIBLE | CBRS_SIZE_DYNAMIC |
CBRS_TOP | ((m_bToolTips)?(CBRS_TOOLTIPS | CBRS_FLYBY):0), IDW_MAIN_BAR) ||
!m_wndMainBar.LoadBitmap(
(m_bColor)?IDR_COLOR_MAINBAR:IDR_MONO_MAINBAR) ||
!m_wndMainBar.SetButtons(MainButtons, sizeof(MainButtons)/sizeof(UINT)))
{
TRACE0("Failed to create mainbar\n");
return -1; // fail to create
}
Run Code Online (Sandbox Code Playgroud)
使用CToolBar::SetSizes分配一个新的按钮大小和位图的大小。您必须知道资源中使用的位图的尺寸。
下面的示例假定位图为 16 x 15 像素。这将根据 DPI 设置使按钮变大(但不会使位图变大)
//get DPI scaling
double fx = GetSystemMetrics(SM_CXSMICON) / 16.0f;
double fy = GetSystemMetrics(SM_CYSMICON) / 16.0f;
if(fx < 1) fx = 1;
if(fy < 1) fy = 1;
CRect temp;
m_wndMainBar.GetItemRect(0, &temp);
temp.MoveToXY(0, 0);
temp.right = int(temp.right * fx);
temp.bottom = int(temp.bottom * fy);
CSize bmp_size(16, 15);
m_wndMainBar.SetSizes(temp.Size(), bmp_size);
Run Code Online (Sandbox Code Playgroud)
要使位图更大,您必须转到资源编辑器并为每个缩放创建一个单独的位图。
例如,对于 125% DPI 缩放,创建一个大 25% 的位图并使用bmp_size(20, 18)而不是bmp_size(16, 15)
此方法可能会变得非常复杂,因此您可以考虑使用带有 的图标LoadIconWithScaleDown。这样你就可以创建一个大图标,比如 150% DPI,并使用相同的图标进行 125% 和 100% DPI 缩放。