Click here to Skip to main content
Click here to Skip to main content

CResizableFormView

By , 6 Jun 2001
 

Introduction

The inspiration for this coding project was the result of a) Paolo Messina's article on his CResizableDialog code, and b) me mentioning I had adapted a CFormView to use a CResizablePage in an embedded property sheet.

The CResizablePage in a formview was a minor exercise compared to this. What I've come up with is a CResizableFormView. This code is based entirely on Paolo's CResizableDialog code.

If you're not interested in how I did it, Skip to the bottom of this article to see how to implement it yourself.

The Concept

It seemed to me that if a dialog box could benefit from this technique, then so could a formview, because after all, a formview is nothing more than a view that acts like a dialog box. It even uses a resource template like a dialog box.

Initially

The initial coding was fairly straightforward. I created a CFormView-derived class called CResizableFormView. Next, I copied all of the utility functions (functions not contributed by ClassWizard). Then, I evaluated the need/availability of some of the windows messages Paolo's code was using. I ended up only needing to handle the WM_SIZE command. Lastly, I decided that the min/max window size code was not applicable to a formview, so I deleted the code involving that particular aspect of CResizableDialog.

Implementation

After getting the code to compile, I tried deriving a new class based on my new CResizableFormView. After some experimentation, I discovered that I could not use the AddAnchor function until AFTER the InitialUpdate had completed. If I didn't follow this rule, the controls would not resize correctly.
void CFormview2View::OnInitialUpdate()
{
    CResizableFormView::OnInitialUpdate();
    GetParentFrame()->RecalcLayout();
    ResizeParentToFit();

    // put our code AFTER the stuff ClassWizard gave us.
    AddAnchor(IDC_LIST1, TOP_LEFT, BOTTOM_RIGHT);
    AddAnchor(IDC_GROUP1, TOP_LEFT, BOTTOM_LEFT);
}

Minor Issues

After playing around with resizing for a while, I discovered that if I made the view window smaller than the original dialog template, a couple of bad things happened involving the controls that were "anchored".
  • The listbox disappeared altogether and would only partially appear as I made the view horizontally larger.
  • The groupbox around the radio controls would just keep resizing until nothing was left (of the groupbox).

It was obvious at that point that I had to include some code to restrict the controls from resizing once they reached (or were smaller than) their original size. I also assumed that I had to be mindful of getting the correct original size, meaning I had to determine each control's size before the window was allowed to be resized the first time. Assuming that the programmer could decide to do something other than use a call to MFC's ResizeParentToFit, I had to make sure to get the control's info immediately after the view was created, but before it was resized. I wanted to impact as little of Paolo's original code as I could manage, so the fix was a while coming.

To remedy the situation, I created a CTypedPtrArray that held a new structure. This structure contains the original size and location of each "anchored" control, as well as its HWND and control ID. To support the structure, I had to also write a couple of new functions. Here are the details:

void CResizableFormView::AddResizedControls()

This function is responsible for determining the HWND and window size / location of the specified control, as well as storing its anchor positions. The parameter list is identical to the original AddAnchor function. Once all of this information was gathered, the control was added to the new CTypedPtrArray for safe-keeping.

void CResizableFormView::AnchorControls()

This function actually calls AddAnchor for all of the controls the programmer specified when he/she called the AddResizedControls function. The significance of this will be evident later.

Results

In order to implement the sizing/positioning fix, A new order of execution was required in the CResizableFormView-derived class. The OnInitialUpdate function now looks like this:
void CFormview2View::OnInitialUpdate()
{
    // jms - 11/28/00 - new code starts
    // We do this here because we need to get the *original* size of the 
    // controls *before* the dialog is sized the first time.  I don't know 
    // what will happen if we do it after, but I didn't want to find out.
    //
    // Since we do this here, I decided to pass the parameters needed by 
    // the AddAnchor() function to eliminate the need to call AddAnchor 
    // from this function (after the view had been resized).
    AddResizedControl(IDC_LIST1, TOP_LEFT, BOTTOM_RIGHT);
    AddResizedControl(IDC_GROUP1, TOP_LEFT, BOTTOM_LEFT);
    // jms - 11/28/00 - new code stops

    CResizableFormView::OnInitialUpdate();
    GetParentFrame()->RecalcLayout();
    ResizeParentToFit();

    // Normally, we would put the calls to AddAnchor AFTER we call 
    // ResizeParentToFit().
    //AddAnchor(IDC_LIST1, TOP_LEFT, BOTTOM_RIGHT);
    //AddAnchor(IDC_GROUP1, TOP_LEFT, BOTTOM_LEFT);

    // jms - 11/28/00 - new code starts
    // Now anchor the controls we specified earlier in this function. We call 
    // this function because it cycles thru the list of controls that are to 
    // be resized. This prevents us from having to call AddAnchor (lots of 
    // redundant typing at this point).
    AnchorControls();

}

But what if you wanted the view to NOT resize itself according to the dialog template. That's easy. Just calculate the size of the parent window of the formview, and call MoveWindow before exiting OnInitialUpdate. Here's an example:

