如何在 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 文件中定义。