Creating Dialog Based Applications with MFC 7
By Jason Pursell, University of Washington, Bothell (2000-2003)
jpursell@u.washington.edu (Used by permission obtained 3/17/04 and adapted by
Vicki H. Allan)
Introduction
Windows programming involves using a graphical user interface with buttons, text areas,
radio buttons, etc. Thus, instead of having your input come from a file and your output go to
a file or to the screen, you can have various kinds of graphical input and output. In a Windows
program, the control is very different. Instead of having a main program which controls the
actions, the control is given to the user. Thus, whenever a button is pressed or a value
entered, the program is expected to respond appropriately. This style of control is termed
event driven, and can be very different to manage.
Visual C++ helps us to arrange the graphical resources (buttons, text areas, etc.) on the screen
and also aids us in mapping user actions (pressing a button, changing a text area) into
functions that the program performs. Thus, if we wanted to create a button which caused a
specific action, we would define a method (associated with the button) such as OnButtonPress
in which we would specify what to do when the button was pressed. Assume this action
causes something to be printed in a text area. In addition, we would need to know what we
should do to change the value that the program displays in the text area. This is done by
associating the text area with a variable of some type. Then, we change the variable and ask
that the system update the associated text area on the string.
In Windows programming, we are dealing with old style C, so some of the data types are
limited. In this tutorial, we are also letting Visual C++ control everything. We even add
global variables using button clicks. This is not necessary, but is an interesting way to
program. Note how the integrated environment tries to save you work.
This tutorial demonstrates how to create a simple dialog based application with MFC 7
and Visual Studio .NET.
Note: This tutorial is based on another one from CodeProject.com, A Beginners Guide
to Dialog Based Applications Part One, by Dr. Asad Altimeemy
The Visual Studio .NET IDE
While the exact view of the IDE may vary a bit, the parts are all the same. The most
frustrating thing for me was not to have the options or views indicated in the
handout. The ways of manipulating the view are shown in bold.
This section gives an overview of the various windows. The actual instructions
for building a project are in the next section.
3,4,5
Figure 1. Some parts of the IDE that are new and/or improved.
Solution Explorer
IDE Component
1. Toolbox
Class View
Resource View
Description
Displays a variety of items for use in Visual Studio
projects such as controls, components, and code/text
fragments.
2. Dialog Editor (MFC)
This is where you create or edit dialog box resources.
Beware this is not Windows Forms!
3. Solution Explorer
Provides you with an organized view of your projects
and their files as well as ready access to the commands
that pertain to them.
4. Class View
5. Resource View
6. Properties Window
Displays a programmatic view of symbols in your code,
for example, namespaces, classes, methods, and
functions. Variables you have added are visible at the
bottom of this view under CProjectNameDlg
Displays the resource files included in your projects and
allows you to get back to the Dialog Editor by
double clicking on IDD_ProjectName_DIALOG
under the Dialog label.
Use this window to view and change the design-time
properties and events of selected objects that are located
in editors and designers.
Table 1. IDE components.
The location of the each of the views (or windows) is totally dependent upon the profile
you are using. The profile can be set from the Start Page under My Profile. The Start
Page usually shows when you start Visual Studio .NET. I like the profile Student Window
Layout.
The Solution Explorer, Class View, and Resource View [3,4,5] are displayed in the same
space. Switching between the views is done by clicking on the tabs below them
(as shown in the picture). If the Resource View [5] is not present in the tabs,
you can also get these views to appear by clicking on icons that are part of your
IDE. To see which views are associated with each icon, hold the mouse over
each icon until the tool tips display. The diagram below shows the icons
associated with each view.
The properties window is shown in (Figure 2). The first two on the left just determine
how the items are displayed in the list. The next three determine what is displayed:
Properties, Control Events, Messages, or Overrides (in that order). Control Events,
Messages, and Overrides are only available when certain things are selected. For
example, Figure 2 is what is available when CDialog1Dlg is selected in the Class View
(as in the picture of the Class View at the top of the page); however, the Overrides icon
is only shown when a class is selected.
Categorize
Properties
Show Overrides
Alphabetize
Properties
Show Messages
Show Control
Events
Show Properties
Figure 2. Buttons of the properties window.
Show properties by group
Creating a New Project (and Solution)
To create a new project, choose FileNewProject on the main menu. The New
Select Visual C++ Projects and MFC Application.
Show properties alphabetically
Project dialog will open.
Figure 3. New Project dialog.
Write Dialog1 in the project Name field. Press OK.
This will create files shown in Figure 4.
.sln Solution File
.ncb Intellisense Database
.sou Solution User Options
Figure 4. Files created inside the Dialog1 solution folder when the Create directory for Solution checkbox is
selected.
There is also a file ReadMe.txt which is created in your directory. It explains all of the
files that are created by Visual C++.
On the menu, choose Application Type (circled in red in Figure 5).
Figure 5. Page one of the wizard.
Figure 6. Application Type page of the wizard.
Choose the options that are circled in red in Figure 6.Click the Finish button.
Use the ClassView tab to see the actual files that have been created. The file Dialog1Dlg.cpp
references the CAboutDlg, CDialog1App and the CDialog1Dlg class declarations.
The visual elements of a project are called resources. For example, icons, dialog boxes,
buttons, and menus are resources. The wizard has created folders of resources. Click on the
Resource tab to see the folders.
Designing the Dialog
Note: If the Dialog Editor is not showing, open the Resource View and expand
the Dialogs folder and double-click IDD_DIALOG1_DIALOG.
In this example, we are going to create a window like that shown in Figure 10 which will help
us input names.
1.
Click the title bar to
select the Dialog itself.
HINT:
Investigate the blue
line after the tutorial.
Try to figure out what
it is used for. It can
be useful.
2.
Selection Handle.
This can be dragged
to change the size of
the dialog. The
width and height are
indicated in are
below circled in blue.
Figure 7. The Dialog Editor in action.
Click on the text TODO: Place dialog controls here, then press Delete.
Next, as in Figure 7 above, select the dialog by clicking on the edge of the title bar. The
dialog will now have sizing handles on its sides and corners. These are used to resize the
dialog. The width and height of the dialog is indicated at the bottom-right of the IDE and
is circled in blue in the diagram above. These size and widths also show the size of
controls you will add later.
Since we dont need the Cancel button in our application, select it by left-clicking it once.
The button is selected when sizing handles surround it. Press Delete.
Next we need to change some of the properties of the OK button.
Select the OK button by left-clicking it once. Figure 8 shows how the Properties window
reflects the properties of the button when it is selected.
Note: Depending on your choice of IDE setup, the properties window might be on the left
or right. If it is not showing, then choose ViewProperties Window or press F4 on
the keyboard.
Figure 8. Properties of the OK button.
Change the Caption of the button by typing Close in the value field of the Caption
property (as shown in Figure 9).
Type in here!
Figure 9. Changing the Caption property of the button.
Now the button on the dialog reads Close.
Next we will add some additional components (termed controls) to create a GUI as shown
if Figure 10. We want to create a way for the user to add names to a list of names.
The user will specify a persons title, first name, and last name. Each of these items will
appear in a different box.
Drag the icon for each component onto the dialog window. This is done by clicking on the
icon for a button and holding the mouse down while moving the mouse. Now resize the
component. You will resize the button by placing the cursor over the corner of the component
until it changes into a two headed diagonal arrow. Hold the cursor down while sliding the
mouse. The component should change size.
Hint: When a control on the form is selected, you can nudge it by pressing one of the
arrow keys on the keyboard. This provides finer adjustments of the controls placement
than is possible by using the mouse. Likewise, you can grow or shrink your control by
holding down the Shift key and then pressing one of the arrow keys. This is an
alternative to using the resizing handles. Try these techniques as you create this dialog.
Use Table 2 as a guide to add controls to the dialog. (While there is nothing magic
about the names we have used, the rest of the tutorial will be much easier to use if you
type them in exactly as shown.) As you add each control to the dialog, change its
properties to the values shown in the table. We want to build a dialog similar to Figure
10.
Please see Figure 11 for an example of the property settings for the combo-box.
Control
Static
Static
Static
Button
Edit
Edit
Combo Box
List Box
Property
Caption
Caption
Caption
ID
Caption
ID
ID
ID
Data
Type
ID
Value
Title
First Name
Last Name
IDC_ADD
Add
IDC_FIRSTNAME
IDC_LASTNAME
IDC_TITLE
Mr.;Mrs.;Miss;Ms.;Dr.
Drop List
IDC_NAMELIST
Table 2. The controls and their properties that we want to change.
Figure 10. Example layout of the Dialog1 dialog.
Here is an example of the properties of the combo-box control that we added. The
properties we will change are highlighted in red.
Notice, these values are not visible in
the dialog editor, but only show up
when you execute.
Figure 11. The properties of the IDC_TITLE combo box control.
Next, we need to specify the length of the combo-box drop down list. Position the mouse
over the drop-down arrow of the combo-box and left-click once. Use the blue resizing
handle to adjust the length, similar to Figure 12. Click the drop down arrow on the
combo-box again to toggle back to the normal control resizing mode.
If you need a multi-line edit box, you need to make sure that you request that your edit box be
treated as multi-line. Resize the edit box to be considerably larger. Under the Properties
window select Multiline, Horizontal Scroll, Auto HScroll, Vertical scroll, Auto VScroll, and
Want return. When you write to this edit box, instead of just using \n you must use \r\n. If
you only use \n or forget to specify the edit box as multi-line, you will get a strange symbol
instead of a newline on your screen.
1. Left-click to toggle the
sizing mode.
2. Adjusts length of selected drop box
Figure 12. Specifying the length of the open Combo-box list.
Assigning Member Variables to Controls
Lets add some variables to tie the GUI to the data. Make sure that the Dialog Editor is
open and active. In order to access the Add Member Variable Wizard, choose
ProjectAdd Variable from the main menu. You can also right click anywhere in the
Dialog Editor and choose Add Variable from the pop-up menu. This is fine for regular
old variables, but for variables that represent controls, you should right-click on the
control. This way the Add Member Variable Wizard will start, but it will load up with the
fields already filled with the default settings for that control.
Figure 13. Right-click menu.
We will work with the First Name Edit control first.
In the Dialog Editor, right-click the First Name Edit control. From the pop-up menu,
choose Add Variable.
As in Figure 14, make sure the following settings are present before pressing the
Finish button: Control variable checked, Control ID to IDC_FIRSTNAME, Access to
private, Category to Value, Variable type to CString, Control type to EDIT, and
Variable name to m_strFirstName. A word of explanation on category: Value means
that it will handle data values, while Control means that you can control some feature of
the component itself like disabling it or setting the size.
Change this
one first
Figure 14. The Add Member Variable Wizard for the First Name edit control.
Use table 3 as a guide to enter the other variables. If the system wont let you enter
the values you want, try changing another field first.
Remember: Right-click on the control to add a variable. There are other ways, but
this is the easiest for you, because it pre-fills some of the fields.
Note: IDC_NAMELIST will have two variables associated with it. One is a CString and
the other is a CListbox. A ListBox is just a collection of items that the user can
select from (using the mouse).
Control ID
Access
Variable
type
Variable Name
Category
Control
type
Control
variable
IDC_LASTNAME
private
CString
m_strLastName
Value
EDIT
yes
IDC_TITLE
private
CString
m_strTitle
Value
COMBOBOX
yes
IDC_NAMELIST
private
CString
m_strFullName
Value
LISTBOX
Yes
IDC_NAMELIST
Private
CListbox
m_NameList
Control
LISTBOX
yes
Table 3. The remaining variables that need to be added via the Add Variable Wizard
As you add various controls, you may find yourself inside the code view, rather than at the
Dialog Editor view. To get back to the Dialog Editor view, select the tab labeled Dialog1.rc
(as circled below).
If you create the variables incorrectly, one way to fix them is to just modify the code that the
system created for you. (In other words, in Dialog1Dlg.cpp and Dialog1Dlg.h, find the
declarations/uses and change them.)
CString is an old style string. You can convert between strings and CStrings easily. You can
just assign a string to a CString via assignment, but to create a CString from a string, you
must use the converstion method c_str(). Thus, if s is a string and cs is a CString, cs =
s.c_str() is legal.
Adding Message Handlers for the Controls
An event handler is a way to associate a function call with an event such as clicking on a
button. There is more than one way to add an event handler. I will show a couple of
ways here. Each has pros and cons.
Note: Read over both methods first and then decide which one you want to use.
Method One
Right-click on the control that you are interested in (the Add button in this case), then
choose Add Event Handler from the pop-up menu (see Figure 15). This will bring up the
Event Handler Wizard. There you can fill in the information.
Click the Add and Edit button when done. This will take you to the newly added method
where you can begin adding your own code.
Figure 15. Accessing the Event Handler Wizard from right-clicking on controls.
Method Two
The second method is done through the Properties window and is quite convenient.
First, left-click on a control to select it. The Properties window will change to represent
the control. Click the Control Events button (using the lightning icon at the top). Next,
click on the BN_CLICKED control event. If you click the down arrow, you will see
Figure 16. We want to use our own name, so instead, just type in the field as in Figure
17.
As soon as you press Enter, the main window will change to code view and the cursor
will be positioned in the newly created method waiting for you to add some of your own
code.
The code that you will add is on the next page.
Dont choose this
unless you want the
function to have the
default name.
Figure 16. The default event handler function name provided by MFC.
We want to use our
own name, so just
left-click once in here
and type: OnAdd
Then press Enter.
Figure 17. Accessing the Add Event Handler through the Properties window.
Now that youve read through both methods, follow one to add an event handler for the
Add button.
Add the following code to the OnAdd function:
void CDialog1Dlg::OnAdd()
{
// TODO: Add your control notification handler code here
CString strTitle;
int
nIndex;
UpdateData();
// DDX - Transfer data from the controls to variables
//get the currently selected text
nIndex = GetDlgItemText(IDC_TITLE, strTitle);
m_strFullName = strTitle + " " + m_strFirstName + " " + m_strLastName;
m_NameList.AddString(m_strFullName);
UpdateData(FALSE);
}
// DDX - Transfer data from variables to controls
Building and Running the Program
To run, just press CTRL+F5 or DebugStart Without Debugging. Choose Yes when
asked to rebuild the files.
Hint Using Start Without Debugging is much faster than Starting (with debugging).
This is especially true when compiling C# or VB.NET programs in Visual Studio .NET.
The debugger is very powerful, though, so when you need it, just choose Start (or
F5).
Figure 18. Debug menu.
If everything was done correctly, you should be able to type values into the fields and
press the Add button to copy the values into the Name list.
The Widows XP theme
settings are in effect,
because MFC 7
automatically creates a
manifest file as part of
the resources.
(This can be disabled)
It is only available on
Windows XP or better.
Figure 19. Example application.
Combining with other classes
You can also add additional classes for the program to use. To do this, create a .h and a .cpp
file as you normally would and add them to the project. One additional step is to add the line:
#include "stdafx.h"
at the top of your .cpp file. This will include the stdafx.h file which contains the inclusion of
standard header files that are used frequently but not often changed. Now you can use this
class in the project as you normally would.
Resources
For more help on getting started with Visual Studio .NET and/or MFC 7 please see
following:
- www.msdn.microsoft.com
- www.msdn.microsoft.com/library/default.asp?url=/vs/techinfo/Default.asp
- www.codeproject.com
- www.gotdotnet.com
The MSDN library that comes with VS .NET also has gigabytes of help on how to use the
IDE and MFC 7. It also contains whitepapers on compatibility issues between MFC 6 and
7 and general standards conformance with Standard C++ (which is much improved).