void CFormview2View::OnInitialUpdate()
{
    AddResizedControl(IDC_LIST1, TOP_LEFT, BOTTOM_RIGHT);
    AddResizedControl(IDC_GROUP1, TOP_LEFT, BOTTOM_LEFT);
    AddResizedControl(IDC_IPADDRESS1, BOTTOM_RIGHT);

    CResizableFormView::OnInitialUpdate();
    GetParentFrame()->RecalcLayout();

    // Here's where the difference lies.  We don't want to call ResizeParentToFit() until
    // AFTER we calculate the true (initial) size of the parent frame. 
    CRect rectFrame;
    CFrameWnd* pFrame = GetParentFrame();
    ASSERT_VALID(pFrame);
    pFrame->GetWindowRect(rectFrame);
    CSize size = rectFrame.Size();

    // Now, we can call ResizeParentToFit() and anchor the controls.
    ResizeParentToFit(FALSE);
    AnchorControls();

    // Finally, we need to actually resize the frame to its original dimensions.  This
    // allows the anchor code to work  with its existing code.
    pFrame->MoveWindow(rectFrame, TRUE);
}

The Final Word

To implement the CResizableFormView, perform the following steps:

  1. Create a CFormView class and its dialog template if necessary.
  2. Change the base class of your CFormView-derived class to CResizableFormView.
  3. #include "ResizableFormView.h" in your derived formview's header file.
  4. If the function isn't already there, use ClassWizard to override the OnInitialUpdate function.
  5. Call AddResizedControls for each control that will be resized/repositioned. Do this BEFORE the call to the base class' OnInitialUpdate.
  6. Call AnchorControls. Do this AFTER the call to ResizeParentToFit

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

John Simmons / outlaw programmer
Software Developer (Senior)
United States United States
Member
I've been paid as a programmer since 1982 with experience in Pascal, and C++ (both self-taught), and began writing Windows programs in 1991 using Visual C++ and MFC. In the 2nd half of 2007, I started writing C# Windows Forms and ASP.Net applications, and have since done WPF, Silverlight, WCF, web services, and Windows services.
 
My weakest point is that my moments of clarity are too brief to hold a meaningful conversation that requires more than 30 seconds to complete. Thankfully, grunts of agreement are all that is required to conduct most discussions without committing to any particular belief system.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
Hint: For improved responsiveness ensure Javascript is enabled and choose 'Normal' from the Layout dropdown and hit 'Update'.
You must Sign In to use this message board.
Search this forum  
    Spacing  Noise  Layout  Per page   
QuestionI want to fit the dialog in form??memberjohnthecoder25 Aug '08 - 19:50 
QuestionMaximized MDI window problemmembersdr27 Jun '06 - 4:10 
GeneralMissing .dsp file to open the project!memberPerryz4 Dec '04 - 17:25 
GeneralRe: Missing .dsp file to open the project!memberbatsword10 Sep '10 - 21:08 
Questionnot resizing?memberlagonz4 May '04 - 16:45 
AnswerRe: not resizing?memberlagonz4 May '04 - 16:52 
GeneralGot an Error!!member*Dreamz16 Apr '04 - 3:24 
GeneralBad Dialog Templatememberniallmack4 Nov '03 - 12:19 
GeneralRe: Bad Dialog Templatemember*Dreamz16 Apr '04 - 3:09 
GeneralBug and Solution - Resizing-problemmemberSiegmund Frenzel7 Aug '03 - 22:05 
GeneralDemo LinkmemberAsha_krishna10 Jul '03 - 13:27 
GeneralDemo download link not workingmemberLobster B15 May '03 - 20:17 
Questionwhere is formview2.dsp file ???memberijt7epn30 Dec '02 - 5:09 
GeneralFit a dialog into a WindowmemberAnonymous17 Apr '02 - 6:47 
GeneralUpdated ArticlememberJohn Simmons / outlaw programmer7 Jun '01 - 1:58 
Questionis ResizeParentToFit() compulsory ????memberwimel16 May '01 - 3:15 
You did a nice job by making this class. It very usefull.Smile | :)
But I don't wanna resize the parent. In my SDI-app with CSplitterWnd I wanna resize the formview to the parent.
 
Is this possible ?
what about GetParentFrame()->RecalcLayout()?
 
Any help is welcome
 

wimel
AnswerRe: is ResizeParentToFit() compulsory ????memberJohn Simmons / outlaw programmer17 May '01 - 2:51 
AnswerRe: is ResizeParentToFit() compulsory ????memberJohn Simmons / outlaw programmer17 May '01 - 3:14 
GeneralRe: is ResizeParentToFit() compulsory ????memberwimel22 May '01 - 21:29 
GeneralRe: is ResizeParentToFit() compulsory ????memberJohn Simmons / outlaw programmer24 May '01 - 4:04 
GeneralRe: is ResizeParentToFit() compulsory ????memberWimel7 Jun '01 - 1:26 
GeneralRe: is ResizeParentToFit() compulsory ????memberJohn Simmons / outlaw programmer7 Jun '01 - 1:44 
GeneralActiveX controls and scrollingmemberShane Warren2 Apr '01 - 10:18 
GeneralFix for disappearing radio buttons/checkboxesmemberMatt Philmon8 Mar '01 - 19:13 
GeneralRe: Fix for disappearing radio buttons/checkboxesmemberJohn Simmons / outlaw programmer15 Mar '01 - 6:22 
QuestionControls on FormView are disabled?membervcprog17 Jan '01 - 17:05 
AnswerRe: Controls on FormView are disabled?memberJohn Simmons / outlaw programmer15 Feb '01 - 4:22 
AnswerRe: Controls on FormView are disabled?susskong(no valid email address)21 Oct '03 - 19:17 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Permalink | Advertise | Privacy | Mobile
Web04 | 2.6.130516.1 | Last Updated 7 Jun 2001
Article Copyright 2000 by John Simmons / outlaw programmer
Everything else Copyright © CodeProject, 1999-2013
Terms of Use
Layout: fixed | fluid