如何在 MDI MFC C++ windows 应用程序中动态更改 CFormView 宽度或高度?
How to change dynamically CFormView WIDTH or HEIGHT in MDI MFC C++ windows application?
1997年用C++编写过MDI MFC程序
我创建了一个名为 XFormFiew 的 Class 扩展 MFC CFormView class。
在OnInitialUpdate()
事件方法中,我写了一些代码来自动修改视图的缩放。
过去,大多数屏幕分辨率为800x600,但现在分辨率更高。在自动放大 XFormView
时,我避免在所有视图中进行一些修改。
缩放代码修改 Left 和 Bottom,还修改活动 CFormView
中包含的每个 CWnd
的宽度和高度
代码如下
void XFormView::OnInitialUpdate()
{
CFormView::OnInitialUpdate();
pLogFont = new LOGFONT;
CFont* pDialogFont = GetFont();
pDialogFont->GetObject(sizeof(LOGFONT),pLogFont);
pLogFont->lfHeight = MulDiv(pLogFont->lfHeight, Config.GetDstH(), Config.GetSrcH());
pLogFont->lfWeight = FW_NORMAL;
pFont = new CFont;
pFont->CreateFontIndirect(pLogFont);
SetFont(pFont);
CWnd* pWnd;
pWnd = GetWindow(GW_CHILD);
while (pWnd != NULL)
{
ZoomWnd(pWnd);
pWnd = pWnd->GetWindow(GW_HWNDNEXT);
}
// TRY to modify WIDTH and HEIGTH of CFormView
ZoomWnd(this);
}
void XFormView::ZoomWnd(CWnd* pWnd)
{
CRect rect;
pWnd->GetWindowRect(&rect);
ScreenToClient(&rect);
rect.left = (int)((long)rect.left * Config.GetDstH() / Config.GetSrcH());
rect.top = (int)((long)rect.top * Config.GetDstV() / Config.GetSrcV());
rect.right = (int)((long)rect.right * Config.GetDstH() / Config.GetSrcH());
rect.bottom = (int)((long)rect.bottom * Config.GetDstV() / Config.GetSrcV());
pWnd->MoveWindow(&rect);
pWnd->SetFont(pFont);
}
代码运行良好,但对于某些视图,缺少垂直和水平滚动条。
Window 始终加载为最大化。如果我取消最大化 Window,滚动条总是丢失。
如果我减小 WIDTH 或 HEIGHT,当宽度或高度达到 RC 资源文件中定义的值时,将显示滚动条。
示例:
IDD_OVFORM DIALOGEX 0, 0, 370, 297
其中 370 是窗体的宽度,297 是高度。
如果我减小 Child Window 宽度,直到宽度减小到 370 时才会发生任何事情。此时会自动显示一个水平滚动条。
我的问题是如何动态更改(不是在 RC 文件中)IDD_OVFORM 的宽度和高度以对应缩放级别,以便正确显示水平和垂直滚动条?
我已经尝试在 OnInitialUpdate() 方法中更改这些属性,但没有任何效果。
如果我在 RC 资源文件中更改宽度和高度,则缩放对于特定缩放级别(但不是所有缩放级别)都可以正常工作。
在互联网上,我找到了一些关于删除滚动条的解决方案,但没有关于使用 MDI MFC 窗体缩放和丢失滚动条的解决方案。
是否有人已经找到解决这个问题的方法?
2019-02-12:感谢 Barmak Shemirani 提供完美的解决方案(选项 3)。
可能你需要手动为每个滚动条调用EnableScrollBars
,判断是否需要
您可能需要调用 SetScrollInfo
然后设置总大小。
两个调用都需要在 CFormView
的父 window 上完成,应该是 CChildFrame
。
调用 SetScrollSizes
以显示特定大小的滚动条:
CRect rect;
GetClientRect(rect);
//this multiplication is to make sure the scrollbar is visible
//remove it in actual code.
rect.right *= 2;
rect.bottom *= 2;
SetScrollSizes(MM_TEXT, rect.Size());
但这不是一般的正确方法。
选项 1:
您可以转到资源编辑器、select 对话框、select 对话框的属性,然后更改对话框的默认字体大小。将字体大小增加到 9 或更大,这将自动使对话框及其控件变大,并使用更大的字体。
选项 2:
在对话框的属性中,您还会看到名为 "Dynamic Layout" 的部分。这使您可以在调整大小时缩放控件,或将它们移动 up/down 和 left/right.
选项 3:
在 运行 时间内更改对话框模板中的字体。为此,您必须覆盖 CFormView::Create
,后者又会调用 CreateDlg
。这些函数名称必须完全如下声明。
请注意,对话框模板中的字体只能在加载对话框之前更改一次。
示例 Visual Studio 2017(此代码可能与旧版 MFC 不兼容)
BOOL XFormView::CreateDlg(LPCTSTR lpszTemplateName, CWnd* pParentWnd)
{
CDialogTemplate dlt;
if(dlt.Load(lpszTemplateName)))
{
// set your own font
dlt.SetFont(L"Arial", 20);
HINSTANCE hInst = AfxFindResourceHandle(lpszTemplateName, RT_DIALOG);
LPCDLGTEMPLATE dlgtemplate = (LPCDLGTEMPLATE)GlobalLock(dlt.m_hTemplate);
// create a modeless dialog
BOOL bSuccess = CreateDlgIndirect(dlgtemplate, pParentWnd, hInst);
GlobalUnlock(dlt.m_hTemplate);
return bSuccess;
}
return CFormView::CreateDlg(lpszTemplateName, pParentWnd);
}
BOOL XFormView::Create
( LPCTSTR lpszClassName
, LPCTSTR lpszWindowName
, DWORD dwRequestedStyle
, const RECT& rect
, CWnd* pParentWnd
, UINT nID
, CCreateContext* pContext
);
{
ASSERT(pParentWnd != NULL);
ASSERT(m_lpszTemplateName != NULL);
m_pCreateContext = pContext; // save state for later OnCreate
// call PreCreateWindow to get prefered extended style
CREATESTRUCT cs;
memset(&cs, 0, sizeof(CREATESTRUCT));
if(dwRequestedStyle == 0)
dwRequestedStyle = AFX_WS_DEFAULT_VIEW;
cs.style = dwRequestedStyle;
if(!PreCreateWindow(cs))
return FALSE;
// create a modeless dialog
if(!CreateDlg(m_lpszTemplateName, pParentWnd))
return FALSE;
m_pCreateContext = NULL;
ModifyStyle(WS_BORDER | WS_CAPTION, cs.style & (WS_BORDER | WS_CAPTION));
ModifyStyleEx(WS_EX_CLIENTEDGE, cs.dwExStyle & WS_EX_CLIENTEDGE);
SetDlgCtrlID(nID);
CRect rectTemplate;
GetWindowRect(rectTemplate);
SetScrollSizes(MM_TEXT, rectTemplate.Size());
// initialize controls etc
if(!ExecuteDlgInit(m_lpszTemplateName))
return FALSE;
// force the size requested
SetWindowPos(NULL, rect.left, rect.top,
rect.right - rect.left, rect.bottom - rect.top,
SWP_NOZORDER | SWP_NOACTIVATE);
// make visible if requested
if(dwRequestedStyle & WS_VISIBLE)
ShowWindow(SW_NORMAL);
return TRUE;
}
和class XFormView.h 中的定义必须包含以下行
protected:
BOOL Create
( LPCTSTR lpszClassName
, LPCTSTR lpszWindowName
, DWORD dwRequestedStyle
, const RECT& rect
, CWnd* pParentWnd
, UINT nID
, CCreateContext* pContext
);
BOOL CreateDlg(LPCTSTR lpszTemplateName, CWnd* pParentWnd);
afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);
OnCreate
声明是绝对必要的。
没有这个声明,什么也不会发生!!!
OnCreate
仅在包含文件中声明,但未在 CPP 文件中定义。
1997年用C++编写过MDI MFC程序
我创建了一个名为 XFormFiew 的 Class 扩展 MFC CFormView class。
在OnInitialUpdate()
事件方法中,我写了一些代码来自动修改视图的缩放。
过去,大多数屏幕分辨率为800x600,但现在分辨率更高。在自动放大 XFormView
时,我避免在所有视图中进行一些修改。
缩放代码修改 Left 和 Bottom,还修改活动 CFormView
CWnd
的宽度和高度
代码如下
void XFormView::OnInitialUpdate()
{
CFormView::OnInitialUpdate();
pLogFont = new LOGFONT;
CFont* pDialogFont = GetFont();
pDialogFont->GetObject(sizeof(LOGFONT),pLogFont);
pLogFont->lfHeight = MulDiv(pLogFont->lfHeight, Config.GetDstH(), Config.GetSrcH());
pLogFont->lfWeight = FW_NORMAL;
pFont = new CFont;
pFont->CreateFontIndirect(pLogFont);
SetFont(pFont);
CWnd* pWnd;
pWnd = GetWindow(GW_CHILD);
while (pWnd != NULL)
{
ZoomWnd(pWnd);
pWnd = pWnd->GetWindow(GW_HWNDNEXT);
}
// TRY to modify WIDTH and HEIGTH of CFormView
ZoomWnd(this);
}
void XFormView::ZoomWnd(CWnd* pWnd)
{
CRect rect;
pWnd->GetWindowRect(&rect);
ScreenToClient(&rect);
rect.left = (int)((long)rect.left * Config.GetDstH() / Config.GetSrcH());
rect.top = (int)((long)rect.top * Config.GetDstV() / Config.GetSrcV());
rect.right = (int)((long)rect.right * Config.GetDstH() / Config.GetSrcH());
rect.bottom = (int)((long)rect.bottom * Config.GetDstV() / Config.GetSrcV());
pWnd->MoveWindow(&rect);
pWnd->SetFont(pFont);
}
代码运行良好,但对于某些视图,缺少垂直和水平滚动条。
Window 始终加载为最大化。如果我取消最大化 Window,滚动条总是丢失。
如果我减小 WIDTH 或 HEIGHT,当宽度或高度达到 RC 资源文件中定义的值时,将显示滚动条。
示例:
IDD_OVFORM DIALOGEX 0, 0, 370, 297
其中 370 是窗体的宽度,297 是高度。
如果我减小 Child Window 宽度,直到宽度减小到 370 时才会发生任何事情。此时会自动显示一个水平滚动条。
我的问题是如何动态更改(不是在 RC 文件中)IDD_OVFORM 的宽度和高度以对应缩放级别,以便正确显示水平和垂直滚动条?
我已经尝试在 OnInitialUpdate() 方法中更改这些属性,但没有任何效果。
如果我在 RC 资源文件中更改宽度和高度,则缩放对于特定缩放级别(但不是所有缩放级别)都可以正常工作。
在互联网上,我找到了一些关于删除滚动条的解决方案,但没有关于使用 MDI MFC 窗体缩放和丢失滚动条的解决方案。
是否有人已经找到解决这个问题的方法?
2019-02-12:感谢 Barmak Shemirani 提供完美的解决方案(选项 3)。
可能你需要手动为每个滚动条调用EnableScrollBars
,判断是否需要
您可能需要调用 SetScrollInfo
然后设置总大小。
两个调用都需要在 CFormView
的父 window 上完成,应该是 CChildFrame
。
调用 SetScrollSizes
以显示特定大小的滚动条:
CRect rect;
GetClientRect(rect);
//this multiplication is to make sure the scrollbar is visible
//remove it in actual code.
rect.right *= 2;
rect.bottom *= 2;
SetScrollSizes(MM_TEXT, rect.Size());
但这不是一般的正确方法。
选项 1:
您可以转到资源编辑器、select 对话框、select 对话框的属性,然后更改对话框的默认字体大小。将字体大小增加到 9 或更大,这将自动使对话框及其控件变大,并使用更大的字体。
选项 2:
在对话框的属性中,您还会看到名为 "Dynamic Layout" 的部分。这使您可以在调整大小时缩放控件,或将它们移动 up/down 和 left/right.
选项 3:
在 运行 时间内更改对话框模板中的字体。为此,您必须覆盖 CFormView::Create
,后者又会调用 CreateDlg
。这些函数名称必须完全如下声明。
请注意,对话框模板中的字体只能在加载对话框之前更改一次。
示例 Visual Studio 2017(此代码可能与旧版 MFC 不兼容)
BOOL XFormView::CreateDlg(LPCTSTR lpszTemplateName, CWnd* pParentWnd)
{
CDialogTemplate dlt;
if(dlt.Load(lpszTemplateName)))
{
// set your own font
dlt.SetFont(L"Arial", 20);
HINSTANCE hInst = AfxFindResourceHandle(lpszTemplateName, RT_DIALOG);
LPCDLGTEMPLATE dlgtemplate = (LPCDLGTEMPLATE)GlobalLock(dlt.m_hTemplate);
// create a modeless dialog
BOOL bSuccess = CreateDlgIndirect(dlgtemplate, pParentWnd, hInst);
GlobalUnlock(dlt.m_hTemplate);
return bSuccess;
}
return CFormView::CreateDlg(lpszTemplateName, pParentWnd);
}
BOOL XFormView::Create
( LPCTSTR lpszClassName
, LPCTSTR lpszWindowName
, DWORD dwRequestedStyle
, const RECT& rect
, CWnd* pParentWnd
, UINT nID
, CCreateContext* pContext
);
{
ASSERT(pParentWnd != NULL);
ASSERT(m_lpszTemplateName != NULL);
m_pCreateContext = pContext; // save state for later OnCreate
// call PreCreateWindow to get prefered extended style
CREATESTRUCT cs;
memset(&cs, 0, sizeof(CREATESTRUCT));
if(dwRequestedStyle == 0)
dwRequestedStyle = AFX_WS_DEFAULT_VIEW;
cs.style = dwRequestedStyle;
if(!PreCreateWindow(cs))
return FALSE;
// create a modeless dialog
if(!CreateDlg(m_lpszTemplateName, pParentWnd))
return FALSE;
m_pCreateContext = NULL;
ModifyStyle(WS_BORDER | WS_CAPTION, cs.style & (WS_BORDER | WS_CAPTION));
ModifyStyleEx(WS_EX_CLIENTEDGE, cs.dwExStyle & WS_EX_CLIENTEDGE);
SetDlgCtrlID(nID);
CRect rectTemplate;
GetWindowRect(rectTemplate);
SetScrollSizes(MM_TEXT, rectTemplate.Size());
// initialize controls etc
if(!ExecuteDlgInit(m_lpszTemplateName))
return FALSE;
// force the size requested
SetWindowPos(NULL, rect.left, rect.top,
rect.right - rect.left, rect.bottom - rect.top,
SWP_NOZORDER | SWP_NOACTIVATE);
// make visible if requested
if(dwRequestedStyle & WS_VISIBLE)
ShowWindow(SW_NORMAL);
return TRUE;
}
和class XFormView.h 中的定义必须包含以下行
protected:
BOOL Create
( LPCTSTR lpszClassName
, LPCTSTR lpszWindowName
, DWORD dwRequestedStyle
, const RECT& rect
, CWnd* pParentWnd
, UINT nID
, CCreateContext* pContext
);
BOOL CreateDlg(LPCTSTR lpszTemplateName, CWnd* pParentWnd);
afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);
OnCreate
声明是绝对必要的。
没有这个声明,什么也不会发生!!!
OnCreate
仅在包含文件中声明,但未在 CPP 文件中定义。